ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考

这篇具有很好参考价值的文章主要介绍了ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、前言

在应用RN开发跨平台APP阶段,从git中拉取项目,应用Jenkins进行组包时,发现最终生成的ipa安装包版本号始终与项目中设置的版本号不一致。

二、问题描述

经过仔细排查,发现Jenkins在Archive编译、归档阶段失败,但是后续Export阶段生成了ipa包。

error: Multiple commands produce '/Users/xxx/Library/Developer/Xcode/DerivedData/xxx-eomylkmdzkgaughihoblturddotc/Build/Products/Debug-iphonesimulator/PopNews.app':

1) Target 'xxx' has create directory command with output '/Users/xxx/Library/Developer/Xcode/DerivedData/xxx-eomylkmdzkgaughihoblturddotc/Build/Products/Debug-iphonesimulator/PopNews.app'

2) That command depends on command in Target 'xxx': script phase “[CP] Copy Pods Resources”

三、问题解决

  1. 在xCode中打开应用,选中项目 target -> Build phase -> Copy Pods Resources -> Output Files -> 移除 ${TARGET_BUILD_DIR}/${UNLOCALIZED_RESOURCES_FOLDER_PATH} /*

  2. 将项目源码压缩上传至Jenkins预设指定目录,进行流水线组包作业。

四、拓展阅读

4.1 版本号查看

androidandroid/app/build.gradle文件中版本设置如下:

android {
    compileSdkVersion rootProject.ext.compileSdkVersion

    compileOptions {
        sourceCompatibility JavaVersion.VERSION_1_8
        targetCompatibility JavaVersion.VERSION_1_8
    }

    defaultConfig {
        applicationId "com.china.shq5785"
        minSdkVersion rootProject.ext.minSdkVersion
        targetSdkVersion rootProject.ext.targetSdkVersion
        versionCode 18072801
        versionName "2.2.5"
        multiDexEnabled true
        testBuildType System.getProperty('testBuildType', 'debug')
        // This will later be used to control the test apk build type
        testInstrumentationRunner 'androidx.test.runner.AndroidJUnitRunner'
        ndk {
            //设置支持的SO库架构
            abiFilters "armeabi", "armeabi-v7a", "x86_64" //, "arm64-v8a"
        }
         missingDimensionStrategy 'react-native-camera', 'general'
    }
    ......
}

ios 在配置文件ios/mrcs.xcodeproj/project.pbxproj中,可查看到如下配置信息:

13B07F941A680F5B00A75B9A /* Debug */ = {
			isa = XCBuildConfiguration;
			baseConfigurationReference = AA6AA411A14368FB4EEC0CD3 /* Pods-mrcs.debug.xcconfig */;
			buildSettings = {
				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
				CLANG_ENABLE_MODULES = YES;
				CODE_SIGN_IDENTITY = "iPhone Distribution";
				CODE_SIGN_STYLE = Manual;
				CURRENT_PROJECT_VERSION = 1;
				DEAD_CODE_STRIPPING = NO;
				DEVELOPMENT_TEAM = U4ALRF5A38;
				ENABLE_BITCODE = NO;
				FRAMEWORK_SEARCH_PATHS = (
					"$(inherited)",
					"$(PROJECT_DIR)/shq5785",
					"$(PROJECT_DIR)",
				);
				GCC_PREFIX_HEADER = shq5785/PrefixHeader.pch;
				GCC_WARN_ABOUT_RETURN_TYPE = NO;
				HEADER_SEARCH_PATHS = "$(inherited)";
				INFOPLIST_FILE = shq5785/Info.plist;
				IPHONEOS_DEPLOYMENT_TARGET = 9.0;
				LD_RUNPATH_SEARCH_PATHS = "$(inherited) @executable_path/Frameworks";
				LIBRARY_SEARCH_PATHS = (
					"$(inherited)",
					"$(PROJECT_DIR)/shq5785",
				);
				MARKETING_VERSION = 2.2.5;
				OTHER_CODE_SIGN_FLAGS = "--deep";
				OTHER_LDFLAGS = (
					"$(inherited)",
					"-ObjC",
					"-lc++",
				);
				PRODUCT_BUNDLE_IDENTIFIER = com.china.shq5785;
				PRODUCT_NAME = shq5785;
				PROVISIONING_PROFILE_SPECIFIER = "1111";
				SWIFT_OBJC_BRIDGING_HEADER = "$(PRODUCT_MODULE_NAME)/shq5785-Bridging-Header.h";
				SWIFT_OPTIMIZATION_LEVEL = "-Onone";
				SWIFT_VERSION = 5.0;
				VERSIONING_SYSTEM = "apple-generic";
			};
			name = Debug;
		};

4.2 ipa包生成过程

iOS 开发的最后一步就是进行 App 的打包和分发,这里分为两个步骤:

  1. Archive:对 Target 进行编译、归档,生成 .xcarchive 文件。

  2. Export:对 .xcarchive 归档文件进一步处理,生成不同渠道的 .ipa 包,进行分发。

当我们在 Xcode 菜单中选择 Product -> Archive 后,编译系统就会对当前的 Xcode 工程进行分析、编译和打包,最终生成目标 Target 的一个 Archive(归档),我们可以在 Window -> Organizer -> Archives 页面查看到所有缓存的历史归档信息:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript
所谓的”归档“,就是对源码进行编译后,将此次编译生成的各种文件、资源、记录统一封装到一个地方,方便进行管理和回溯。

右键选择一个归档文件 archive,然后点击 Show in Finder,可以看到它在 Finder 中表示为一个 .xcarchive 后缀的文件。

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

这个 .xcarchive 文件包含了应用和它的符号表信息(symbol information)以及其它的相关资源,右键选择显示包内容,可以查看一个 Archive 归档中具体的文件结构:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript
其中每个文件夹的含义:

  • BCSymbolMaps
    Xcode 对 BitCode 符号表进行混淆(Symbol Hiding)后生成的对照表,和 dSYM 文件会一一对应。

  • dSYMs
    存储此次编译的符号表(debug symbols),用来符号化解析崩溃堆栈。

  • Products
    存储此次编译生成的的 App 包(.app)。

要注意的是这个包虽然包括了 App 运行需要的可执行文件以及其它资源,但是和最终用户下载的版本会有所不同。后续的 export 操作会对其进行进一步处理。

  • SCMBlueprint
    如果 Xcode 打开了版本管理(Preferences -> Source Control -> Enable Source Control),SCMBlueprint 文件夹会存储此次编译的版本控制信息,包括使用的 git 版本、仓库、分支等。

如果未来想要回溯此次编译的源码版本,可以从这个 SCMBlueprint 中找到必要的信息。

  • SwiftSupport
    如果在 Target 的 Build Settings 中打开了ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES,此次编译使用的 Swift 版本对应的标准库文件(.dylib)会被放到这个文件夹中。

发布 App 时,这些标准库也会被复制到 ipa bundle 中。

不过现在 Swift 的 ABI 已经稳定了,Xcode 10.2 及以后的版本打出来的包,在 iOS 12.2 及以后的系统的 app bundle 中不用再自带链接库了,节省了一定的体积。

了解 ipa 文件
.ipa(iOS App Store Package) 文件是最终被安装到 iPhone 上的应用格式,包含了运行 App 所必需的的签名、二进制包、资源等内容。

在 Organizer 中无论用什么方式 export 应用的安装包,最终生成的都是一个 .ipa 文件。

.ipa 本身是压缩包文件,如果要查看 ipa 中的内容,可以右键查看包内容,观察解压以后的包,主要包含以下内容:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript
App 的签名信息会被放到 _CodeSignature 文件夹中。

  • info.plist
    存储 App 主要信息的 plist 文件也会被一并打包到 ipa 中。

  • entitlements
    entitlement 直译成中文是“权益”、“权限”的意思。

当你在 Capabilities 中开启一些特定的权限时,Xcode 会自动给你生成一个.entitlements 文件,在这个文件中通过 xml 的格式将这些授权记录下来。

App 瘦身
要对 App 安装包体积进行压缩,首先要知道安装包占用的多少空间,这些空间由哪些部分组成,然后再进行针对性的优化。

查看最终用户安装包大小
实际上在 Xcode 本地 archive 出来的 app 包或者 export 出来的 ipa 包和最终用户下载的版本会有所不同(通常体积会大很多)。因为苹果可能会对 App 进行重新编译(如果上传了 BitCode),也会针对不同的设备型号、iOS 版本分发不同的资源(比如 2x、3x 的图片),最后还会对整个 .ipa 进行压缩,以减少从 App Store 下载时耗费的流量。

那么如何估算用户最终下载版本的包体积大小呢?其实在 iTunes Connect 页面可以直接查询到。

打开 iTunes Connect,选择 我的App -> 活动 -> 所有构建版本,然后选择一个要查看的版本:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript
找到 App Store 文件大小按钮:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

在弹出的列表中,可以看到在最新版本的 iOS 系统下,不同设备下载的包体积大小:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

列表中的两列:

  • 下载大小:表示通过无线下载的压缩 App 大小;

  • 安装大小:安装后此 App 将在用户设备上占用的磁盘空间大小;

如何分析 App 包 Size?
为了更直观地查看哪些资源占用了 App 安装包的体积,我们可以借助一些文件工具来分析解压后的 ipa 包,比如说 derlien

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

可以很直观地看到各种不同类型文件所占的比例。

检查未使用资源
随着 App 的不断迭代,我们往往会无意间引入很多用不到的资源,或者一些资源的引用已经从代码中去除了,但是没有及时从 bundle 中删除,造成 App 包体积的浪费。

为了查找这些不再使用的资源,可以借助开源工具 LSUnusedResources 来检测整个工程。

LSUnusedResources 应用过程如下:

  1. 可以从下面的地址下载 LSUnusedResources
    源码,然后进行编译…

  2. 将源码在Mac上运行,可以看到如下界面:ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript在Project> Path目录中,点击Browse…选择要检测工程的根目录,然后点击Search,开始进行检索…,你可以在下方的日志窗中看到检测结果>

  3. 检测完成后,可以点击Export将此日志导出,然后开始进行清理工作.切勿不管三七二十一直接开删,毕竟是机器检测,不可完全信赖。

针对一些特殊情况,比如代码中使用例如 [UIImage imageNamed:[NSString stringWithFormat:@"icon_tag_%d", index]] 的方式引用资源,LSUnusedResources 也支持使用正则表达式来模糊匹配。

压缩图片
图片文件是安装包中最常见的资源了,常常会占有相当一部分比例,未压缩的图片体积往往相当大,通过一些工具压缩图片资源,节省空间:

  • 无损压缩:ImageOptim

  • 有损压缩:tinypng

使用 Asset Catalogs 存储资源
相比于直接将图片拖入工程目录的方式,使用 Asset Catalogs 会更节省体积。Asset Catalogs 会用一个高度优化的特殊格式来存所有图片,对 png 图片也会进行最大化的压缩。

Xcode 工程模板会自动生成一个 Assets.xcassets 文件,我们也可以按需创建另外的 .xcassets,最终在 ipa 包中,这些 xcassets 都会被压缩到 Assets.car 文件中,一定程度上也保证了安全性。

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

除了图片资源外,Asset Catalogs 也可以存储文本、Data 甚至 AR、apple TV 相关的资源,非常全能,所以比较好的实践就是:

能用 Asset Catalogs 管理的资源,尽量使用 Asset Catalogs 来管理

分析 LinkMap 文件
上面提到,App 包占用空间中很大一部分比例是最终编译生成的可执行文件(MACH-O),可执行文件的大小不仅和代码体积有关,也受编译器版本、编译选项、链接库、目标架构等影响。

可以通过分析编译时产生的 LinkMap 来了解 MACH-O 文件的组成部分。

要找到对应的 LinkMap,首先在 Xcode Target -> Build Settings -> Write Link Map File 设置为 YES,然后在 Target -> Build Settings -> Path to Link Map File 选项中设置好 LinkMap 的生成地址(一般用 build 文件夹中的默认地址就好了),archive 成功后,我们就可以在对应地址找到该次编译的 LinkMap 了:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

LinkMap 记录了编译时的链接信息,用来描述可执行文件的构造成分,包括代码段__TEXT 和数据段 __DATA 的分布情况:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript
网上有很多脚本可以对 LinkMap 进行分析统计,比如:

可视化工具

  • js脚本

  • 命令行工具

获取到分析结果后,可以精确了解各个模块、链接库、方法在可执行文件中的位置和占用空间:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

对于一些占比特别大的模块,常见的优化思路有:

寻找可替代的,小体积的依赖库,或者自己实现

去掉静态库中不需要的指令集,比如 armv7s,x86等,只保留发布需要的 armv7,arm64

提高代码重用性

进一步分析代码中没有被使用的方法、模块,对代码库进行精简。

使用 bitcode
bitcode 是在 LLVM 体系中介于前端语言(OC、Swift、C)和后端语言(X86、ARM的机器码)之间的中间语言。

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

一次完整的编译(从源码到.O目标文件)包含三个主要步骤:

  • 前端(Frontend):负责把各种类型的源代码编译为 bitcode 中间码表示。

  • 优化(Optimizer):负责对 bitcode 进行各种类型的优化,将 bitcode 代码进行一些逻辑等价的转换,使得代码的执行效率更高,体积更小。

  • 后端(Backend):也叫 CodeGenerator,负责把优化后的 bitcode 编译为指定目标架构的机器码,比如 x86、arm64 等等。

可以在 Xcode Target -> Build Settings -> Enable Bitcode 中打开 bitcode 选项,这样在 archive 时,会将中间生成的 bitcode 嵌入到链接后的二进制文件(.o)中,用于提交到 App Store。

上面提到,bitcode 作为 LLVM 的中间语言,是可以从它直接编译出最终程序的,Apple 拿到我们上传的 bitcode 后,会使用最新的技术、编译器针对不同的终端设备重新编译 App,而这些重新编译的版本往往比本地 Xcode 编译的版本体积更小、效率更高。

如果后续需要支持新的平台或者有新的编译技术革新,苹果就不用依赖开发者重新上传了,直接使用现成的 bitcode 编译出新的版本。

值得注意的是:在打包时,如果一些三方的依赖库没有开启 bitcode,或者开启了但是没有在最终引用的链接库中带有 bitcode,那么整个工程就无法用 bitcode 来编译了。

按需加载资源(On-Demand Resources)
iOS9 以后,苹果提供了 On-Demand Resources 功能来减少安装包的体积。可以将一些资源标记为 “按需加载”,在需要使用的时候请求操作系统从 App Store 中下载。这个功能非常适合一些大型游戏、带有付费内容或者大量不常使用的多媒体资源的 App。

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

当然,按需加载只是针对 App 使用的资源文件,不包括二进制可执行文件或者源码。

On-Demand Resources 的配置可以很轻松地在 Xcode 中完成。

首先在 Target -> Resource Tags 中创建资源 tag,一个 tag 表示一组可以被独立下载的资源,后面我们就会使用这个 tag 在程序中请求操作系统下载对应的资源包到本地。

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

不同的 tag 包含的资源是可以重复的,App Store 会自己 differ,不会重复下载。

然后找到想要按需加载的资源文件,为它们分配一个或多个之前创建的 tag。

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

最后在代码中,可以使用 NSBundleResourceRequest

  • 请求下载 on-demand 资源;

  • 将资源标记为已使用状态(这样下载的资源会被清理掉,节省本地空间);

  • 管理资源下载过程,配置优先级、追踪下载进度等等;

  • 检测磁盘容量警告;

下面的代码是一个简单的资源下载请求:

// 配置要下载的 tags
NSSet *tags = [NSSet setWithObjects: @"birds", @"bridge", @"city"];// 创建 NSBundleResourceRequest 对象
resourceRequest = [[NSBundleResourceRequest alloc] initWithTags:tags];// 请求资源,处理回调
[resourceRequest beginAccessingResourcesWithCompletionHandler: ^(NSError * __nullable error) {if (error) {// 处理错误self.resourcesLoaded = NO;return;}// 下载成功,可以直接使用这些资源了self.resourcesAvailable = YES;}
];

下图总结了一个 on-demand 资源的生命周期:

ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考,# React Native,react native,react.js,javascript

题外话:苹果取消了移动网络下载 150M 的限制,说明随着手机容量的增加和移动网络的普及,大家对 App 安装包体积不再那么敏感了,只要我们遵循一些最佳实践,一般不会在这一块有太大的问题。文章来源地址https://www.toymoban.com/news/detail-659796.html

到了这里,关于ReactNative进阶(三十四):ipa Archive 阶段报错error: Multiple commands produce问题修复及思考的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 运行Dockerfile报错:ERROR: failed to solve: process “/bin/sh -c sed -ri ‘s#archive.ubuntu.com...exit code

    docker build 使用Dockerfile打包tengine的镜像 Dockerfile文件写错???– 检查没有问题 docker build命令执行问题???–检查没有问题

    2024年02月12日
    浏览(41)
  • ReactNative进阶(五十一)project.pbxproj 配置文件详解

    Xcode 作为日常开发iOS程序的IDE,支持 C 、 C++ 、 Objective-C 、 Swift 、 Ruby 等语言进行编写。日常开发入口就是 Xcode workspace 或者 Xcode project 。 workspace 是一个 Xcode 文档,它将项目和其他文件、project分组。一个 workspace 可以包含任意数量的 Xcode project ,以及资源文件(JSON、脚本、

    2024年02月04日
    浏览(33)
  • ReactNative进阶(二十一)开源插件 react-native-device-info 获取设备信息

    项目开发过程中,需要获取设备信息,例如获取设备名称。可通过使用开源的第三方组件react-native-device-info,该组件适用于 iOS 和 Android 双平台。 在 ReactNative 项目中可通过 npm 命令下载 react-native-device-info 组件依赖包: android 需要在 AndroidManifest.xml 配置文件添加 android.permiss

    2024年02月07日
    浏览(40)
  • Python工具箱系列(三十四)

    SQLAlchemy是著名的ORM(Object Relational Mapping-对象关系映射)框架。其主要作用是在编程中,把面向对象的概念跟数据库中表的概念对应起来。对许多语言(例如JAVA/PYTHON)来说就是定义一个对象,并且这个对象对应着一张数据库的表。而这个对象的实例,就对应着表中的一条记录。

    2024年02月06日
    浏览(51)
  • 第三十四课.模糊神经网络

    在本篇内容中,我们将了解模糊神经网络(Fuzzy Neural Network),在此之前,先了解模糊理论。现实世界总是充满不确定性。因此,在建模系统的时候,我们需要考虑这种不确定性。我们之前其实已经在概率论中接触过这种不确定的建模形式。类似于概率论,Zadeh开发了一种不同

    2023年04月23日
    浏览(43)
  • 数据库—数据库备份(三十四)

    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 目录 前言 一、概述 二、数据备份的重要性 三、造成数据丢失的原因 四、备份类型 4.1物理与逻辑角度 4.2数据库备份策略角度 五、常见的备份方法 5.1 物理备份 5.2 使用专用备份工具 5.3 通过启用二进制

    2024年02月15日
    浏览(46)
  • 第三十四章 Unity人形动画(上)

    在我们DirectX课程中,我们讲过一个模型最少拥有网格和材质,可以没有动画。游戏场景中的静态物体就可以是这样的模型,例如花草树木,建筑物等等,他们通过MeshRenderer就可以渲染。对于一个带有动画的FBX文件,里面不仅仅包含了网格和材质,还包括了“骨架”和动画数据

    2024年02月05日
    浏览(54)
  • 第四十天 Java基础学习(三十四)

    一、Tomcat的初步使用 ● Tomcat目录结构与介绍 •bin:用来存放Tomcat服务器的可执行程序,主要有两大类,一类是以.sh结尾的(linux命令),另一类是以.bat结尾的(windows命令) •conf:用来存放Tomcat服务器的配置文件 •lib:用来存放Tomcat服务器的jar包 •logs:用来存放Tomcat服务器运行

    2024年02月16日
    浏览(41)
  • 第三十四章 配置服务器访问

    本页介绍如何配置`` 网关连接的应用程序。对于这些配置任务,可以使用 Web Gateway 管理页面。其他文章介绍了如何配置默认设置和服务器。 每个 Web 应用程序都必须配置其 CSP 文件的路径。每个路径的配置标识负责运行应用程序的 IRIS 服务器。用于指定故障转移和负载平衡的

    2024年04月14日
    浏览(32)
  • C++学习笔记(三十四):c++ array

    本节介绍c++标准库模板中的array和c风格的array的区别,及什么时候使用std::array。

    2024年01月23日
    浏览(50)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包