溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

詳解Swift中對(duì)C語言接口緩存的使用以及數(shù)組與字符串轉(zhuǎn)為指針類型的方法

發(fā)布時(shí)間:2020-09-05 10:29:37 來源:腳本之家 閱讀:151 作者:zenny_chen 欄目:移動(dòng)開發(fā)

詳解Swift中對(duì)C語言接口緩存的使用以及數(shù)組與字符串轉(zhuǎn)為指針類型的方法

由于Swift編程語言屬于上層編程語言,而Swift中由于為了低層的高性能計(jì)算接口,所以往往需要C語言中的指針類型,由此,在Swift編程語言剛誕生的時(shí)候就有了UnsafePointer與UnsafeMutablePointer類型,分別對(duì)應(yīng)為const Type*類型與Type *類型。

而在Swift編程語言中,由于一般數(shù)組(Array)對(duì)象都無法直接用于C語言中含有指針類型的函數(shù)參數(shù)(比如:void*),所以往往需要將數(shù)組轉(zhuǎn)為指針類型,此外也需要將數(shù)組中元素內(nèi)容存放到連續(xù)的存儲(chǔ)空間。此外,Swift中的字符串對(duì)象都是String結(jié)構(gòu)體對(duì)象,因此也需要將它們轉(zhuǎn)換為C語言中const char *類型相兼容的類型,因此這里將給大家介紹一些比較簡便、且純Swift接口的使用方法,而不是借助于Objective-C的Foundation庫。

/**
 此函數(shù)用于將一個(gè)數(shù)組(Array)的首個(gè)元素的值做加1操作
 - parameters:
  - p: inout [Int]類型,傳入的數(shù)組對(duì)象首地址
 - returns: Void
*/
func test(inout p: [Int]) {
  p[0] += 1
}

class ViewController: NSViewController {
  
  override func viewDidLoad() {
    super.viewDidLoad()
    
    var a = 0
    
    // 下面使用UnsafeMutablePointer自帶的alloc類方法分配10個(gè)Int元素的存儲(chǔ)空間
    // 這個(gè)方法其實(shí)應(yīng)該就是對(duì)malloc函數(shù)的封裝了,用于外部的C接口十分有用
    let buf = UnsafeMutablePointer<Int>.alloc(10)
    let dst = UnsafeMutablePointer<Int>.alloc(10)
    
    // 分別對(duì)buf與dst存儲(chǔ)空間進(jìn)行初始化賦值
    for i in 0 ..< 10 {
      buf[i] = i
      dst[i] = 0
    }
    
    // 使用assignBackwardFrom將buf存儲(chǔ)空間中后5個(gè)元素拷貝到dst存儲(chǔ)空間的前5個(gè)元素中
    dst.assignBackwardFrom(buf.advancedBy(5), count: 5)
    
    // 用Array分配一個(gè)10個(gè)Int元素的數(shù)組對(duì)象arr
    var arr = [Int](count: 10, repeatedValue: 0)
    
    // 將dst中的所有元素拷貝到arr中
    for i in 0 ..< 10 {
      arr[i] = dst[i]
    }
    print(arr)
    
    // 釋放buf與dst。注意,這里必須用dealloc來釋放,且里面的參數(shù)也要與alloc的參數(shù)對(duì)應(yīng)!
    // 不釋放會(huì)引發(fā)內(nèi)存泄漏
    buf.dealloc(10)
    dst.dealloc(10)
    
    // arr數(shù)組對(duì)象指向另一個(gè)[1, 2, 3]數(shù)組字面量構(gòu)成的Array對(duì)象
    arr = [1, 2, 3]
    
    // 調(diào)用test函數(shù),使得arr的第一個(gè)元素的值加1
    test(&arr)
    print("arr = \(arr)")
    
    // 這里使用Array的withUnsafeMutableBufferPointer方法將數(shù)組元素內(nèi)容轉(zhuǎn)為
    // 指向一個(gè)連續(xù)存儲(chǔ)空間的首地址。
    // 所以p的類型為:UnsafeMutablePointer<Int>
    let p = arr.withUnsafeMutableBufferPointer() {
      // 這里,形參是一個(gè)含有一個(gè)UnsafeMutableBufferPointer的形參,
      // 返回類型為UnsafeMutablePointer的函數(shù)類型。
      (inout buffer: UnsafeMutableBufferPointer<Int>) -> UnsafeMutablePointer<Int> in
      return buffer.baseAddress
    }
    
    a = 0
    
    // 我們先查看原先數(shù)組對(duì)象中的元素內(nèi)容
    for i in 0 ..< 3 {
      a += p[i]
    }
    print("a = \(a)")
    
    // 我們通過p指針對(duì)象對(duì)數(shù)組arr的內(nèi)容進(jìn)行修改
    p[0] -= 1
    p[1] += 1
    p[2] += 2
    
    // 然后打印出修改后的arr數(shù)組對(duì)象中的元素內(nèi)容
    print("arr is: \(arr)")
    
    // 這里用nulTerminatedUTF8方法先轉(zhuǎn)為ContiguousArray<CodeUnit>對(duì)象類型,
    // 其中,CodeUnit是UInt8類型。
    // 然后用withUnsafeBufferPointer轉(zhuǎn)為UnsafePointer<CChar>類型
    let cstr = "abcd".nulTerminatedUTF8.withUnsafeBufferPointer() {
      return UnsafePointer<CChar>($0.baseAddress)
    }
    
    // 這里再將剛才生成的UnsafePointer<CChar>類型的C格式字符串轉(zhuǎn)回String對(duì)象
    let string = String.fromCString(cstr)!
    print("string is: \(string)")
  }
}

上述代碼中,UnsafeMutablePointer的alloc方法也是個(gè)很不錯(cuò)的方法,這個(gè)方法應(yīng)該是直接對(duì)C語言標(biāo)準(zhǔn)庫malloc的封裝,可以使得我們方便地在Swift中分配連續(xù)的存儲(chǔ)空間,比如對(duì)于像圖像處理、矩陣計(jì)算等算法尤為實(shí)用。當(dāng)然,如果我們的邏輯主要用Array來處理,然后再交給底層的C語言接口做高性能計(jì)算,那么也可以使用數(shù)組對(duì)象轉(zhuǎn)連續(xù)存儲(chǔ)空間withUnsafeBufferPointer方法,這種處理方式在網(wǎng)絡(luò)數(shù)據(jù)通信中也比較實(shí)用。

如有疑問請(qǐng)留言或者到本站社區(qū)交流討論,本站關(guān)于IOS 開發(fā)的文章還有很多,希望大家能搜索查閱,感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI