前一段时间在知乎上看到个问题:Linux如何优化可执行程序的体积?
在我们的日常工作中,一般对程序的体积都有严格的要求,有时候仅仅因为几字节的代码段体积或者多了几十毫秒的运行时间,整个项目就达不到验收标准,甚至不能成功上线。这里我抛砖引玉先提出几个思路,大家如果有好的优化策略欢迎打在评论区。
大体思路有这些:
- 好好写代码,减小代码段体积,别人300代码的逻辑我们50行搞定,程序体积肯定有机会更小一些,这个就得考验开发者自己的编程功底了
- 如果是C++程序,可以尽量减少模板的使用,模板实例化可能会导致代码膨胀
- 不用引用没有用的头文件
- 使用strip,像脱衣服一样,移除程序的所有符号,这也是很多开发者常用的方式
- strip只会清除普通符号,不会动态符号表中的符号,某些动态符号其实也可以隐藏掉,进而来减小库的体积,可以使用-fvisibility=hidden命令
- 巧用.bss段,未初始化的全局变量和局部静态变量会存在.bss段中,这些变量不占用程序空间
- inline-limit:内联过多会导致代码段体积较大,可以通过此优化选项减少内联的数量
- 开启Os编译,这是产生较小代码体积的优化选项
- 适当使用编译选项-fdata-sections和-ffunction-sections
- 考虑链接动态库而非静态库
以上说的太笼统了?贴心如我早就准备好了,不谢~
strip使用
在Linux中可以使用man strip查看strip使用方法,最主要的就是移除所有符号的-s参数,用于清除所有的符号信息:
- strip -s xxx
在使用strip之前先使用nm查看下可执行程序的符号信息:
- ~/test$ nm a.out
- 0000000000200da0 d _DYNAMIC
- 0000000000200fa0 d _GLOBAL_OFFSET_TABLE_
- 000000000000089b t _GLOBAL__sub_I__Z4funcPc
- 0000000000000930 R _IO_stdin_used
- w _ITM_deregisterTMCloneTable
- w _ITM_registerTMCloneTable
- 0000000000000852 t _Z41__static_initialization_and_destruction_0ii
- 00000000000007fa T _Z4funcPc
- 000000000000081c T _Z4funci
- U _ZNSt8ios_base4InitC1Ev@@GLIBCXX_3.4
- U _ZNSt8ios_base4InitD1Ev@@GLIBCXX_3.4
- 0000000000201020 B _ZSt4cout@@GLIBCXX_3.4
- 0000000000000934 r _ZStL19piecewise_construct
- 0000000000201131 b _ZStL8__ioinit
- U _ZStlsISt11char_traitsIcEERSt13basic_ostreamIcT_ES5_PKc@@GLIBCXX_3.4
- 0000000000000b24 r __FRAME_END__
- 0000000000000940 r __GNU_EH_FRAME_HDR
- 0000000000201010 D __TMC_END__
- 0000000000201010 B __bss_start
- U __cxa_atexit@@GLIBC_2.2.5
- w __cxa_finalize@@GLIBC_2.2.5
- 0000000000201000 D __data_start
- 00000000000007b0 t __do_global_dtors_aux
- 0000000000200d98 t __do_global_dtors_aux_fini_array_entry
- 0000000000201008 D __dso_handle
- 0000000000200d88 t __frame_dummy_init_array_entry
- w __gmon_start__
- 0000000000200d98 t __init_array_end
- 0000000000200d88 t __init_array_start
- 0000000000000920 T __libc_csu_fini
- 00000000000008b0 T __libc_csu_init
- U __libc_start_main@@GLIBC_2.2.5
- 0000000000201010 D _edata
- 0000000000201138 B _end
- 0000000000000924 T _fini
- 0000000000000688 T _init
- 00000000000006f0 T _start
- 0000000000201130 b completed.7698
- 0000000000201000 W data_start
- 0000000000000720 t deregister_tm_clones
- 00000000000007f0 t frame_dummy
- 000000000000083d T main
- 0000000000000760 t register_tm_clones
当前这个可执行程序的文件大小是8840字节:
- -rwxrwxrwx 1 a a 8840 Nov 29 14:54 a.out
使用strip清除符号信息:
- ~/test$ strip -s a.out
strip后再查看可执行文件的符号信息:
- ~/test$ nm a.out nm: a.out: no symbols
发现什么符号都没有了,但还是可以执行。
strip后的可执行程序文件大小是6120字节:
- -rwxrwxrwx 1 a a 6120 Nov 29 14:54 a.out
具体可以看我这篇文章:《Linux有一个命令你一定要知道》
-fvisibility=hidden可以这样使用:
- $ g++ -fvisibility=hidden -c layer.cxx -o layer.o