随着项目一步一步的开发,团队小伙伴的增多,业务量的增多,APP体积也在慢慢增大,APP瘦身恰逢其时。
瘦身离不开这三个方面(暂时想到,同时也参考了其他公司的瘦身经验),首先想到的是图片资源,其次是代码优化,最后是编译build setting的瘦身。
首先说下图片资源吧,这个可能是最好想到的,因为项目中最常见的就是图片过多导致的包越来越大。
一、图片瘦身
1. 对于在项目中没有使用的图片资源的剔除,(可能是老版本遗留下来,但是现有版本已经不用,甚至代码都已经改没了,但是图片还在的情况,)
这个时候可以使用LSUnusedResources查找没有用到的图片。 LSUnusedResources是一个开源的项目,你可以在对应的github上下载下来直接运行就可以使用,使用方式在github上已经介绍的很详细了。
下载好LSUnusedResources 之后在Mac上运行项目。Project Path 选择路径,Search搜索
注意:使用的时候注意不要误删!!不要误删,对于LSUnusedResources这个项目其实也有一些问题,比如在项目中通过for循环加入的图片,没有具体的引用图片的名称也是会被检测出来的,这个时候就需要我们手动的去查看一下,项目中这些到底有没有使用。如果没有使用可以直接删除,但是如果有引用,但是你删除了这时候就会导致crash。
检测不到的图片
(1)xib, storyboard里面设置的图片
(2)项目中使用@"image_%02d"代码设置加载的图片
(3)项目中宏定义使用到的图片。
(4)图片保存在本地,需要服务器返回字段名, 本地根据返回的字符串,记载图片的。此最为奇葩,应该是很老的代码,或者有什么特殊要需求吧,建议能不用就不用。这是恨不规范的写法
2. 重复图片的剔除(就是同样大小尺寸,只是名称不同的图片)
fdupes 是Linux下的一个工具,可以在指定的目录及子目录中查找重复的文件。fdupes通过对比文件的MD5签名,以及逐字节比较文件来识别重复内容。项目中图片分两处存放,Assets.xcassets和images文件夹,所以在这两个目录查找就可以。如果没有安装fdupes直接在电脑终端运行
brew install fdupes
只需等待fdupes更新完成后,就可以安装在项目中可以直接使用查找重复的资源,
fdupes -r 文件夹位置 image文件夹位置
例如:fdupes -r /Users/developerpp/Desktop/testDemo/Assets.xcassets
3. 无损压缩图片
可以使用imageOptim工具直接压缩相应的图片,他是通过优化压缩参数,移除无用的文件数据和不必要的颜色搭配来实现无损压缩的。
如果UI同学在给你设计图的时候没有压缩图片,那么你的项目也是会变得异常的大,通过这个免费的软件可以直接瘦身压缩你需要的图片。
当然也可以使用这个链接下的网页工具,但是这个是有损压缩。http://tinypng.com。
终端查询图片大小
cd 目录
使用命令
du -hs *
或者
du -shc *
第二个命令能在最后显示一个Total大小,即当前目录的总大小。
du -sh * | sort -rh
显示当前目录下所有文件(包含文件夹)大小,并排序
二、无效文件检测
安装 fui 工具 在终端中执行命令 https://github.com/dblock/fui
sudo gem install fui -n /usr/local/bin
使用方式(耗时可能比较长时间)
到工程目录下,执行 fui find 命令,可以找出所有的没有用到的class文件
cd 目录
fui find
此外还有一些工具可以使用
1、XcodeProjectArrangementTool,用【My Mac】模拟器运行,如下图:
对检测到的unused文件,还是需要一个个去排查,要根据项目实际情况处理,不知道的最好和小伙伴之间多沟通,总是没有坏处的
2、WHC_ScanUnusedClass,用【My Mac】模拟器运行,如下图:
需要注意的是显示没用的,但实际是有用的
(1)只通过+load方法实现业务逻辑的(因为不需要文件导入)
(2)只在xxx.xib中使用的文件类
(3)只实现枚举配置的文件类
(4)通过NSClassFromString(@“xxx”)获取文件类的
(5)通过键值对建立关联关系配置的,如下:
三、可执行文件瘦身
linkmap是Xcode产生可执行文件的同时产生的连接信息,用来描述可执行文件的构造成分,具体的是在项目中打开wirte link file 设置为yes。build之后在项目中可以找到对应的位置(我是用过模拟器运行的,所以对对应的位置是模拟器的,真机可以直接在真机的相应的位置进行查找。)
相应的位置是在执行下面的js ,由于使用的环境是node,因此需要安装node环境 可以执行 brew install node
环境安装完成后执行 node linkmap.js filepath(这个就是刚刚找到的linkmap的文件)
下载好linkmap.js文件并且执行node linkmap.js filepath(这个就是上图的linkmap的文件位置)就可以看到相应变异后文件的大小,这样就可以使用根据具体的代码进行分析优化了。
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)
}
})