位置: IT常识 - 正文

推荐一个跨平台内存分配器 feixuwu C++博客(跨平台 gui)

编辑:rootadmin
推荐一个跨平台内存分配器 - feixuwu - C++博客推荐一个跨平台内存分配器 昨天一个同事一大早在群里推荐了一个google project上的开源内存分配器(http://code.goog 推荐一个跨平台内存分配器

推荐整理分享推荐一个跨平台内存分配器 feixuwu C++博客(跨平台 gui),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:跨平台rn,推荐一个跨平台的app,跨平台网站,推荐一个跨平台的app,推荐一个跨平台的app,跨平台的好处,跨平台的好处,推荐一个跨平台的软件,内容如对您有帮助,希望把文章链接给更多的朋友!

昨天一个同事一大早在群里推荐了一个google project上的开源内存分配器(http://code.google.com/p/google-perftools/),据说google的很多产品都用到了这个内存分配库,而且经他测试,我们的游戏客户端集成了这个最新内存分配器后,FPS足足提高了将近10帧左右,这可是个了不起的提升,要知道3D组的兄弟忙了几周也没见这么大的性能提升。

如果我们自己本身用的crt提供的内存分配器,这个提升也算不得什么。问题是我们内部系统是有一个小内存管理器的,一般来说小内存分配的算法都大同小异,现成的实现也很多,比如linux内核的slab、SGI STL的分配器、ogre自带的内存分配器,我们自己的内存分配器也和前面列举的实现差不多。让我们来看看这个项目有什么特别的吧。

一、使用方法

打开主页,由于公司网络禁止SVN从外部更新,所以只能下载了打包的源代码。解压后,看到有个doc目录,进去,打开使用文档,发现使用方法极为简单:To use TCMalloc, just link TCMalloc into your application via the "-ltcmalloc" linker flag.再看算法,也没什么特别的,还是和slab以及SGI STL分配器类似的算法。unix环境居然只要链接这个tcmalloc库就可以了!,太方便了,不过我手头没有linux环境,文档上也没提到windows环境怎么使用,打开源代码包,有个vs2003解决方案,打开,随便挑选一个测试项目,查看项目属性,发现仅仅有2点不同:1、链接器命令行里多了 "..\..\release\libtcmalloc_minimal.lib",就是链接的时候依赖了这个内存优化库。2、链接器->输入->强制符号引用 多了 __tcmalloc。这样就可以正确的使用tcmalloc库了,测试了下,测试项目运行OK!

二、如何替换CRT的malloc推荐一个跨平台内存分配器  feixuwu  C++博客(跨平台 gui)

从前面的描述可知,项目强制引用了__tcmalloc, 搜索了测试代码,没发现用到_tcmalloc相关的函数和变量,这个选项应该是为了防止dll被优化掉(因为代码里没有什么地方用到这个dll的符号)。初看起来,链接这个库后,不会影响任何现有代码:我们没有引用这个Lib库的头文件,也没有使用过这个dll的导出函数。那么这个dll是怎么优化应用程序性能的呢?实际调试,果然发现问题了,看看如下代码 void* pData = malloc(100);00401085 6A 64 push 64h 00401087 FF 15 A4 20 40 00 call dword ptr [__imp__malloc (4020A4h)] 跟踪 call malloc这句,step进去,发现是78134D09 E9 D2 37 ED 97 jmp `anonymous namespace'::LibcInfoWithPatchFunctions<8>::Perftools_malloc (100084E0h) 果然,从这里开始,就跳转到libtcmalloc提供的Perftools_malloc了。原来是通过API挂钩来实现无缝替换系统自带的malloc等crt函数的,而且还是通过大家公认的不推荐的改写函数入口指令来实现的,一般只有在游戏外挂和金山词霸之类的软件才会用到这样的挂钩技术,而且金山词霸经常需要更新补丁解决不同系统兼容问题。

三、性能差别原因

如前面所述,tcmalloc确实用了很hacker的办法来实现无缝的替换系统自带的内存分配函数(本人在使用这类技术通常是用来干坏事的。。。),但是这也不足以解释为什么它的效率比我们自己的好那么多。回到tcmalloc 的手册,tcmalloc除了使用常规的小内存管理外,对多线程环境做了特殊处理,这和我原来见到的内存分配器大有不同,一般的内存分配器作者都会偷懒,把多线程问题扔给使用者,大多是加个bool型的模板参数来表示是否是多线程环境,还美其名曰:可定制,末了还得吹嘘下模板的优越性。tcmalloc是怎么做的呢? 答案是每线程一个ThreadCache,大部分操作系统都会支持thread local storage 就是传说中的TLS,这样就可以实现每线程一个分配器了,这样,不同线程分配都是在各自的threadCache里分配的。我们的项目的分配器由于是多线程环境的,所以不管三七二十一,全都加锁了,性能自然就低了。

仅仅是如此,还是不足以将tcmalloc和ptmalloc2分个高下,后者也是每个线程都有threadCache的。关于这个问题,doc里有一段说明,原文贴出来:ptmalloc2 also reduces lock contention by using per-thread arenas but there is a big problem with ptmalloc2's use of per-thread arenas. In ptmalloc2 memory can never move from one arena to another. This can lead to huge amounts of wasted space.大意是这样的:ptmalloc2 也是通过tls来降低线程锁,但是ptmalloc2各个线程的内存是独立的,也就是说,第一个线程申请的内存,释放的时候还是必须放到第一个线程池中(不可移动),这样可能导致大量内存浪费。

四、代码细节1、无缝替换malloc等crt和系统分配函数。

前面提到tcmalloc会无缝的替换掉原有dll中的malloc,这就意味着使用tcmalloc的项目必须是 MD(多线程dll)或者MDd(多线程dll调试)。tcmalloc的dll定义了一个static TCMallocGuard module_enter_exit_hook;的静态变量,这个变量会在dll加载的时候先于DllMain运行,在这个类的构造函数,会运行PatchWindowsFunctions来挂钩所有dll的 malloc、free、new等分配函数,这样就达到了替换功能,除此之外,为了保证系统兼容性,挂钩API的时候还实现了智能分析指令,否则写入第一条Jmp指令的时候可能会破环后续指令的完整性。

2、LibcInfoWithPatchFunctions 和ThreadCache。

LibcInfoWithPatchFunctions模板类包含tcmalloc实现的优化后的malloc等一系列函数。LibcInfoWithPatchFunctions的模板参数在我看来没什么用处,tcmalloc默认可以挂钩最多10个带有malloc导出函数的库(我想肯定是够用了)。ThreadCache在每个线程都会有一个TLS对象:__thread ThreadCache* ThreadCache::threadlocal_heap_。

3、可能的问题

设想下这样一个情景:假如有一个dll 在tcmalloc之前加载,并且在分配了内存(使用crt提供的malloc),那么在加载tcmalloc后,tcmalloc会替换所有的free函数,然后,在某个时刻,在前面的那个dll代码中释放该内存,这岂不是很危险。实际测试发现没有任何问题,关键在这里:span = Static::pageheap()->GetDescriptor(p); if (!span) { // span can be NULL because the pointer passed in is invalid // (not something returned by malloc or friends), or because the // pointer was allocated with some other allocator besides // tcmalloc. The latter can happen if tcmalloc is linked in via // a dynamic library, but is not listed last on the link line. // In that case, libraries after it on the link line will // allocate with libc malloc, but free with tcmalloc's free. (*invalid_free_fn)(ptr); // Decide how to handle the bad free request return; }

tcmalloc会通过span识别这个内存是否自己分配的,如果不是,tcmalloc会调用该dll原始对应函数(这个很重要)释放。这样就解决了这个棘手的问题。

五、其他

其实tcmalloc使用的每个技术点我从前都用过,但是我从来没想过用API挂钩来实现这样一个有趣的内存优化库(即使想过,也是一闪而过就否定了)。从tcmalloc得到灵感,结合常用的外挂技术,可以很轻松的开发一个独立工具:这个工具可以挂载到指定进程进行内存优化,在我看来,这可能可以作为一个外挂辅助工具来优化那些内存优化做的很差导致帧速很低的国产游戏。

本文链接地址:https://www.jiuchutong.com/zhishi/303767.html 转载请保留说明!

上一篇:Hive官方手册翻译(Getting Started) 实践检验真理 51CTO技术博客(hive.)

下一篇:Editorial Calendar为你的WordPress添加编辑日历 让文章定时发布更简单

  • 材料入库的会计分录材料采购损失
  • 小规模纳税人销售自建不动产
  • 公司账户美金入账
  • 境外所得抵免限额小于实际缴纳
  • 外地预交的附加税金在申报的时候怎么填
  • 融资租赁设备增值税怎样抵扣
  • 公司社保收费标准
  • 发出库存商品没发货
  • 投资款打到个人账户算诈骗吗
  • 土地租金一次付清的会计处理怎么做?
  • 补缴的以前年度的税费及滞纳金用更正申报企业所得税吗
  • 抵扣消费税的利益主要
  • 行政事业单位国有资产处置收入包括
  • 盈余公积为0说明什么问题
  • 销售换货怎么做账务处理
  • 凭证类别不合法怎么处理
  • 企业汇算清缴后税务局会不会查
  • 现金折扣税务会计
  • 公司基本户没开影响报税吗
  • 收到转账支票 转给下一家
  • 会计师事务所执业许可和监督管理办法
  • 企业收到银行未收到怎么做账
  • 企业所得税调增调减项目比例
  • 安装win7系统的硬件要求
  • php的类
  • win10更新kb5006670
  • win10右键个性化显示该文件
  • 如何使用linux服务器
  • 进程program
  • php查看变量数据类型
  • 出口换汇成本一般是多少
  • thinkphp技巧
  • 行政单位捐赠收入属于什么收入
  • 公司注销方便吗
  • php分页思路
  • 电子客票行程单怎么获取
  • 预缴的企业所得税怎么做账
  • mysql中文乱码怎样用代码解决
  • 进项税需要转出的分录
  • 普通发票和增值税发票的税率
  • 减免增值税计入营业外收入影响利润
  • 企业存货按照经济内容可以分为
  • 库存商品转成本意味着什么
  • 营改增利息收入增值税
  • 减值准备的定义
  • 增值税留底注销时怎么办
  • 餐饮行业加盟模式
  • 固定资产税前一次性扣除会计处理与税务处理
  • 税费改革的思路
  • 发票冲红后税款一定会退回吗
  • 房地产企业城建税和教育费附加怎么算
  • 企业领用产品的会计分录
  • 国外公司与国内公司签约
  • 支付银行贷款利息的会计处理
  • 青苗赔偿规定
  • 开发阶段的支出计入什么科目
  • 企业支付宝要手续费吗
  • 商业汇票到期无法兑现
  • 所得税会计凭证包括哪些
  • 国际贸易单证有什么制单要求
  • 自动备份mssql server数据库并压缩的批处理脚本
  • mysql删除数据库用户
  • windows server 2008下一些设置技巧及优化经验总结
  • 驱动安装过程中发生错误
  • win7系统360浏览器自启动怎么关闭
  • ubuntu14.04升级
  • schedulerv2.exe - schedulerv2是什么进程 有什么用
  • win10原装版
  • 升级win10后东西还在吗
  • opengl programming guide
  • cmd命令进行ftp下载文件夹
  • 使用jquery的步骤
  • 简述图像批处理的操作过程
  • 开发日记3雷火剑
  • js校验密码复杂度
  • android editText inputType 各个字段的含义
  • android遇到的难题,怎么解决的
  • 关于成品油的吨油换算
  • 上海二手房个税退税政策
  • 沈阳新公司办理社保流程
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设