iOS链接库的冲突

December 09, 2023
测试
测试
测试
测试
4 分钟阅读

最近在打包的时候,遇到一个坑。特此记录一下

起因是发现 Unity 5.4 版本,使用c#写的下载,下载速度无法突破 2M/s,同样的网络,后来横向对比使用原来 Cocos2d 开始的游戏,可以达到 7M/s。海外推广一般是小包(iOS是150M以内,安卓50M以内) + 扩展包,如果下载速度过慢,对市场推广和用户转化都会有影响(下载等待时间变长了)。然后就决定基于 libcurl 写了一个C++的下载模块,以替换现有的 C# 下载模块。

韩国版本在添加了下载模块进行测试时,遇到了崩溃。使用 Release Run 时,在下载补丁那里必崩(编译是正常的),崩溃的地方是在 curl_easy_init 就挂掉了,很神奇。

最初怀疑是库的添加顺序,尝试调整顺序后发现不是。然后使用排除法,删相关的库,发现有一个库看上去有点奇怪,解压出来发现里面引用了 OpenSSL库,而我们的下载模块也用到了这个库。到这里原因就找到了,因为.a静态库与 第三方的 framework 有冲突(我们使用4个.a文件)。解决方案就是使用 .framework动态库,将4个.a整合成一个动态的.framework,然后添加到Xcode里。

问题解决后,在网上搜索了相关的资料。把相关的知识再重新梳理一遍。

库(Library)分为:静态链接库、动态链接库,下面将三大主流平台动态、静态库做一个简单的对比。

平台

静态库

动态库

Windows

*.lib

*.dll

Linux

*.a

*.so

Mac OS

*.a,*.framework

*.dylib,*.tbd,*.framework

你可能会奇怪,为啥 *.framework 既是动态库又是动态库,系统的 framework 是动态库(Real Framework),我们建立的 framework 是静态库(或者称为伪动态库 —— Fake Framework)。

下面是二个链接,使用 Xcode  build 出 iOS 可使用的 Framework

https://github.com/kstenerud/iOS-Universal-Framework

https://github.com/jverkoey/iOS-Framework

iOS 系统的 UIKit.framework 不需要拷贝到目标程序中,我们生成的 Framework 即便是动态的,最后还是需要拷贝到App中,因此苹果又把这种 framework 称为 Embedded Framework。

为什么这么修改就可以了呢?  iOS中的Embedded Framework可以理解为独立的没有main函数的可执行文件。这样就避免冲突了

注:需要注意的是 iOS 8之前是不支持动态库的,只支持静态库。

image
image

否则,会报错。

dyld: Library not loaded: @rpath/xx.framework/xx Referenced from: /var/containers/Bundle/Application/xxx/xx.app/app Reason: image not found

每次手工将 framework 拖到 xcode肯定是很蛋疼的,增好 Unity 提供了 Xcode API 来解决这个问题。

使用  PBXProjectExtensions 类的 AddFileToEmbedFrameworks 方法

https://docs.unity3d.com/ScriptReference/iOS.Xcode.Extensions.PBXProjectExtensions.html

using UnityEditor.iOS.Xcode.Custom;

string targetName = PBXProject.GetUnityTargetName();
PBXProject proj = new PBXProject();
proj.ReadFromString(File.ReadAllText(projPath));
string target = proj.TargetGuidByName(targetName);

string strOSVersion = PlayerSettings.iOS.targetOSVersionString;
string strMajorVersion = strOSVersion.Split('.')[0];
if (int.Parse(strMajorVersion) >= 8)
{
    proj.AddFileToBuild(target, proj.AddFile("Frameworks/xx.framework", "Frameworks/xx.framework", PBXSourceTree.Source));    

    const string defaultLocationInProj = "Frameworks/xx/";
    const string coreFrameworkName = "xx.framework";

    string framework = Path.Combine(defaultLocationInProj, coreFrameworkName);
    string fileGuid = proj.AddFile(framework, "Frameworks/" + framework, PBXSourceTree.Sdk);

    PBXProjectExtensions.AddFileToEmbedFrameworks(proj, target, fileGuid);
}

更多参考链接:

iOS静态库的链接与加载

iOS动态库、静态库及使用场景、方式

继续阅读

更多来自我们博客的帖子

如何安装 BuddyPress
由 测试 December 17, 2023
经过差不多一年的开发,BuddyPress 这个基于 WordPress Mu 的 SNS 插件正式版终于发布了。BuddyPress...
阅读更多
Filter如何工作
由 测试 December 17, 2023
在 web.xml...
阅读更多
如何理解CGAffineTransform
由 测试 December 17, 2023
CGAffineTransform A structure for holding an affine transformation matrix. ...
阅读更多