一、LinkMap文件分析
说明:LinkMap数据是根据文章《LinkMap文件分析》中方法实验实测数据。
如何获得LinkMap文件
1.在XCode中开启编译选项Write Link Map File \n
XCode -> Project -> Build Settings -> 把Write Link Map File选项设为yes,并指定好linkMap的存储位置
2.工程编译完成后,在编译目录里找到Link Map文件(txt类型)
在Build Settings里Path to Link Map File对应Link Map的路径,以下是默认路径:
/Users/xxx/Library/Developer/Xcode/DerivedData/YourApp/Build/Intermediates.noindex/YourApp.build/Debug-iphoneos/YourApp.build/YourApp-LinkMap-normal-arm64.txt
LinkMap里有了每个目标文件每个方法每个数据的占用大小数据,所以只要写个脚本,就可以统计出每个.o最后的大小,属于一个.a静态链接库的.o加起来,就是这个库在APP里占用的空间大小。
LinkMap包含以下部分:
# Path: .../*.app文件
# Arch: arm64
# Object files: .../*.o文件
# Sections: Text和Data分类大小
# Symbols: 与Object files对应起来,是Object files的详细Symbal信息,包括地址和大小
# Dead Stripped Symbols:
// linkmap.js
var readline = require('readline'),
fs = require('fs');
var LinkMap = function(filePath) {
this.files = []
this.filePath = filePath
}
LinkMap.prototype = {
start: function(cb) {
var self = this
var rl = readline.createInterface({
input: fs.createReadStream(self.filePath),
output: process.stdout,
terminal: false
});
var currParser = "";
rl.on('line', function(line) {
if (line[0] == '#') {
if (line.indexOf('Object files') > -1) {
currParser = "_parseFiles";
} else if (line.indexOf('Sections') > -1) {
currParser = "_parseSection";
} else if (line.indexOf('Symbols') > -1) {
currParser = "_parseSymbols";
}
return;
}
if (self[currParser]) {
self[currParser](line)
}
});
rl.on('close', function(line) {
cb(self)
});
},
_parseFiles: function(line) {
var arr =line.split(']')
if (arr.length > 1) {
var idx = Number(arr[0].replace('[',''));
var file = arr[1].split('/').pop().trim()
this.files[idx] = {
name: file,
size: 0
}
}
},
_parseSection: function(line) {
},
_parseSymbols: function(line) {
var arr = line.split('\t')
if (arr.length > 2) {
var size = parseInt(arr[1], 16)
var idx = Number(arr[2].split(']')[0].replace('[', ''))
if (idx && this.files[idx]) {
this.files[idx].size += size;
}
}
},
_formatSize: function(size) {
if (size > 1024 * 1024) return (size/(1024*1024)).toFixed(2) + "MB"
if (size > 1024) return (size/1024).toFixed(2) + "KB"
return size + "B"
},
statLibs: function(h) {
var libs = {}
var files = this.files;
var self = this;
for (var i in files) {
var file = files[i]
var libName
if (file.name.indexOf('.o)') > -1) {
libName = file.name.split('(')[0]
} else {
libName = file.name
}
if (!libs[libName]) {
libs[libName] = 0
}
libs[libName] += file.size
}
var i = 0, sortLibs = []
for (var name in libs) {
sortLibs[i++] = {
name: name,
size: libs[name]
}
}
sortLibs.sort(function(a,b) {
return a.size > b.size ? -1: 1
})
if (h) {
sortLibs.map(function(o) {
o.size = self._formatSize(o.size)
})
}
return sortLibs
},
statFiles: function(h) {
var self = this
self.files.sort(function(a,b) {
return a.size > b.size ? -1: 1
})
if (h) {
self.files.map(function(o) {
o.size = self._formatSize(o.size)
})
}
return this.files
}
}
if (!process.argv[2]) {
console.log('usage: node linkmap.js filepath -hl')
console.log('-h: format size')
console.log('-l: stat libs')
return
}
var isStatLib, isFomatSize
var opts = process.argv[3];
if (opts && opts[0] == '-') {
if (opts.indexOf('h') > -1) isFomatSize = true
if (opts.indexOf('l') > -1) isStatLib = true
}
var linkmap = new LinkMap(process.argv[2])
linkmap.start(function(){
var ret = isStatLib ? linkmap.statLibs(isFomatSize)
: linkmap.statFiles(isFomatSize)
for (var i in ret) {
console.log(ret[i].name + '\t' + ret[i].size)
}
})
执行:
node linkmap.js LinkMap-normal-arm64.txt -hl
得到:
YourSDK(YourSDK-arm64-master.o) 122.76KB
二、xcodebuild总包大小分析
分析:Release 包(arm64)为什么比debug包(x86_64)大那么多?
分析步骤
1. 抽出YourSDK的arm64看包大小(1.6M)
lipo -thin arm64 -output YourSDK_arm64.a YourSDK.framework/YourSDK
➜ ls -al YourSDK_arm64.a
-rw-r--r-- 1 yushu.lxy staff 1606208 11 18 15:58 YourSDK_arm64.a
2. 整体
size YourSDK_armv64.a
__TEXT __DATA __OBJC others dec hex
86591 39492 0 1190617 1316700 14175c YourSDK_arm64.a(YourSDK-arm64-master.o)
30 240 0 4566 4836 12e4 YourSDK_arm64.a(libPods-YourSDK.a-arm64-master.o)
查看.a文件下.o文件大小
ar -t -v YourSDK_arm64.a
rw-r--r-- 501/20 5776 Nov 18 15:58 2019 __.SYMDEF SORTED
rw-r--r-- 0/0 1593168 Jan 1 08:00 1970 YourSDK-arm64-master.o
rw-r--r-- 0/0 6976 Jan 1 08:00 1970 libPods-YourSDK.a-arm64-master.o
3. 详细
size -m YourSDK_arm64.a
YourSDK_arm64.a(YourSDK-arm64-master.o):
Segment : 1316713
Section (__TEXT, __text): 68676(比debug的包小)
Section (__TEXT, __objc_methname): 8393
Section (__TEXT, __cstring): 5091
Section (__TEXT, __objc_classname): 1636
Section (__TEXT, __objc_methtype): 1937
Section (__TEXT, __literal16): 128(只有debug包有)
Section (__TEXT, __literal4): 8(只有debug包有)
Section (__TEXT, __literal8): 144(比debug的包小)
Section (__TEXT, __eh_frame): 16480
Section (__TEXT, __gcc_except_tab): 608
Section (__TEXT, __ustring): 42
Section (__TEXT, __const): 64
Section (__DATA, __const): 1560
Section (__DATA, __cfstring): 4736
Section (__DATA, __objc_classlist): 400
Section (__DATA, __objc_catlist): 8
Section (__DATA, __objc_protolist): 56
Section (__DATA, __objc_imageinfo): 8
Section (__DATA, __objc_const): 24224
Section (__DATA, __objc_selrefs): 2752
Section (__DATA, __objc_protorefs): 8
Section (__DATA, __objc_classrefs): 592
Section (__DATA, __objc_superrefs): 192
Section (__DATA, __objc_ivar): 252(比debug的包小)
Section (__DATA, __objc_data): 4000
Section (__DATA, __data): 672
Section (__DATA, __bss): 32
Section (__LD, __compact_unwind): 11424
Section (__LLVM, __bundle): 1179193(+)
total 1316700
total 1316713
YourSDK_arm64.a(libPods-YourSDK.a-arm64-master.o):
Segment : 4838
Section (__TEXT, __text): 0
Section (__TEXT, __objc_classname): 30
Section (__DATA, __objc_classlist): 8
Section (__DATA, __objc_imageinfo): 8
Section (__DATA, __objc_const): 144
Section (__DATA, __objc_data): 80
Section (__LLVM, __bitcode): 4480 (+)
Section (__LLVM, __cmdline): 86 (+)
total 4836
total 4838
YourSDK SDK实际大小应该是:1316713-1179193 = 137KB(与LinkMap分析接近)
分析:
上面段数据LLVM bundle部分是release包比debug包大的原因,是因为release包包含了bitCode数据。
(1)bitcode是源码和机器码之间的中间形式。
(2)为什么苹果要强推bitcode?开发者把bitcode提交到App Store Connect之后,如果苹果发布了使用新芯片的iPhone,支持更高效的指令,开发者不需要做任何操作,App Store Connect自己就可以编译出针对新产品优化过的app并通过App Store分发给用户,不需要开发者自己重新打包上架。
(3)苹果可以将bitcode代码进行一些逻辑等价的转换,使得代码的执行效率更高,体积更小。
实验:APP的project开启bitcode和关闭bitcode,查看手机中app的实际大小,开启bitcode的包更小。
size -m YourSDK_arm64.a
YourSDK_arm64.a(YourSDK-arm64-master.o):
Segment : 943490
Section (__TEXT, __text): 53184
Section (__TEXT, __objc_classname): 633
Section (__TEXT, __objc_methname): 7919
Section (__TEXT, __objc_methtype): 850
Section (__TEXT, __cstring): 7352
Section (__TEXT, __literal8): 64
Section (__TEXT, __gcc_except_tab): 1724
Section (__TEXT, __const): 112
Section (__DATA, __const): 1864
Section (__DATA, __cfstring): 4000
Section (__DATA, __objc_classlist): 200
Section (__DATA, __objc_protolist): 8
Section (__DATA, __objc_imageinfo): 8
Section (__DATA, __objc_const): 21760
Section (__DATA, __objc_selrefs): 1776
Section (__DATA, __objc_classrefs): 352
Section (__DATA, __objc_superrefs): 24
Section (__DATA, __objc_ivar): 620
Section (__DATA, __objc_data): 2000
Section (__DATA, __data): 96
Section (__DATA, __bss): 40
Section (__DATA, __common): 8
Section (__LD, __compact_unwind): 15968
Section (__LLVM, __bundle): 822914
total 943476
total 943490
分析结论
(1) LinkMap分析的大小不包含打入framework中的bundle信息,建议用LinkMap分析的包大小至少要加上bundle的大小。
(2)经多次安装实验,真实安装包后, APP大小增长是比LinkMap给出的包大小大的。
(3)给外部的包大小信息,暂时跟大家保持一致“使用LinkMap给出的数据”。