本文针对cocoapods进行打包的资源做个讲解,针对自身项目注意调整资源访问策略。

资源文件打包方式

使用pod lib create AppResourceModule 来进行演示。

use_frameworks! 

target 'AppResourceModule_Example' do
  pod 'AppResourceModule', :path => '../'
end

podfile文件中 use_frameworks!作用。

#把pod中的库都作为静态库来进行依赖时
#use_frameworks! 进行注释

ios app资源库 分类 ios app资源库 建议_cocoapods

#把pod的库作为动态库来进行连接(framework)
use_frameworks! //打开

ios app资源库 分类 ios app资源库 建议_xcode_02

其次 像 pod 'AppResourceModule' //引入库时,是否为动态库进行连接完全处决于该库的podspec中的设置. 

s.static_framework = true 

#如果库的podspec指定为使用静态库的方式提供给pod,就算podfile中使用use_frameworks!也会以静态库的方式打进包中。
也就是说打出来的framework是伪静态库。

如何区分动态库和伪动态库。

什么是"伪"动态库,本质是静态库,只是穿上了framework的外衣。

  • 把 AppResourceModule库的podspce中设置s.static_framework = true即将库强制指定为静态库
  • 把Podfile中的 use_frameworks! 打开。告诉pod我要打动态库的方式来提供。
  • 重新podinstall

ios app资源库 分类 ios app资源库 建议_xcode_03

 

 为什么说是静态连接的方式,咱们可以从主工程.app的包中解压看下framework文件夹下是否有这个.framework就知道了。

ios app资源库 分类 ios app资源库 建议_macos_04

 

可以看到并没有动态库,实际上都被静态编译到了App的二进制包中了。

纯的动态库

  • 把podspec 中s.static_framework = false或注释
  • 把Podfile中的 use_frameworks! 打开
  • 重新podinstall

ios app资源库 分类 ios app资源库 建议_ios_05

再来解压.app包看下 

ios app资源库 分类 ios app资源库 建议_ios_06

可以看到已经是动态库framework了,并没有被提前编译到app的二进制文件中。

打包后资源被放置的位置分析

纯Framework的方式打包资源分析

前置条件:

  • 资源文件路径:

ios app资源库 分类 ios app资源库 建议_ios app资源库 分类_07

 

ios app资源库 分类 ios app资源库 建议_xcode_08

 

ios app资源库 分类 ios app资源库 建议_xcode_09

 

  • Podspec 指定资源打包方式:
把资源都打包到bundle包里
s.resource_bundles = {
    # 包的名称                包的资源所在的目录位置,即将目录下所有资源打到AppResourceModule.bundle里
     'AppResourceModule' => ['AppResourceModule/Assets/**/{*.xcassets}','AppResourceModule/Assets/**/{*}']
   }
  • 注释 s.static_framework = true 或改为false

打出来的包,资源文件被放置在.framework内部

app->解包后
Frameworks-->AppResourceModule.framework-->AppResourceMoudle.bundle

ios app资源库 分类 ios app资源库 建议_ios_10

ios app资源库 分类 ios app资源库 建议_xcode_11

如果podspec不使用bundle的方式时 

s.resources = 'AppResourceModule/Assets/**/{*}'
#则直接把资源都copy到.framework的根目录下。

ios app资源库 分类 ios app资源库 建议_ios_12

伪静态库打包资源分析和完全使用静态库时分析情况是一样的

  • 设置s.static_framework = true
  • Podfile注释不注use_frameworks! 作为伪静态包 Podfile注释#use_frameworks! 作为完全静态包
  • 必须要命名用bundle来存放资源
#  s.resources = 'AppResourceModule/Assets/**/{*}' 这种方式不能使用,如果要使用手动创建一个bundle,把所有资源放里面。
   s.resource_bundles = {
     'AppResourceModule' => ['AppResourceModule/Assets/**/{*}']
   }

ios app资源库 分类 ios app资源库 建议_cocoapods_13

有多少资源被打进库,如何避免重复?

当资源文件夹中有Bundle资源,些时路径不要使用{*},因为会把bundle解包的资源也复制一份

如Res.bundle中只有四个资源图片:

ios app资源库 分类 ios app资源库 建议_cocoapods_14

当使用 

s.resource_bundles = {
     'AppResourceModule' => ['AppResourceModule/Assets/**/{*}'] #全量加载
}

pod install后,可以看到资源文件被重复添加了一次。 

ios app资源库 分类 ios app资源库 建议_xcode_15

如果改为:

s.resource_bundles = {
     'AppResourceModule' => ['AppResourceModule/Assets/**/{*.bundle}','AppResourceModule/Assets/**/{*.xcassets}']
     或 (前后位置没有关系)
     'AppResourceModule' => ['AppResourceModule/Assets/**/{*.xcassets}','AppResourceModule/Assets/**/{*.bundle}']
     
   }

重新pod install 后打包,可以看到很干净的资源了 

ios app资源库 分类 ios app资源库 建议_cocoapods_16

 

总结:

  • 对于真动态库,.xcassets会被压缩成Assets.car后存放在s.resource_bundles指定的文名名.bundle(AppResourceModule.bundle)中存放。所有被指定的资源都被打包到该bundle中,连Res.bundle也不另外。
  • 其次AppResourceModule.bundle,将被打包后放在AppResourceModule.framework的根目录下。相对于主程序.app的目录路径为 .app/Framework/AppResourceModule.framework/AppResourceModule.bundle/Assets.car,或Res.bundle。
  • 对于静态库或伪动态库,则打成的AppResourceModule.bunld路径为.app/AppResourceModule.bundle

            访问时使用的Bundle方式注意可能不是mainbundle了。需要针对性的调整bundle访问方式。如果不确认的则做双重保障,静动态库路径一起判断。