【问题标题】:How to convert UnsafePointer<[Float]> or fit Float Array data to Unsafe[Mutable]Pointer<Float> in Swift 3?如何在 Swift 3 中将 UnsafePointer<[Float]> 或将 Float Array 数据转换为 Unsafe[Mutable]Pointer<Float>?
【发布时间】:2016-12-01 09:09:40
【问题描述】:

我有一个来自 HDF5 文件的浮点值 (floatArray) 数组,我想将此数据用作接受 Unsafe[Mutable]Pointer (inputForFunction) 的函数的输入。我是 UnsafePointer 的新手,所以我需要别人的帮助。

我想到的一种方法是将数组存储到文件中,然后打开它并将文件描述符映射到变量的内存中以供函数输入:

  // read the data from HDF5 file
  var floatArray: [Float] = try originalHDF5DataSet.read()

  // store the data
  let dataToStore = NSData(bytes: &floatArray, length: sizeOfArray * MemoryLayout<Float>.size)
  let tmpPath = URL(fileURLWithPath: NSTemporaryDirectory())
  let filePath = tmpPath.appendingPathComponent("floatArrayData").appendingPathExtension("dat")
  do{
    try data.write(to: filePath, options: .atomic)
    let fileManager : FileManager   = FileManager.default
    if fileManager.fileExists(atPath:filePath.path){
      print("Success")
    }else{
      print("Fail")
    }

    // open the data 
    let dataPath = filePath.path
    let fd = open(dataPath, O_RDONLY, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH)
    assert(fd != -1, "Error: failed to open output file at \""+dataPath+"\"  errno = \(errno)\n")
    // memory map the parameters
    let hdr = mmap(nil, Int(sizeOfArray), PROT_READ, MAP_FILE | MAP_SHARED, fd, 0)
    // cast Void pointers to Float
    let inputForFunction = UnsafePointer<Float>(hdr!.assumingMemoryBound(to: Float.self))
  } catch{
    print("failed")
  }

但是,它可能不是最好的实现,因为存储数据的过程似乎很耗时。

这就是为什么我正在考虑使用 UnsafePointer 将指针从 floatArray 传递到 inputForFunction,但我不确定如何实现它。仅供参考,我想到的一种方法是直接使用withUnsafePointermethod:

var floatArray: [Float] = try originalHDF5DataSet.read()
var inputForFunction = UnsafeMutablePointer<Float>.allocate(capacity: Int(sizeOfArray))
withUnsafePointer(to: &floatArray[0]) { (ptr: UnsafePointer<Float>) in
  inputForFunction = UnsafeMutablePointer(mutating: ptr)
}

或者另一种方法是使用withMemoryRebound(但以下方法不起作用):

var floatArray: [Float] = try originalHDF5DataSet.read()
var inputForFunction = UnsafeMutablePointer<Float>.allocate(capacity: Int(sizeOfArray))
withUnsafePointer(to: &floatArray) { (ptrDataArray: UnsafePointer<[Float]>) in
  inputForFunction = ptrDataArray.withMemoryRebound(to: inputForFunction.self, capacity: Int(sizeOfArray), { (ptr: UnsafePointer<[Float]>) -> UnsafeMutablePointer<Float> in
    return ptr[0]
  })
}

我的问题是

  • 如何将floatArray的数据地址传递给inputForFunction
  • 如果我需要从 UnsafePointer 转换为 Unsafe[Mutable]Pointer,该怎么做?
  • 如果需要,如何使用withMemoryRebound 来实现此目的?我可以分配吗 闭包内inputForFunction 的地址/值?
  • 当我检查floatArrayfloatArray[0] 的地址时,它们是不同的(这似乎与C/C++ 不同)。应该将哪个地址传递给inputForFunction

【问题讨论】:

  • 要向其传递数据的函数声明的究竟如何?
  • 这是针对 iOS 上的 CNN 的。输入用于 kernelWeights 和偏差。 init(设备:MTLDevice,convolutionDescriptorfullyConnectedDescriptor:MPSCNNConvolutionDescriptor,kernelWeights:UnsafePointer,biasTerms:UnsafePointer?,标志:MPSCNNConvolutionFlags)

标签: swift pointers memory-management unsafe-pointers unsafemutablepointer


【解决方案1】:

您可以找到许多解释如何从[Float] 获取Unsafe(Mutable)Pointer&lt;Float&gt; 的文章。他们中的一些人回答了你的一些问题。稍后尝试找到它们。

  • 如何将floatArray的数据地址传递给inputForFunction?

    您通常使用数组的方法withUnsafeBufferPointerwithUnsafeMutableBufferPointer。您可以在闭包参数中获取Unsafe(Mutable)BufferPointer&lt;Element&gt;,其中包含Unsafe(Mutable)Pointer&lt;Element&gt; 作为其baseAddress 属性。

    var floatArray: [Float] = try originalHDF5DataSet.read()
    floatArray.withUnsafeBufferPointer {unsafeBufferPointer in
        let inputForFunction = unsafeBufferPointer.baseAddress
        //`unsafeBufferPointer` (including its `baseAddress`) is valid only in this closure.
        //So, do not pass `inputForFunction` outside of the closure, use it inside.
        //...
    }
    
  • 如果我需要从 UnsafePointer 转换为 Unsafe[Mutable]Pointer,该怎么做?

    当你找到像UnsafePointer&lt;[Float]&gt; 这样的东西时,你就走错了路。在 Swift 中,Array 是一个隐藏结构,可能包含对其元素的(多级)间接引用。你通常不需要使用UnsafePointer&lt;Array&lt;Float&gt;&gt;

  • 如果需要,如何使用 withMemoryRebound 来达到这个目的?我可以将地址/值分配给闭包内的 inputForFunction 吗?

    当您有一个AType 数组并希望将其作为Unsafe(Mutable)Pointer&lt;AnotherType&gt; 传递时,您可以使用withMemoryRebound。但在您的情况下,您只需要使用Float,因此您的目的可能不需要withMemoryRebound

  • 当我检查 floatArray 和 floatArray[0] 的地址时,它们是不同的(这似乎与 C/C++ 不同)。应该将哪个地址传递给 inputForFunction?

    没有。正如我上面写的, address of floatArray 只是指向某个包含隐藏结构的地方,所以它对你的目的完全没用。而floatArray[0]的地址也不保证是整个内容的第一个元素的地址。 Swift 可以创建一个只包含一个元素的时间区域,并传递时间区域的地址。


顺便问一下,您真的需要将您的[Float] 转换为Unsafe(Mutable)Pointer&lt;Float&gt; 吗?

如果你的函数签名是这样的:

func takingUnsafePointer(_ pointer: UnsafePointer<Float>)

你可以这样称呼它:

takingUnsafePointer(floatArray)
//In this case `floatArray` can be a `let` constant.

否则,如果函数签名是:

func takingUnsafeMutablePointer(_ pointer: UnsafeMutablePointer<Float>)

称它为:

takingUnsafeMutablePointer(&floatArray)
//In this case `floatArray` needs to be a `var`, not `let`.

请记住,在这两种情况下,传递的指针仅在函数调用时有效。您不应该将指针导出到其他地方。

【讨论】:

  • 也许我不需要转换。如果我将 floatArray 作为参数传递,这是否意味着将数组的地址作为 let 常量传递?
  • @kazoo,我想我可以说是的。检查Using book 中的this part。甚至可以将 Array 文字传递给 C 函数的 const 指针参数。
  • 那部分我得到了它。我仍然不确定传递 baseAddress 和 floatArray 的区别。我打印了地址的实际值,发现它们是不同的(另外,baseAddress 和 floatArray[0] 的地址相同。)。正如您所提到的, floatArray 指向具有隐藏结构的某个位置。如果是这样,是否可以通过传递floatArray 将整个 floatArray 的数据传递给函数?我应该改用 baseAddress 吗?还是两者完全一样?
  • @kazoo,在 Swift 中,当将数组传递给指针参数时,Swift 可能会将数组的全部内容复制到一个临时区域,并传递该区域的地址。在创建可能包含数组的时间区域作为其baseAddress 的缓冲区指针时,可能会发生同样的情况。这可能并不总是发生,因此在某些情况下地址可能相同,在其他情况下可能不同。可以肯定的一件事是,在执行函数或闭包时会保留该区域。除非您的函数在不同的函数调用中需要相同的地址,否则两者都可以正常工作。
【解决方案2】:

Array 类型有withUnsafeBufferPointer 方法。

此方法接受 UnsafeBufferPointer 的闭包,就像您要求的 UnsafePointer&lt;[Float]&gt;

如果你需要UnsafePointer 到数组的开头,你可以使用UnsafeBufferPointerbaseAddress 属性。

示例代码:

let bufferPointer: UnsafeBufferPointer<Float> = floatArray.withUnsafeBufferPointer { bufferPointer in
    return bufferPointer
}

let baseAddress: UnsafePointer<Float> = floatArray.withUnsafeBufferPointer { bufferPointer in
    return bufferPointer.baseAddress
}

编辑:或者如果它的功能只是传递&amp;floatArray 可能会起作用。

【讨论】:

  • Apple 的 API Reference of withUnsafeBufferPointer 明确指出 指针参数仅在闭包执行期间有效。 你最好不要导出 bufferPointer 或其 @987654336 @ 在闭包之外。不幸的是,您的代码可能在某些有限的条件下工作,但未指定条件。你应该避免这种危险的使用。
  • @OOPer,是Unsafe。是的,我认为在 func 中传递 &amp;floatArray 可能足以解决 Op 的问题。
  • 我认为在 func 中传递&amp;floatArray 可能足以解决 Op 的问题。 我同意。您的 EDIT 可能是 OP 的最佳解决方案。所以,我将后一部分添加到我的答案中。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2017-06-13
  • 1970-01-01
  • 1970-01-01
  • 2014-06-28
  • 2011-06-17
  • 2023-03-31
相关资源
最近更新 更多