iOS开发APP 瘦身优化

随着项目一步一步的开发,团队小伙伴的增多,业务量的增多,APP体积也在慢慢增大,APP瘦身恰逢其时。

瘦身离不开这三个方面(暂时想到,同时也参考了其他公司的瘦身经验),首先想到的是图片资源,其次是代码优化,最后是编译build setting的瘦身。

首先说下图片资源吧,这个可能是最好想到的,因为项目中最常见的就是图片过多导致的包越来越大。

一、图片瘦身

1. 对于在项目中没有使用的图片资源的剔除,(可能是老版本遗留下来,但是现有版本已经不用,甚至代码都已经改没了,但是图片还在的情况,)

这个时候可以使用LSUnusedResources查找没有用到的图片。 LSUnusedResources是一个开源的项目,你可以在对应的github上下载下来直接运行就可以使用,使用方式在github上已经介绍的很详细了。

下载好LSUnusedResources 之后在Mac上运行项目。Project Path 选择路径,Search搜索

截屏2022-05-23 14.25.30.png

注意:使用的时候注意不要误删!!不要误删,对于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
显示当前目录下所有文件(包含文件夹)大小,并排序

截屏2022-05-23 14.35.02.png

二、无效文件检测

安装 fui 工具 在终端中执行命令 https://github.com/dblock/fui

sudo gem install fui -n /usr/local/bin

使用方式(耗时可能比较长时间)

 到工程目录下,执行 fui find 命令,可以找出所有的没有用到的class文件
cd 目录
fui find

此外还有一些工具可以使用

1、XcodeProjectArrangementTool,用【My Mac】模拟器运行,如下图:

b7ee22f11c2148daad5bbf60d0bfd0f5.png

对检测到的unused文件,还是需要一个个去排查,要根据项目实际情况处理,不知道的最好和小伙伴之间多沟通,总是没有坏处的

2、WHC_ScanUnusedClass,用【My Mac】模拟器运行,如下图:

81c517587ef54e37849d47634b142aff.png

需要注意的是显示没用的,但实际是有用的

(1)只通过+load方法实现业务逻辑的(因为不需要文件导入)
(2)只在xxx.xib中使用的文件类
(3)只实现枚举配置的文件类
(4)通过NSClassFromString(@“xxx”)获取文件类的
(5)通过键值对建立关联关系配置的,如下:

三、可执行文件瘦身

linkmap是Xcode产生可执行文件的同时产生的连接信息,用来描述可执行文件的构造成分,具体的是在项目中打开wirte link file 设置为yes。build之后在项目中可以找到对应的位置(我是用过模拟器运行的,所以对对应的位置是模拟器的,真机可以直接在真机的相应的位置进行查找。)

5264529-b158790e83c73768.png

相应的位置是在执行下面的js ,由于使用的环境是node,因此需要安装node环境 可以执行 brew install node

环境安装完成后执行 node linkmap.js filepath(这个就是刚刚找到的linkmap的文件)

Ierts & Colors Taut fosrg.png
5264529-3df316dcb1e5fd8b.png

下载好linkmap.js文件并且执行node linkmap.js filepath(这个就是上图的linkmap的文件位置)就可以看到相应变异后文件的大小,这样就可以使用根据具体的代码进行分析优化了。

5264529-dd4c337b9e60bbef.png

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)

    }

})

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349

推荐阅读更多精彩内容