如何在Mac上使用Swift調用C接口開發條形碼應用

雖然Objective-C還活的很好,可是蘋果已經把重心轉移到Swift上。將來Mac和iOS的開發必然是以Swift爲主。由於Swift還比較新,不少SDK尚未提供Swift版本。這裏分享下如何使用Swift來調用C。html

參考原文:How to Bridge C Code to Create Swift Barcode Reader on Macios

做者:Xiao Linggit

翻譯:yushulxgithub

軟件下載

混合使用Swift和C

蘋果在iBooks裏提供了電子書Using Swift with Cocoa and Objective-C。在網頁上能夠閱讀Interacting with C APIsapp

Swift和C類型映射關係參考:
async

使用Swift和C在Mac上實現1D/2D條形碼應用

在Xcode中使用快捷鍵Command+Shift+N建立新工程。ide

頭文件和依賴的庫直接拖入工程就好了。Xcode會自動關聯。
函數

Command+N建立一個C文件,用於調用底層的C動態連接庫(Dynamsoft Barcode dylib)。

完成以後,Xcode會彈出提示:

確認以後,Xcode會自動生成一個橋接頭文件。把C用到的頭文件添加進去:

#import "native_lib.h"

參考Dynamsoft Barcode SDK的在線代碼示例作一些修改:

#include "native_lib.h"
 
int dbr_release_memory(pBarcodeResultArray paryResult)
{
    DBR_FreeBarcodeResults(&paryResult);
    printf("Game Over\n");
    return 0;
}
 
pBarcodeResultArray dbr_decodeBarcodeFile(char* pszImageFile)
{
    // Parse command
    __int64 llFormat = (OneD |QR_CODE);
 
    int iMaxCount = 0x7FFFFFFF;
    int iIndex = 0;
    ReaderOptions ro = {0};
    pBarcodeResultArray paryResult = NULL;
    int iRet = -1;
    char * pszTemp = NULL;
    char * pszTemp1 = NULL;
    struct timeval begin, end;
 
    if (NULL == pszImageFile)
    {
        printf("The syntax of the command is incorrect.\n");
        return NULL;
    }
 
    // Set license
    DBR_InitLicense("A825E753D10C6CAC7C661140EC5ABEC3");
 
    // Read barcode
    gettimeofday(&begin, NULL);
    ro.llBarcodeFormat = llFormat;
    ro.iMaxBarcodesNumPerPage = iMaxCount;
    iRet = DBR_DecodeFile(pszImageFile, &ro, &paryResult);
    gettimeofday(&end, NULL);
 
    // Output barcode result
    pszTemp = (char*)malloc(4096);
    if (iRet != DBR_OK)
    {
        sprintf(pszTemp, "Failed to read barcode: %s\r\n", DBR_GetErrorString(iRet));
        printf("%s", pszTemp);
        free(pszTemp);
        return NULL;
    }
 
    if (paryResult->iBarcodeCount == 0)
    {
        sprintf(pszTemp, "No barcode found. Total time spent: %.3f seconds.\r\n",
                ((float)((end.tv_sec * 1000 * 1000 +  end.tv_usec) - (begin.tv_sec * 1000 * 1000 + begin.tv_usec))/(1000 * 1000)));
        printf("%s", pszTemp);
        DBR_FreeBarcodeResults(&paryResult);
        return 0;
    }
 
    sprintf(pszTemp, "Total barcode(s) found: %d. Total time spent: %.3f seconds\r\n\r\n", paryResult->iBarcodeCount,
            ((float)((end.tv_sec * 1000 * 1000 +  end.tv_usec) - (begin.tv_sec * 1000 * 1000 + begin.tv_usec))/(1000 * 1000)));
    printf("%s", pszTemp);
 
    return paryResult;
}

修改下生成的頭文件:

#ifndef __DBRConsole__native_lib__
#define __DBRConsole__native_lib__
 
#include <stdio.h>
 
#include <stdlib.h>
#include <string.h>
#include <sys/time.h>
 
#include "If_DBR.h"
 
pBarcodeResultArray dbr_decodeBarcodeFile(char* fileName);
int dbr_release_memory(pBarcodeResultArray paryResult);
const char * GetFormatStr(__int64 format);
 
#endif /* defined(__DBRConsole__native_lib__) */

用Swift編寫命令行工具

把Swift中的String轉換成char *:

var file: String = "/Applications/Dynamsoft/Barcode Reader 3.0 Trial/Images/AllSupportedBarcodeTypes.tif" // barcode file
//let namePtr = strdup(filePath.bridgeToObjectiveC().UTF8String)
 
var filePtr = strdup(file.cStringUsingEncoding(NSUTF8StringEncoding)!)
var fileName: UnsafeMutablePointer<CChar> = UnsafeMutablePointer(filePtr)

註釋掉的接口bridgeToObjectiveC 在早期的Swift版本中是能夠用的。Xcode 6.4中已經去掉了。

命令行獲取Barcode結果:

var result : pBarcodeResultArray = dbr_decodeBarcodeFile(fileName)
 
free(filePtr)
 
println("Total barcode: \(String(result.move().iBarcodeCount))\n.......")
 
var count = result.move().iBarcodeCount
var pBarcodeResult: pBarcodeResult = nil
var barcodeIndex = 1
 
// print barcode recognition results
for i in 0..<Int(count) {
    pBarcodeResult = result.move().ppBarcodes.advancedBy(i).move()
    println("Barcode: \(barcodeIndex++)")
    println("Page: \(String(pBarcodeResult.move().iPageNum))")
    var lFormat: __int64 = pBarcodeResult.move().llFormat
    var format = String.fromCString(GetFormatStr(lFormat))
    println("Type: \(format!)")
    println("Value: \(String.fromCString(pBarcodeResult.move().pBarcodeData)!)")
    println(".......")
 
}
 
// free C memory
dbr_release_memory(result)

用Swift建立Cocoa應用

AppDelegate.swift中建立按鈕和文本控件:

@IBOutlet weak var window: NSWindow!
@IBOutlet weak var btLoad: NSButton!
@IBOutlet weak var btRead: NSButton!
@IBOutlet weak var text: NSTextField!
@IBOutlet weak var filePath: NSTextField!

建立按鈕響應函數:

@IBAction func onClick(sender: NSButton) {
        var title = sender.title
        switch(title) {
        case "Load Barcode File":
            dispatch_async(dispatch_get_main_queue()) {
                self.openPanel()
            }
            break
        case "Read Barcode":
 
            if self.filePath.stringValue == "" {
                text.stringValue = "Please add image path!"
                return
            }
 
            println("default:" + self.filePath.stringValue)
            var dbr = DBR()
            text.stringValue = dbr.decodeFile(self.filePath.stringValue)!
            break
        default:
            break
        }
 
    }

在Interface Builder中把代碼和UI元素關聯起來:

使用NSOpenPanel 加載文件:

func openPanel() {
        var openPanel = NSOpenPanel()
        openPanel.allowsMultipleSelection = false
        openPanel.canChooseDirectories = false
        openPanel.canCreateDirectories = false
        openPanel.canChooseFiles = true
        openPanel.beginWithCompletionHandler { (result) -> Void in
            if result == NSFileHandlingPanelOKButton {
                if let path = openPanel.URL?.path {
                    self.filePath.stringValue = path
                }
            }
        }
 
    }

建立DBR.swift用於讀取條形碼:

import Foundation
 
class DBR {
    func decodeFile(file: String)->String? {
 
        var filePtr = strdup(file.cStringUsingEncoding(NSUTF8StringEncoding)!)
        var fileName: UnsafeMutablePointer<CChar> = UnsafeMutablePointer(filePtr)
        var result : pBarcodeResultArray = dbr_decodeBarcodeFile(fileName)
 
        free(filePtr)
 
        println("Total barcode: \(String(result.move().iBarcodeCount))\n.......")
 
        var count = result.move().iBarcodeCount
        var barcodeIndex = 1
        var results: String = ""
        // print barcode recognition results
        for i in 0..<Int(count) {
            var pBarcodeResult = result.move().ppBarcodes.advancedBy(i).move()
 
            results += "Barcode: \(barcodeIndex++)\n"
            results += "Page: \(String(pBarcodeResult.move().iPageNum))\n"
 
            var lFormat: __int64 = pBarcodeResult.move().llFormat
            var format = String.fromCString(GetFormatStr(lFormat))
 
            results += "Type: \(format!)\n"
            results += "Value: \(String.fromCString(pBarcodeResult.move().pBarcodeData)!)\n"
            results += ".......\n"
 
        }
 
        // free C memory
        dbr_release_memory(result)
 
        return results
    }
}

運行1D/2D條形碼應用:

源碼

https://github.com/yushulx/swift-barcode-reader

相關文章
相關標籤/搜索