如何在Swift中調用C庫(完結篇)
上篇如何在 Swift中調用C庫(進階篇) 中,我們已經解決了大部分的問題,本篇我們來講講如何完善這個示例。
之前我們的示例中包含了以下幾個項目:
- hiredis-bridge 用于橋接兼容hiredis庫中的一些方法。
- CHiredis 用于映射hiredis和hiredis-bridge中的方法到Swift中去。
- SwiftHiredis 通過CHiredis來調用hiredis和hiredis-bridge中的方法。
這些項目都是運行在Linux平臺下,我們來看看如何將這個示例兼容Mac平臺。
在Mac上安裝Reids和Hiredis
使用Homebrew來安裝:
brew install redis hiredis
然后使用下面命令運行redis:
redis-server /usr/local/etc/redis.conf
修改hiredis-bridge
主要是要修改hiredis-bridge的Makefile
TARGET = hiredis_bridge LIB_NAME = hiredis_bridge PREFIX ?= /usr/local UNAME := $(shell uname) all: $(TARGET) $(TARGET): *.c ifeq ($(UNAME), Linux) # build for linux clang -c *.c ar -rcs lib$(LIB_NAME).a *.o endif ifeq ($(UNAME), Darwin) # build for darwin clang -c *.c -I$(PREFIX)/include clang -dynamiclib *.o -I$(PREFIX)/include $(PREFIX)/lib/libhiredis.dylib -o lib$(LIB_NAME).dylib endif rm *.o install: mkdir -p $(TARGET)/usr/local/lib mkdir -p $(TARGET)/usr/local/include/$(TARGET) cp *.h $(TARGET)/usr/local/include/$(TARGET) ifeq ($(UNAME), Linux) # copy .a cp lib$(LIB_NAME).a $(TARGET)/usr/local/lib/ rm lib$(LIB_NAME).a endif ifeq ($(UNAME), Darwin) # copy .dylib cp lib$(LIB_NAME).dylib $(TARGET)/usr/local/lib/ rm lib$(LIB_NAME).dylib endif mkdir -p $(PREFIX) cp -r $(TARGET)/usr/local/* $(PREFIX)/ rm -r $(TARGET)
Makefile主要作用是編譯,然后把hiredis-bridge的頭文件和庫文件放到對應的位置上去。不過因為Linux和Mac環境的區別,編譯的方法會不一樣。Linux編譯出來是靜態庫,后綴為.a;Mac需要編譯出動態庫,后綴為.dylib。然后都會把頭文件(.h)放到/usr/local/include/目錄下,庫文件(.a或.dylib)放到/usr/local/lib/目錄下。
代碼更新到github上: https://github.com/fengluo/hiredis-bridge/tree/c3a7f568bde781a37ebd29584a6357129284d1f6 tag: 0.2.0
創建CHiredis-OSX
hiredis在Linux和Mac下的安裝位置不一樣。Linux下,hiredis的頭文件位置在/usr/include/,而Mac下則位于/usr/local/include/。SPM(Swift Package Manager)在 文檔 中有提到過這個問題,并聲稱會在將來解決。我們現在的解決方法則是創建一個新的映射庫。
只要把之前的 CHiredis 復制過來,修改module.modulemap中的header路徑即可:
module CHiredis [system] { header "/usr/local/include/hiredis/hiredis.h" header "/usr/local/include/hiredis_bridge/hiredis-bridge.h" link "hiredis" link "hiredis_bridge" export * }
并且需要在Package.swift中定義包的名稱,否則編譯后這個包會以CHiredis-OSX為名出現。
import PackageDescription let package = Package( name: "CHiredis" )
代碼上傳到github上: https://github.com/fengluo/CHiredis-OSX tag: 0.2.0
修改SwiftHiredis
在SwiftHiredis,我們需要基于現在的系統平臺來選擇依賴的映射庫:CHiredis or CHiredis-OSX。只要簡單的改動即可:
import PackageDescription #if os(OSX) let CHiredisURL = "https://github.com/fengluo/CHiredis-OSX.git" #else let CHiredisURL = "https://github.com/fengluo/CHiredis.git" #endif let package = Package( dependencies: [ .Package(url: CHiredisURL, majorVersion: 0, minor: 2)] )
這樣對于該示例的跨平臺兼容就解決了。在SwiftHiredis編譯運行測試一下:
swift build .build/debug/SwiftHiredis
就會得到上篇文章一樣的結果 1234 。
代碼更新到github上: https://github.com/fengluo/SwiftHiredis/tree/b1927cc9680103e51c9d114a18427d49c26d461f tag: 0.3.0
總結
至此,『如何在Swift中調用C庫』這個系列我就寫完了,基本上都是經驗之談。以后會繼續講解Swift在服務端的開發。
Enjoy it!