如何快速用上Swift静态库详解

前言

Swift 支持静态库打包已经有一段时间了,CocoaPods 也提供了 static_framework 参数。然而大部分的第三方依赖都没有及时更新。

本文给出个相对方便一些的方案用上静态库,个人感觉在依赖不那么复杂的时候成本还是比较低的。

效果如下:

示例地址:UseStaticFramework

想办法为每一个 pod 添加 static_framework 是关键。

直接修改 podspec 不太现实,因为 CocoaPods 并没有提供相关的接口。但是当一个 pod 指定 podspec 地址时,这个 podspec 会被保存在本地。

如果 pod 没有更新,pod install 将直接从本地获取 pod 信息,这就为我们修改 pod 提供了可能。

target 'UseStaticFramework' do
 pod 'RxSwift', :git => 'https://github.com/ReactiveX/RxSwift.git'
end

pre_install do |installer|
 installer.sandbox.specifications_root.children.each do |podspec|
  if podspec.extname() == '.json'
   edit_pod_spec podspec
  end
 end
end

def edit_pod_spec(file)
 code = File.read(file)
 json = JSON.parse(code)
 json['static_framework'] = true
 File.write(file, JSON.generate(json))
end

在 Podfile 中添加以上代码,执行两次 bundle exec pod install 即可将依赖 RxSwift 变成静态库。相比单独建一个 Specs 方便很多了,特别是在 RxSwift 有更新时,我们也无需增加成本,执行 bundle exec pod update 即可。

有些依赖稍微麻烦些,比如 RxCocoa 。就目前来看,Swift 静态库似乎还不能混编,好在 RxCocoa 支持 SPM,在 SPM 中有一个 RxCocoaRuntime 依赖。

创建一个 RxCocoaRuntime.podspec 使用,再调整一下 RxCocoa 的 podspec 即可,注意添加 SWIFT_PACKAGE 编译标记:

pod 'RxCocoa', :git => 'https://github.com/ReactiveX/RxSwift.git'
pod 'RxCocoaRuntime', :podspec => 'https://raw.githubusercontent.com/DianQK/UseStaticFramework/master/RxCocoaRuntime.podspec'
def edit_pod_spec(file)
 code = File.read(file)
 json = JSON.parse(code)
 json['static_framework'] = true
 if json['name'] == 'RxCocoa'
  json['xcconfig'] = {
   :OTHER_SWIFT_FLAGS => '$(inherited) "-D" "SWIFT_PACKAGE"'
  }
  json['source_files'] = ['RxCocoa/RxCocoa.swift', 'RxCocoa/Common/**/*.{swift}', 'RxCocoa/Traits/**/*.{swift}', 'RxCocoa/Foundation/**/*.{swift}', 'RxCocoa/Runtime/**/*.{swift}', 'Platform/**/*.swift']
  json['preserve_paths'] = ['RxCocoa/RxCocoa.h', 'RxCocoa/*.swift', 'RxCocoa/Common/**/*.{swift,h,m}', 'RxCocoa/Traits/**/*.{swift,h,m}', 'RxCocoa/Foundation/**/*.{swift,h,m}', 'RxCocoa/Runtime/**/*.{swift,h,m}', 'Platform/**/*.swift']
  json['dependencies'] = {
   :RxSwift => '~> 4.1',
   :RxCocoaRuntime => '~> 4.1'
  }
 end
 File.write(file, JSON.generate(json))
end

执行两次 bundle exec pod install,完成。

Apollo 这种也能搞,稍微麻烦一些,有些代码没有引入 UIKit,最终导致按照上面的方案编译不过去。

pod 'SQLite.swift', :git => 'https://github.com/stephencelis/SQLite.swift.git'
pod 'SQLiteObjc', :podspec => 'https://raw.githubusercontent.com/DianQK/UseStaticFramework/master/SQLiteObjc.podspec'

pod 'Apollo', :git => 'https://github.com/apollographql/apollo-ios.git'
pod 'Apollo/SQLite', :git => 'https://github.com/apollographql/apollo-ios.git'
# edit_pod_spec
if json['name'] == 'SQLite.swift'
 json['xcconfig'] = {
  :OTHER_SWIFT_FLAGS => '$(inherited) "-D" "SWIFT_PACKAGE"'
 }
 json['dependencies'] = {
  :SQLiteObjc => '~> 0.11.4'
 }
 json['subspecs'] = [{
  :name => 'standard',
  :source_files => 'Sources/{SQLite,SQLiteObjc}/**/*.{swift}',
  :exclude_files => 'Sources/**/Cipher.swift',
  :library => 'sqlite3'
 }]
end

post_install do |installer|
 %w(Pods/Apollo/Sources/ApolloSQLite/*.swift).flat_map { |x| Dir.glob(x) }.each do |file|
 code = File.read(file)
 unless code.include? "import UIKit"
  FileUtils.chmod("+w", file)
  File.write(file, "import UIKit\n" + code)
 end
 end
end

给这些没添加 import UIKit 代码补上就行了。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • 如何快速用上Swift静态库详解

    前言 Swift 支持静态库打包已经有一段时间了,CocoaPods 也提供了 static_framework 参数.然而大部分的第三方依赖都没有及时更新. 本文给出个相对方便一些的方案用上静态库,个人感觉在依赖不那么复杂的时候成本还是比较低的. 效果如下: 示例地址:UseStaticFramework 想办法为每一个 pod 添加 static_framework 是关键. 直接修改 podspec 不太现实,因为 CocoaPods 并没有提供相关的接口.但是当一个 pod 指定 pod

  • C语言中调用Swift函数实例详解

    C语言中调用Swift函数实例详解 在Apple官方的<Using Swift with Cocoa and Objectgive-C>一书中详细地介绍了如何在Objective-C中使用Swift的类以及如何在Swift中使用Objective-C中的类.在后半部分也介绍了如何在Swift中使用C函数,不过对于如何在C语言中使用Swift函数却只字未提.这里我就为大家分享一下如何在C语言中调用Swift函数. 我们首先要知道的是,所有Swift函数都属于闭包.其次,Swift函数的调用约定与

  • JVM 方法调用之静态分派(详解)

    分派(Dispatch)可能是静态也可能是动态的,根据分派依据的宗量数可分为单分派和多分派.这两种分派方式的两两组合就构成了静态单分派,静态多分派,动态单分派,动态多分派这4种组合.本章讲静态分派. 1.静态分派 所有依赖静态类型来定位方法执行版本的分派动作称为静态分派.静态分派的典型应用是方法重载.静态分派发生在编译阶段,因此确定静态分派的动作实际上不是由虚拟机来执行的. 那么什么是静态类型(static type)呢? Super object = new Sub(); 像上面的语句,Sup

  • 虚拟机Linux桥接模式下设置静态IP详解

    本文研究的主要是虚拟机Linux桥接模式下设置静态IP的相关内容,具体介绍如下. 之前一直使用NAT模式,测试时android端远程访问虚拟机的mysql时发现无法连接,但是访问同学拷过来的虚拟机Linux的mysql却成功了,想了下原因是他设置的桥接模式.关于两种模式的区别,网上可以搜到一大堆文章,通俗点讲,NAT模式下,虚拟机从属于主机,也就是访问外部网络必须通过主机来访问,因此虚拟机的IP只有主机才能识别.而桥接模式下,虚拟机和主机是平行关系,共享一张网卡(使用网卡的多个接口),可以直接访

  • PyTorch快速搭建神经网络及其保存提取方法详解

    有时候我们训练了一个模型, 希望保存它下次直接使用,不需要下次再花时间去训练 ,本节我们来讲解一下PyTorch快速搭建神经网络及其保存提取方法详解 一.PyTorch快速搭建神经网络方法 先看实验代码: import torch import torch.nn.functional as F # 方法1,通过定义一个Net类来建立神经网络 class Net(torch.nn.Module): def __init__(self, n_feature, n_hidden, n_output):

  • Vue+Koa2 打包后进行线上部署的教程详解

    最近使用Vue和Koa2重构了自己的博客,过程中踩了不少坑,查了很多资料,最后总算成功上线.之后我计划围绕这个过程写一系列文章,讲讲如何用Vue+Koa2写一套网站. 而现在,先来讲讲最后一步,在写完Vue和Koa2后,如何将它们部署到线上. 1.将Vue和Koa2结合 很多人在打完包后就不知道怎么做了,毕竟后面都是后端的事情.如果你用的是Vue-cli3.0,那么打包这一步会非常简单,只需要执行一条命令即可,其它的不用关心: npm run build 之后会生成一个dist的文件夹,将它放到

  • 大规格文件的上传优化思路详解

    在开发过程中,收到这样一个问题反馈,在网站上传 100 MB 以上的文件经常失败,重试也要等老半天,这就难为需要上传大规格文件的用户了.那么应该怎么做才能快速上传,就算失败了再次发送也能从上次中断的地方继续上传呢?下文为你揭晓答案~ 温馨提示:配合 Demo 源码一起阅读效果更佳 整体思路 第一步是结合项目背景,调研比较优化的解决方案. 文件上传失败是老生常谈的问题,常用方案是将一个大文件切片成多个小文件,并行请求接口进行上传,所有请求得到响应后,在服务器端合并所有的分片文件.当分片上传失败,可

  • SpringBoot web静态资源配置详解

    引言: SpringBoot web项目开发中往往会涉及到一些静态资源的使用,比如说图片,css样式,js等等,今天我们来讲讲这些常见的静态资源应该放在哪个位置,怎么放在自己想放的位置. 1. 项目创建 我们先创建一个空的项目,项目的依赖配置为starter-web依赖,创建好的项目下面有一个resources文件夹,里面有一些空的默认的文件夹,然后有一个配置文件. templates文件下面一般是放置模板页面的,比如html,jsp之类的,static文件一般是是放置静态资源,比如说,图片,文

  • Python基础之time库详解

    一.前言 time库运行访问多种类型的时钟,这些时钟用于不同的场景.本篇,将详细讲解time库的应用知识. 二.获取各种时钟 既然time库提供了多种类型的时钟.下面我们直接来获取这些时钟,对比其具体的用途.具体代码如下: import time print(time.monotonic()) print(time.monotonic_ns()) print(time.perf_counter()) print(time.perf_counter_ns()) print(time.process

  • Python爬虫进阶之Beautiful Soup库详解

    一.Beautiful Soup库简介 BeautifulSoup4 是一个 HTML/XML 的解析器,主要的功能是解析和提取 HTML/XML 的数据.和 lxml 库一样. lxml 只会局部遍历,而 BeautifulSoup4 是基于 HTML DOM 的,会加载整个文档,解析 整个 DOM 树,因此内存开销比较大,性能比较低. BeautifulSoup4 用来解析 HTML 比较简单,API 使用非常人性化,支持 CSS 选择器,是 Python 标准库中的 HTML 解析器,也支

随机推荐