更人性化的找出 iOS 中未使用的图

unImage.png

痛点

删除 iOS 项目中没有用到的图片市面上已经有很多种方式,但是我试过几个都不能很好地满足我的需求,因此使用 Python 写了这个脚本,它可能也不能很好的满足你的需求,因为这种静态查找始终会存在问题,每个人写的代码风格不一,导致匹配字符不一。所以只有掌握了脚本的写法,才能很好的满足自己的需求。如果你的项目中使用 OC,而且使用纯代码布局,使用这个脚本完全没有问题。当然你可以修改脚本来达到自己的需求。本文主要希望能够帮助更多的读者节省更多时间做一些有意义的工作,避免那些乏味重复的工作。

如何使用:

  • 1.修改 DESPATH 为你项目的路径;
  • 2.直接在脚本所在的目录下,打开终端执行 python unUseImage.py,这里的 unUseImage.py 为脚本文件名。你可以在 这里 找到脚本文件;
  • 3.执行完成后,桌面会出现一个 unUseImage 文件夹。文件夹中的 error.log 文件记录了可能存在未匹配到图片的文件目录,image.log 记录了项目中没使用的图片路径,images 存放了未使用到的图片。

重要提示:

当确认 images 文件夹中含有正在使用的图时,复制图片名字到 EXCEPT_IMAGES 中,再次执行脚本,确认 images 文件夹中不再包含使用的图后,修改 IS_OPEN_AUTO_DELTrue,执行脚本,脚本将自动清除所有未使用的图。

脚本:

# coding=utf-8 

import os
import re
import shutil

# 是否开启自动删除,开启后当检查到未用到的图,
# 将自动被删除。建议确认所有的图没用后开启
IS_OPEN_AUTO_DEL = False

# 将要解析的项目名称 
DESPATH = "/Users/wangsuyan/desktop/project/Shopn/Shopn"

# 可能检查出错的图片,需要特别留意下
ERROR_DESPATH = "/Users/wangsuyan/Desktop/unUseImage/error.log"

# 解析结果存放的路径
WDESPATH = "/Users/wangsuyan/Desktop/unUseImage/image.log"

# 项目中没有用到的图片
IMAGE_WDESPATH = "/Users/wangsuyan/Desktop/unUseImage/images/"

# 目录黑名单,这个目录下所有的图片将被忽略
BLACK_DIR_LIST = [
    DESPATH + '/ThirdPart', # Utils 下所有的文件将被忽略 
]

# 已知某些图片确实存在,比如像下面的图,脚本不会自动检查出,需要手动加入这个数组中
# NSString *name = [NSString stringWithFormat:@"loading_%d",i];
# UIImage *image = [UIImage imageNamed:name];
EXCEPT_IMAGES = [
    'loading_',
    'launch-guide'
]

# 项目中所有的图
source_images = dict()
# 项目中所有使用到的图
use_images = set()
# 异常图片
err_images = set()

# 目录是否在黑名单中 BLACK_DIR_LIST
def isInBlackList(filePath):
    if os.path.isfile(filePath):
        return filename(filePath) in BLACK_DIR_LIST
    if filePath:
        return filePath in BLACK_DIR_LIST
    return False

# 是否为图片
def isimage(filePath):
    ext = os.path.splitext(filePath)[1]
    return ext == '.png' or ext == '.jpg' or ext == '.jpeg' or ext == '.gif'

# 是否为 APPIcon
def isappicon(filePath):
    return 'appiconset' in filePath

def filename(filePath):
    return os.path.split(filePath)[1]

def is_except_image(filePath):
    name = filename(filePath)
    for item in EXCEPT_IMAGES:
        if item in name:
            return True
    return False

def auto_remove_images():
    f = open(WDESPATH, 'r')
    for line in f.readlines():
        path = DESPATH + line.strip('\n')
        if not os.path.isdir(path):
            if 'Assets.xcassets' in line:
                path = os.path.split(path)[0]
                if os.path.exists(path):
                    shutil.rmtree(path)
            else:
                os.remove(path)


def un_use_image(filePath):
    if re.search(r'\w@3x.(png|jpg|jpeg|gif)', filePath):
        return

    if re.search(r'\w(@2x){0,1}.(png|jpg|jpeg|gif)', filePath):
        exts = os.path.splitext(filePath)
        result = (filename(filePath).replace('@2x', '')).replace(exts[1],'')
        source_images[result] = filePath

def find_image_name(filePath):
    f = open(filePath)
    for index, line in enumerate(f):
        line = line.strip()
        regx = r'\[\s*UIImage\s+imageNamed\s*:\s*@"(.+?)"'
        matchs = re.findall(regx, line)
        if matchs:
            for item in matchs:
                use_images.add(item)
        else:
            err_matchs = re.findall(r'\[UIImage imageNamed:', line)
            if err_matchs:
                name = filename(filePath)
                for item in err_matchs:
                    err_images.add(str(index + 1) + ':' + name + '\n' + line + '\n')

def find_from_file(path):
    paths = os.listdir(path)
    for aCompent in paths:
        aPath = os.path.join(path, aCompent)
        if isInBlackList(aPath):
            print('在黑名单中,被自动忽略' + aPath)
            continue
        if os.path.isdir(aPath):
            find_from_file(aPath)
        elif os.path.isfile(aPath) and isimage(aPath) and not isappicon(aPath) and not is_except_image(aPath):
            un_use_image(aPath)
        elif os.path.isfile(aPath) and os.path.splitext(aPath)[1]=='.m':
            find_image_name(aPath)

if __name__ == '__main__':
    if os.path.exists(IMAGE_WDESPATH):
        shutil.rmtree(IMAGE_WDESPATH)

    os.makedirs(IMAGE_WDESPATH)

    wf = open(WDESPATH, 'w')
    find_from_file(DESPATH)
    for item in set(source_images.keys()) - use_images:
        value = source_images[item]
        wf.write(value.replace(DESPATH, '') + '\n')
        ext = os.path.splitext(value)[1]
        shutil.copyfile(value, IMAGE_WDESPATH + item + ext)

    wf.close()

    ef = open(ERROR_DESPATH, 'w')
    for item in err_images:
        ef.write(item)
    ef.close()

    if IS_OPEN_AUTO_DEL:
        auto_remove_images()

推荐阅读

【iOS 国际化】如何把国际化时需要3天的工作量缩减到10分钟

===== 我是有底线的 ======
喜欢我的文章,欢迎关注我的新浪微博 Lefe_x,我会不定期的分享一些开发技巧

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,138评论 25 707
  • 22年12月更新:个人网站关停,如果仍旧对旧教程有兴趣参考 Github 的markdown内容[https://...
    tangyefei阅读 35,156评论 22 257
  • 作为女生,好像天生就对食物特别的喜欢,不管走到哪里都会想要吃东西。同时,作为女生,二十多岁一过,人的新陈代谢就开始...
    isabellaLilove阅读 243评论 0 0
  • 文 | 晨儿 每天差不多都是相同的重复,潮起潮落,昨天和前天颠倒顺序 也没有任何不便,我不时想,这叫什么人生啊。 ...
    晨妤儿阅读 430评论 0 2
  • 2017年9月24日 星期日 今晚看《80天环游地球》第35章,特别舒适的快车。 福克先生一行人登上了穿越美...
    鑫隆妈妈阅读 210评论 0 0