iOS 组件化或SDK时对资源加载注意点

本文详细介绍了使用CocoaPods进行资源打包的方法,包括静态库和动态库的构建。通过podspec文件配置s.static_framework参数可以控制库的类型。静态库会将资源编译进.app二进制包,而动态库则将资源放入.framework中。使用resource_bundles可以有效地组织和打包资源,避免重复。在访问资源时,需要注意bundle的使用方式,特别是静态库和动态库的区别。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

资源文件打包方式

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

use_frameworks! 

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

podfile文件中 use_frameworks!作用。

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

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

其次 像 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

 

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

 

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

纯的动态库

  • 把podspec 中s.static_framework = false或注释

  • 把Podfile中的 use_frameworks! 打开

  • 重新podinstall

再来解压.app包看下 

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

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

纯Framework的方式打包资源分析

前置条件:

  • 资源文件路径:

 

 

 

  • 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

如果podspec不使用bundle的方式时 

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

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

  • 设置s.static_framework = true

  • Podfile注释不注use_frameworks! 作为伪静态包 Podfile注释#use_frameworks! 作为完全静态包

  • 必须要命名用bundle来存放资源

#  s.resources = 'AppResourceModule/Assets/**/{*}' 这种方式不能使用,如果要使用手动创建一个bundle,把所有资源放里面。
   s.resource_bundles = {
     'AppResourceModule' => ['AppResourceModule/Assets/**/{*}']
   }

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

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

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

当使用 

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

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

如果改为:

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

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

 

总结:

  • 对于真动态库,.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访问方式。如果不确认的则做双重保障,静动态库路径一起判断。

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

边缘998

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值