iOS:动态库

1.动态库原理

1.1自己生成动态库(失败)

我们还是用在静态库中的TestExample案例先生成动态库

image.png

为了方便我们使用创建build.sh,来使用脚本

clang -target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-I./dylib \
-c test.m -o test.o

pushd ./dylib
echo "编译TestExample.m --- TestExample.o"
clang -target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-c TestExample.m -o TestExample.o
clang -dynamiclib \
-target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
TestExample.o -o libTestExample.dylib
echo "编译TestExample.m --- libTestExample.dylib"
clang -target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-L./dylib \
-lTestExample \
test.o -o test

运行后报错还是和上面一样 dyld: Library not loaded: libTestExample.dylib

1.2.先生成静态库再链接成动态库

//把我们的o文件变成a文件。也可以使用ar -rc a.a a.o
libtool -static -arch_only x86_64 TestExample.o -o libTestExample.a

ld -dylib -arch x86_64 \
-macosx_version_min 11.0 \
-syslibroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-lsystem -framework Foundation \
-all_load \  //如果没有这个参数,链接器会默认not all load
libTestExample.a -o libTestExample.dylib

运行之后还是报错

Process 25266 launched: '/Users/MacW/Desktop/loginlearn/强化版/强化班-4-动态库/上课代码/动态库原理/test' (x86_64)
dyld: Library not loaded: libTestExample.dylib
  Referenced from: /Users/MacW/Desktop/loginlearn/强化版/强化班-4-动态库/上课代码/动态库原理/test
  Reason: image not found

错误显示库没有加载进来
我们查看test加载所需要的动态库

otool -l test | grep 'DYLIB' -A 5
//日志如下
 cmd LC_LOAD_DYLIB
      cmdsize 48
         name libTestExample.dylib (offset 24)
   time stamp 2 Thu Jan  1 08:00:02 1970
      current version 0.0.0
compatibility version 0.0.0
--
          cmd LC_LOAD_DYLIB
      cmdsize 96
         name /System/Library/Frameworks/Foundation.framework/Versions/C/Foundation (offset 24)
   time stamp 2 Thu Jan  1 08:00:02 1970
      current version 1770.106.0
compatibility version 300.0.0
--
          cmd LC_LOAD_DYLIB
      cmdsize 56
         name /usr/lib/libobjc.A.dylib (offset 24)
   time stamp 2 Thu Jan  1 08:00:02 1970
      current version 228.0.0
compatibility version 1.0.0
--
          cmd LC_LOAD_DYLIB
      cmdsize 56
         name /usr/lib/libSystem.B.dylib (offset 24)
   time stamp 2 Thu Jan  1 08:00:02 1970
      current version 1292.0.0
compatibility version 1.0.0
--
          cmd LC_LOAD_DYLIB
      cmdsize 104
         name /System/Library/Frameworks/CoreFoundation.framework/Versions/A/CoreFoundation (offset 24)
   time stamp 2 Thu Jan  1 08:00:02 1970
      current version 1770.106.0
compatibility version 150.0.0

我们看到我们自己创建的库 name libTestExample.dylib (offset 24),而其他的动态库的name确实库的路径。
我们的动态库能够自己保存自己的路径

otool -l ./dylib/libTestExample.dylib | grep 'ID' -A 5
//-A是向下寻找5行,-B是向上查找
 cmd LC_ID_DYLIB
      cmdsize 48
         name libTestExample.dylib (offset 24)
   time stamp 1 Thu Jan  1 08:00:01 1970
      current version 0.0.0
compatibility version 0.0.0
--
     cmd LC_UUID
 cmdsize 24
    uuid F26CDB5B-F71B-3658-86BE-C93CBE279ABE
Load command 9
       cmd LC_BUILD_VERSION
   cmdsize 32

我们看到name 还是 libTestExample.dylib
我们可以使用install_name_tool去给动态库添加路径

man install_name_tool
NAME
       install_name_tool - change dynamic shared library install names

install_name_tool就是改变动态的路径。
给动态路添加路径

install_name_tool -id /Users/MacW/Desktop/loginlearn/强化版/强化班-4-动态库/上课代码/动态库原理/dylib/libTestExample.dylib  libTestExample.dylib

重新查看动态库的LC_ID_DYLIB

otool -l libTestExample.dylib | grep 'ID' -A 5
 cmd LC_ID_DYLIB
      cmdsize 144
         name /Users/MacW/Desktop/loginlearn/强化版/强化班-4-动态库/上课代码/动态库原理/dylib/libTestExample.dylib (offset 24)
   time stamp 1611538070 Mon Jan 25 09:27:50 2021
      current version 0.0.0
compatibility version 0.0.0
--
     cmd LC_UUID
 cmdsize 24
    uuid F26CDB5B-F71B-3658-86BE-C93CBE279ABE
Load command 9
       cmd LC_BUILD_VERSION
   cmdsize 32

然后重新编译和链接我们的test文件运行成功
我们通过install_name_tool,重新给编译过的动态库加的路径,其实ld提供给我们一个参数可以直接加的install_name
所以我们初期的sh脚本如下

clang -target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-I./dylib \
-c test.m -o test.o

pushd ./dylib
echo "编译TestExample.m --- TestExample.o"
clang -target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-c TestExample.m -o TestExample.o

echo "编译TestExample.o --- libTestExample.a"

# Xcode->静态库
libtool -static -arch_only x86_64 TestExample.o -o libTestExample.a
# -dynamiclib: 动态库
# dylib 最终链接产物 -》
ld -dylib -arch x86_64 \
-macosx_version_min 11.0 \
-syslibroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-lsystem -framework Foundation \
-install_name /Users/MacW/Desktop/loginlearn/强化版/强化班-4-动态库/上课代码/动态库原理/dylib/libTestExample.dylib \
-all_load \
libTestExample.a -o libTestExample.dylib
popd
echo "链接libTestExample.dylib -- test EXEC"
clang -target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-L./dylib \
-lTestExample \
test.o -o test

运行没问题

2.动态库复用性

我们在上面指定的路径是我们的绝对路径,如果动态库换了位置,很明显还是会出现之前出现的问题, Reason: image not found

2.1@rpath解决绝对路径的困扰

@rpath
Runpath search Paths
dyld搜索路径
运行时@rpath指示dyld按顺序搜索路径列表,以找到动态库。
@rpath保存一个或多个路径的变量
所以我们修改脚本

-install_name @rpath/dylib/libTestExample.dylib

运行还是报错,为什么呢?我们可以查看test文件是否有rpath的设置

2.2 rpath

otool -l  test | grep 'RPATH' -A 5

我们通过install_name_tool给test添加rpath路径

 install_name_tool -add_rpath /Users/MacW/Desktop/loginlearn/强化版/强 化班-4-动态库/上课代码/动态库原理 test

正常运行。但是add_rpath也是我们手动添加的,没有可复用性。

2.3.executable_path和loader_path

@executable_path:表示可执行程序所在的目录,解析为可执行文件的绝对路径。
@loader_path:表示被加载的Mach-O 所在的目录,每次加载时,都可能 被设置为不同的路径,由上层指定。

install_name_tool -rpath /Users/MacW/Desktop/loginlearn/强化版/强化班-4-动态库/上课代码/动态库原理  @executable_path test

运行test成功

3.动态库链接动态库

test.m文件使用Frameworks中的TestExample.framework
在TestExample.framework库中使用TestExampleLog.framework

3.1先编译并链接TestExampleLog.framework

clang -target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-I./Headers \
-c TestExampleLog.m -o TestExampleLog.o

clang -dynamiclib  \
-target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-Xlinker -install_name -Xlinker @rpath/TestExampleLog.framework/TestExampleLog \
TestExampleLog.o -o TestExampleLog

install_name是提供给外部需要链接TestExampleLog的路径。
通过查看otool -l TestExampleLog | grep 'ID' -A 5能看出外部需要链接的路径

          cmd LC_ID_DYLIB
      cmdsize 72
         name @rpath/TestExampleLog.framework/TestExampleLog (offset 24)
   time stamp 1 Thu Jan  1 08:00:01 1970
      current version 0.0.0
compatibility version 0.0.0
--
     cmd LC_UUID
 cmdsize 24
    uuid 73F8F987-0378-346D-B014-890611B76871
Load command 9
       cmd LC_BUILD_VERSION
   cmdsize 32

3.2.再编译链接TestExample.framework

clang -target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-I./Headers \
-I./Frameworks/TestExampleLog.framework/Headers \
-c TestExample.m -o TestExample.o

clang -dynamiclib  \
-target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-Xlinker -install_name -Xlinker @rpath/TestExample.framework/TestExample \
-Xlinker -rpath -Xlinker @loader_path/Frameworks \
-F./Frameworks \
-framework TestExampleLog \
TestExample.o -o TestExample


echo "-------DYLIB---------"
otool -l TestExample | grep 'DYLIB' -A 5
echo "-------ID---------"
otool -l TestExample | grep 'ID' -A 5

install_name是提供给调用者加载的路径
rpath是提供给调用TestExampleLog的初始路径

3.3编译并链接test

clang -target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-I./Frameworks/TestExample.framework/Headers \
-c test.m -o test.o

clang   \
-target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-Xlinker -rpath -Xlinker @executable_path/Frameworks \
-F./Frameworks \
-framework TestExample \
test.o -o test

运行正常打印

3.4.test文件怎么使用TestExampleLog

如果我们想使用TestExampleLog,我们需要能再TestExample中导出符号中能看到TestExampleLog

objdump --macho --exports-trie TestExample
TestExample:
Exports trie:
0x000080C0  _OBJC_METACLASS_$_TestExample
0x000080E8  _OBJC_CLASS_$_TestExample

我们没有看出有导出符号,所以我们无法在test使用TestExampleLog,我们需要使用reexport_framework
我们修改TestExample内的脚本

clang -dynamiclib  \
-target x86_64-apple-macos11.0.1 \
-fobjc-arc \
-isysroot /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX11.0.sdk \
-Xlinker -install_name -Xlinker @rpath/TestExample.framework/TestExample \
-Xlinker -rpath -Xlinker @loader_path/Frameworks \
-Xlinker -reexport_framework -Xlinker TestExampleLog \
-F./Frameworks \
-framework TestExampleLog \
TestExample.o -o TestExample

查看DYLIIB信息

 otool -l TestExample | grep 'DYLIB' -A 5
  cmd LC_ID_DYLIB
      cmdsize 72
         name @rpath/TestExample.framework/TestExample (offset 24)
   time stamp 1 Thu Jan  1 08:00:01 1970
      current version 0.0.0
compatibility version 0.0.0
--
          cmd LC_REEXPORT_DYLIB
      cmdsize 72
         name @rpath/TestExampleLog.framework/TestExampleLog (offset 24)
   time stamp 2 Thu Jan  1 08:00:02 1970
      current version 0.0.0
compatibility version 0.0.0

可以知道TestExample重新加一个cmd,LC_REEXPORT_DYLIB
我们在test中导入头文件

#import <Foundation/Foundation.h>
#import "TestExample.h"
#import "TestExampleLog.h"

int main(){
    NSLog(@"testApp----");
    TestExample *manager = [TestExample new];
    [manager lg_test: nil];
    TestExampleLog *log = [TestExampleLog new];
    NSLog(@"testApp----%@",log);
    return 0;
}

我们的脚本文件也需要-I头文件,重新运行脚本,正常打印。

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

推荐阅读更多精彩内容