本程序示范了如何用Swift调用自定义C语言模块的方法。一起来看看吧,希望对大家学习swift有所帮助。
简介
示范程序中有一个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偏好设置选择匹配的目录,否则无法测试。
来源:SegmentFault