本文为 Perfect 官方投稿,作者 Rocky,版权归原作者所有,未经作者同意,请勿转载。
原文地址:http://mp.weixin.qq.com/s/zdC6TZFLaOYkRlnl1QQ_pg
技术之路,共同进步,欢迎投稿、约稿,给文章纠错,请发送邮件至mobilehub@csdn.net。
本程序示范了如何用 Swift 调用自定义 C 语言模块的方法。您可以直接点击这里下载本程序,或者按照以下教程逐步完成。
简介
示范程序中有一个 C 语言的源程序 CSwift.C 和一个头文件 CSwift.h,我们的目标是构造一个 CSwift 的函数库,能够让 Swift 源程序执行 CSwift 程序中的函数。
快速上手
本程序需要 Swift 3.0 以上版本。
下载、编译和测试
$ git clone https://github.com/RockfordWei/CSwift.git
$ cd CSwift
$ swift build
$ ./build.lib.sh
$ swift test
源程序采用 C 语言写成,测试程序则是 Swift 语言编写。因此如果通过测试,则恭喜您,已经成功实现了 Swift 语言调用 C 语言的整个过程。
详细步骤
您可以完全不依赖所有上述内容,而一步一步从零开始制作 C 函数库和调用 C 库的 Swift 代码:
构造空白的函数库
仍然假定函数库名称为 CSwift。首先找一个空白目录,然后执行:
$ mkdir CSwift
$ cd CSwift
$ swift package init --type=system-module
$ mkdir CSwift
$ cd CSwift
$ swift package init
$ mv Tests ..
$ cd ..
$ rm -rf CSwift
$ mkdir Sources
细心的读者会发现,上面的 bash 命令行在 CSwift 文件夹下面建立了第二个 CSwift 文件夹,但是使用了不同的 Swift Package 了命令。第一个命令是“创建 swift 空白项目,而且项目类型是系统模块”;而第二个命令是“创建 swift 空白项目,项目类型是函数库”。这种做法主要是为了能够在同一个项目中用 Swift 去测试 C 语言的模块。
Module Map
下一步是修理一下目标的模块映射表。请把 module.modulemap
修改为如下程序:
module CSwift [system] {
header "./Sources/CSwift.h"
link "CSwift"
export *
}
C 模块编程
好了,现在请在 Source 文件夹下面建立两个 C 语言文件:CSwift.c 和 CSwift.h,内容如下:
Sources/CSwift.h
extern int c_add(int, int);
#define C_TEN 10
Sources/CSwift.c
int c_add(int a, int b) { return a + b ; }
C 模块的编译和库函数链接
上述步骤完成后,可以开始进行 C 函数库的编译了。首先准备一个编译脚本 build.lib.sh,内容如下:
build.lib.sh
export LIB_BUILD=.build/debug
cd Sources
clang -c CSwift.c
cd ..
mv Sources/CSwift.o $LIB_BUILD
ar -r $LIB_BUILD/libCSwift.a $LIB_BUILD/CSwift.o
export LIB_BUILD=
上面的程序首先把 CSwift.c 用 clang 编译器(也就是 swift 的母编译器)链接为一个.o 目标文件,然后通过 ar -r
命令链接为静态函数库,关键是函数库的位置一定要与 Swift 的目标库目录路径一致。
如果您准备发行,则用.build/release 代替上面程序中的 $LIB_BUILD
变量。
保存后执行下列命令:
$ cd CSwift
$ chmod +x build.lib.sh
$ swift build
$ ./build.lib.sh
到此为止,C 语言函数库就应该准备好了。
Swift 程序调用
请修改 Tests/CSwiftTests/CSwiftTests.swift
文件,内容如下:
import XCTest@testable
import CSwift
class CSwiftTests: XCTestCase {
func testExample() {
// 测试调用 C 函数
let three = c_add(1, 2)
XCTAssertEqual(three, 3)
// 测试调用 C 语言的符号
XCTAssertEqual(C_TEN, 10)
}
static var allTests : [(String, (CSwiftTests) -> () throws -> Void)] {
return [
("testExample", testExample),
]
}
}
测试
最后一步最简单,直接执行:
$ swift test
如果没有问题,那就一切 OK 了!
其他
如果您在使用 Xcode,则需要使用 swift package generate-xcodeproj
,但是需要调整上述 build.lib.sh 内容的编译目标目录,并配合 Xcode 偏好设置选择匹配的目录,否则无法测试。
了解最新移动开发、VR/AR 干货技术分享,请关注 mobilehub 微信公众号(ID: mobilehub)。