iOS APP可執(zhí)行文件的組成

轉(zhuǎn)載自:http://blog.cnbang.net/tech/2296/
iOS APP編譯后,除了一些資源文件,剩下的就是一個(gè)可執(zhí)行文件,有時(shí)候項(xiàng)目大了,引入的庫多了,可執(zhí)行文件很大,想知道這個(gè)可執(zhí)行文件的構(gòu)成是怎樣,里面的內(nèi)容都是些什么,哪些庫占用空間較高,可以用以下方法勘察:
1.XCode開啟編譯選項(xiàng)Write Link Map File
XCode -> Project -> Build Settings -> 搜map -> 把Write Link Map File選項(xiàng)設(shè)為yes,并指定好linkMap的存儲(chǔ)位置

linkmap

2.編譯后,到編譯目錄里找到該txt文件,文件名和路徑就是上述的Path to Link Map File 位于~/Library/Developer/Xcode/DerivedData/XXX-eumsvrzbvgfofvbfsoqokmjprvuh/Build/Intermediates/XXX.build/Debug-iphoneos/XXX.build/
這個(gè)LinkMap里展示了整個(gè)可執(zhí)行文件的全貌,列出了編譯后的每一個(gè).o目標(biāo)文件的信息(包括靜態(tài)鏈接庫.a里的),以及每一個(gè)目標(biāo)文件的代碼段,數(shù)據(jù)段存儲(chǔ)詳情。

1

以TakePart項(xiàng)目為例,在LinkMap里首先列出來的是目標(biāo)文件列表:

# Object files:
[  0] linker synthesized
[  1] /Users/BigGIfts/Library/Developer/Xcode/DerivedData/TakePart-bkiirdprmbotohckzelyyxbkawmy/Build/Intermediates/TakePart.build/Debug-iphonesimulator/TakePart.build/TakePart.app.xcent
[  2] /Users/BigGIfts/Library/Developer/Xcode/DerivedData/TakePart-bkiirdprmbotohckzelyyxbkawmy/Build/Intermediates/TakePart.build/Debug-iphonesimulator/TakePart.build/Objects-normal/x86_64/UIView+WebCacheOperation.o
...
[ 80] /Applications/Xcode.app/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator10.2.sdk/usr/lib/libc++abi.dylib
[ 81] /Applications/Xcode.app/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator10.2.sdk/System/Library/Frameworks//CoreFoundation.framework/CoreFoundation
[ 82] /Applications/Xcode.app/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator10.2.sdk/System/Library/Frameworks//MobileCoreServices.framework/MobileCoreServices
[ 83] /Applications/Xcode.app/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator10.2.sdk/System/Library/Frameworks//ImageIO.framework/ImageIO

前面中括號(hào)里的是這個(gè)文件的編號(hào),后面會(huì)用到,像項(xiàng)目里引用到靜態(tài)鏈接庫里的目標(biāo)文件都會(huì)在這里列出來。

2

接著是一個(gè)段表,描述各個(gè)段在最后編譯成的可執(zhí)行文件中的偏移位置及大小,包括了代碼段(__TEXT,保存程序代碼段編譯后的機(jī)器碼)和數(shù)據(jù)段(__DATA,保存變量值)

# Sections:
# Address   Size        Segment Section
0x100001BD0 0x001CE159  __TEXT  __text
0x1001CFD2A 0x00000A0E  __TEXT  __stubs
0x1001D0738 0x000010AA  __TEXT  __stub_helper
0x1001D17E2 0x0001CE24  __TEXT  __objc_methname
0x1001EE606 0x00009665  __TEXT  __objc_methtype
0x1001F7C6B 0x000026CB__TEXT    __objc_classname
0x1001FA340 0x0001A6DF  __TEXT __cstring
0x100214A20 0x0000D974__TEXT    __gcc_except_tab
0x1002223A0 0x0000667C__TEXT    __const
0x100228A1C 0x00000884  __TEXT  __ustring
0x1002292A0 0x00000174 __TEXT   __entitlements
0x100229414 0x00005BE4  __TEXT  __unwind_info
0x10022F000 0x00000010  __DATA  __nl_symbol_ptr
0x10022F010 0x00000348  __DATA\__got
0x10022F358 0x00000D68  __DATA  __la_symbol_ptr
0x1002300C0 0x00000030  __DATA  __mod_init_func
0x1002300F0 0x00000008 __DATA   __mod_term_func
0x100230100 0x00009478  __DATA  __const
0x100239578 0x0000A760__DATA __cfstring
0x100243CD8 0x00000B78__DATA    __objc_classlist
0x100244850 0x00000100  __DATA __objc_catlist
0x100244950 0x00000010  __DATA  __objc_nlcatlist
0x100244960 0x00000230__DATA    __objc_protolist
0x100244B90 0x00000008  __DATA  __objc_imageinfo
0x100244B98 0x00049FB8  __DATA __objc_const
0x10028EB50 0x000068B0  __DATA\__objc_selrefs
0x100295400 0x00000028  __DATA  __objc_protorefs
0x100295428 0x00000C50  __DATA  __objc_classrefs
0x100296078 0x00000878  __DATA  __objc_superrefs
0x1002968F0 0x00003248  __DATA __objc_ivar
0x100299B38 0x000072B0  __DATA  __objc_data
0x1002A0DF0 0x00005528  __DATA  __data
0x1002A6320 0x00000708 __DATA __bss
0x1002A6A28 0x00000048  __DATA __common

首列是數(shù)據(jù)在文件的偏移位置,第二列是這一段占用大小,第三列是段類型,代碼段和數(shù)據(jù)段,第四列是段名稱。
每一行的數(shù)據(jù)都緊跟在上一行后面,如第二行__stubs的地址0x1001CFD2A就是第一行__text的地址0x100001BD0加上大小0x001CE159,整個(gè)可執(zhí)行文件大致數(shù)據(jù)分布就是這樣。
這里可以清楚看到各種類型的數(shù)據(jù)在最終可執(zhí)行文件里占的比例,例如__text表示編譯后的程序執(zhí)行語句,__data表示已初始化的全局變量和局部靜態(tài)變量,__bss表示未初始化的全局變量和局部靜態(tài)變量,__cstring表示代碼里的字符串常量,等等。

3

接著就是按上表順序,列出具體的按每個(gè)文件列出每個(gè)對(duì)應(yīng)字段的位置和占用空間

# Symbols:
# Address   Size        File  Name
0x100001BD0 0x000000E0  [  2] -[UIView(WebCacheOperation) operationDictionary]
0x100001CB0 0x000000D0  [  2] -[UIView(WebCacheOperation) sd_setImageLoadOperation:forKey:]
0x100001D80 0x00000340  [  2] -[UIView(WebCacheOperation) sd_cancelImageLoadOperationWithKey:]
0x1000020C0 0x00000083  
...

同樣首列是數(shù)據(jù)在文件的偏移地址,第二列是占用大小,第三列是所屬文件序號(hào),對(duì)應(yīng)上述Object files列表,最后是名字。
例如第一行代表了文件序號(hào)為2(反查上面就是UIView+WebCacheOperation.o)的operationDictionary方法占用了0x000000E0大小。

使用

這個(gè)文件可以讓你了解整個(gè)APP編譯后的情況,也許從中可以發(fā)現(xiàn)一些異常,還可以用這個(gè)文件計(jì)算靜態(tài)鏈接庫在項(xiàng)目里占的大小,有時(shí)候我們?cè)陧?xiàng)目里鏈了很多第三方庫,導(dǎo)致APP體積變大很多,我們想確切知道每個(gè)庫占用了多大空間,可以給我們優(yōu)化提供方向。LinkMap里有了每個(gè)目標(biāo)文件每個(gè)方法每個(gè)數(shù)據(jù)的占用大小數(shù)據(jù),所以只要寫個(gè)腳本,就可以統(tǒng)計(jì)出每個(gè).o最后的大小,屬于一個(gè).a靜態(tài)鏈接庫的.o加起來,就是這個(gè)庫在APP里占用的空間大小。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容