iOS中Framework Library嵌套使用

iOS中Framework Library嵌套使用

组件化的时候遇到一些Framework使用依赖的问题, 之前未系统的验证过各种库嵌套逻辑,下面就以集中常见的库嵌套使用来看看默认的link逻辑

讨论一下几种情况

  1. StaticFramework嵌套使用
  2. StaticFramework中嵌套使用StaticLibrary
  3. StaticLibrary嵌套使用
  4. StaticLibrary嵌套使用StaticFramework
  5. DynamicFramework嵌套使用StaticLibrary
  6. DynamicFramework嵌套使用StaticFramework
  7. DynamicFramework嵌套使用DynamicFramework
1. StaticFramework嵌套使用
  • 创建叫做StaticFrameworkOne的StaticFramework
    内部实现staticFrameworkOneTestMethod方法

    - (void)staticFrameworkOneTestMethod{
    NSLog(@"%s",__func__);
    }
    
  • 创建叫做StaticFrameworkTwo的StaticFramework
    内部实现staticFrameworkTwoTestMethod方法
    引入StaticFrameworkOne库,并调用其方法

    - (void)staticFrameworkTwoTestMethod{
       NSLog(@"%s",__func__);
       StaticFrameworkOneTest *one = [[StaticFrameworkOneTest alloc] init];
       [one staticFrameworkOneTestMethod];
    }
    
  • 将编译生成好的StaticFrameworkTwo Framework加入Demo工程, 并调用其staticFrameworkTwoTestMethod方法

    - (void)viewDidLoad {
       [super viewDidLoad];
       
       StaticFrameworkTwoTest *two = [[StaticFrameworkTwoTest alloc] init];
       [two staticFrameworkTwoTestMethod];
    }
    

    run demo工程,报错


    sfaddsf.png
  • demo工程加入StaticFrameworkOne framework, 运行工程不再报错, 并输出:

    2018-04-24 09:38:06.974602+0800 FrameworkExample[9362:3232257] -[StaticFrameworkTwoTest staticFrameworkTwoTestMethod]
    2018-04-24 09:38:06.974777+0800 FrameworkExample[9362:3232257] -[StaticFrameworkOneTest staticFrameworkOneTestMethod]
    

结论: StaticFramework中嵌套使用其他StaticFramework, 不会包含内部嵌套的StaticFramework的内容,想使用该库,需要同时link其内部的StaticFramework.
使用场景: 自己framework中包含了其他业务也使用到的库, 为避免最终工程符号冲突或者避免工程中多个库文件,采取这种嵌套方式比较适合.

2. StaticFramework中嵌套使用StaticLibrary
  • 创建叫做StaticLibraryOne的StaticLibrary, 实现方法- (void)StaticLibraryOneMethod

    - (void)StaticLibraryOneMethod{
        NSLog(@"%s",__func__);
    }
    
  • 将编译好的libStaticLibraryOne.a加入StaticFrameworkOne, 并调用其方法

    - (void)staticFrameworkOneTestMethod{
        NSLog(@"%s",__func__);
        
        StaticLibraryOne *libraryOne = [[StaticLibraryOne alloc] init];
        [libraryOne StaticLibraryOneMethod];
    }
    
  • 编译StaticFrameworkOne, 将其framework加入demo工程, 并调用其方法:

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        StaticFrameworkOneTest *one = [[StaticFrameworkOneTest alloc] init];
        [one staticFrameworkOneTestMethod];
    }
    
  • 日志输出

    2018-04-24 09:53:17.170627+0800 FrameworkExample[9604:3251674] -[StaticFrameworkOneTest staticFrameworkOneTestMethod]
    2018-04-24 09:53:17.170839+0800 FrameworkExample[9604:3251674] -[StaticLibraryOne StaticLibraryOneMethod]
    

结论: staticFramework中嵌套使用staticLibrary, 会将staticLibrary中的内容全部link到framework内部, 外部使用staticFramework时, 无需再单独link内部staticLibrary
使用场景: 对外提供完整的一个功能模块时可以使用这种嵌套方式
注意: 如果外部也使用到了内部嵌套的staticLibrary, 可能会出现符号冲突问题, 所以内部嵌套使用staticLibrary时, 最好做了类名方法名全局变量名等名称的修改(加一个前缀之类的)

3. StaticLibrary嵌套使用
  • 创建叫StaticLibraryTwo的staticLibrary,并实现其方法:

    - (void)StaticLibraryTwoMethod{
        NSLog(@"%s",__func__);
    }
    
  • StaticLibraryTwo加入StaticLibraryOne, 并调用StaticLibraryTwo中的方法:

    - (void)StaticLibraryOneMethod{
        NSLog(@"%s",__func__);
        
        StaticLibraryTwo *two = [[StaticLibraryTwo alloc] init];
        [two StaticLibraryTwoMethod];
    }
    
  • 编译StaticLibraryOne, 将其加入demo工程, 并调用其方法

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        StaticLibraryOne *one = [[StaticLibraryOne alloc] init];
        [one StaticLibraryOneMethod];
    }
    
  • 日志输出:

    2018-04-24 10:08:03.486327+0800 FrameworkExample[9845:3271122] -[StaticLibraryOne StaticLibraryOneMethod]
    2018-04-24 10:08:03.486516+0800 FrameworkExample[9845:3271122] -[StaticLibraryTwo StaticLibraryTwoMethod]
    

结论: staticLibrary嵌套使用, 会将link所有内容, 这也是我们之前使用得最多的一种嵌套方式了

4. StaticLibrary嵌套使用StaticFramework
  • StaticFrameworkOne 加入到StaticLibraryOne并调用其方法

    - (void)StaticLibraryOneMethod{
        NSLog(@"%s",__func__);
        
        StaticFrameworkOne *one = [[StaticFrameworkOne alloc] init];
        [one staticFrameworkOneTestMethod];
    }
    
  • 编译StaticFrameworkOne, 并将其framework加入demo工程, 并调用其方法:

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        StaticLibraryOne *one = [[StaticLibraryOne alloc] init];
        [one StaticLibraryOneMethod];
    }
    
  • 运行demo工程 报错:


    sladdsf.png
  • StaticFrameworkOne加入demo工程, 运行日志输出:

    2018-04-24 10:18:35.474917+0800 FrameworkExample[10117:3287409] -[StaticLibraryOne StaticLibraryOneMethod]
    2018-04-24 10:18:35.475168+0800 FrameworkExample[10117:3287409] -[StaticFrameworkOneTest staticFrameworkOneTestMethod]
    

结论: staticLibrary嵌套使用staticFramework, 不会将staticFramework的内容link进来, 外部使用是需要再单独link staticFramework的

5. DynamicFramework嵌套使用StaticLibrary
  • 创建叫DynamicFrameworkOne的DynamicFramework, 添加StaticLibraryOne,并实现方法:

    - (void)dynamicFrameworkOneTestMethod{
        NSLog(@"%s",__func__);
        
        StaticLibraryOne *one = [[StaticLibraryOne alloc] init];
        [one StaticLibraryOneMethod];
    }
    
  • 编译DynamicFrameworkOne, 并将其添加到demo工程调用其方法:

    - (void)viewDidLoad {
    [super viewDidLoad];
        
    DynamicFrameworkOneTest *one = [[DynamicFrameworkOneTest alloc] init];
    [one dynamicFrameworkOneTestMethod];
    }
    
  • 日志输出:

    2018-04-24 11:05:01.424460+0800 FrameworkExample[10812:3360190] -[DynamicFrameworkOneTest dynamicFrameworkOneTestMethod]
    2018-04-24 11:05:01.424644+0800 FrameworkExample[10812:3360190] -[StaticLibraryOne StaticLibraryOneMethod]
    

结论: dynamicFramework中嵌套使用StaticLibrary, 会将StaticLibrary中的内容也link到dynamicFramework内部

6. DynamicFramework嵌套使用StaticFramework
  • DynamicFrameworkOne中添加StaticFrameworkOne,并实现方法:

    - (void)dynamicFrameworkOneTestMethod{
        NSLog(@"%s",__func__);
        
        StaticFrameworkOneTest *one = [[StaticFrameworkOneTest alloc] init];
        [one staticFrameworkOneTestMethod];
    }
    
  • 编译DynamicFrameworkOne, 并将其添加到demo工程调用其方法:

    - (void)viewDidLoad {
    [super viewDidLoad];
        
    DynamicFrameworkOneTest *one = [[DynamicFrameworkOneTest alloc] init];
    [one dynamicFrameworkOneTestMethod];
    }
    
  • 日志输出:

    2018-04-24 11:15:01.080553+0800 FrameworkExample[10984:3372929] -[DynamicFrameworkOneTest dynamicFrameworkOneTestMethod]
    

2018-04-24 11:15:01.080731+0800 FrameworkExample[10984:3372929] -[StaticFrameworkOneTest staticFrameworkOneTestMethod]
```

结论: dynamicFramework中嵌套使用staticFramework, 会将staticFramework中的内容也link到dynamicFramework内部

7. DynamicFramework嵌套使用DynamicFramework
  • 创建叫DynamicFrameworkTwo的DynamicFramework,并实现方法:

    - (void)dynamicFrameworkTwoTestMethod{
    NSLog(@"%s",__func__);
    }
    
  • DynamicFrameworkTwo添加到DynamicFrameworkOne,中并调用其方法:

    - (void)dynamicFrameworkOneTestMethod{
        NSLog(@"%s",__func__);
        
        DynamicFrameworkTwoTest *two = [[DynamicFrameworkTwoTest alloc] init];
        [two dynamicFrameworkTwoTestMethod];
    }
    
  • 编译并将framework添加到demo工程, 运行时发生奔溃:

    dyld: Library not loaded: @rpath/DynamicFrameworkTwo.framework/DynamicFrameworkTwo
      Referenced from: /Users/null/Library/Developer/CoreSimulator/Devices/4E06CF70-83F3-4145-BEB9-87817EE0C0F2/data/Containers/Bundle/Application/26EC93DF-56A9-4663-8274-A54E6043E0C1/FrameworkExample.app/Frameworks/DynamicFrameworkOne.framework/DynamicFrameworkOne
      Reason: image not found
    
  • 再将DynamicFrameworkTwoframework添加进demo工程 运行,输出日志:

    2018-04-24 11:50:47.631977+0800 FrameworkExample[11533:3417578] -[DynamicFrameworkOneTest dynamicFrameworkOneTestMethod]
    2018-04-24 11:50:47.632169+0800 FrameworkExample[11533:3417578] -[DynamicFrameworkTwoTest dynamicFrameworkTwoTestMethod]
    

推论: DynamicFramework嵌套使用,不会link内部的DynamicFramework, 外部使用的时候需要单独link内部使用的DynamicFramework
? : 查看具体二进制文件, 看其内部是否有对应的内部DynamicFramework文件

结语:

  1. 还有一些嵌套方式后续补充(staticLibrary嵌套DynamicFramework....)
  2. 具体是否真的未link内部库文件, 需要通过打开二进制文件来校验
  3. 是否有什么配置可以修改各种库嵌套事的link逻辑?

推荐阅读更多精彩内容