var dataFile: NSData = NSMutableData.init(data: wav.subdataWithRange(NSRange.init(location: currentByte, length: wavDataSize)))
如何将此代码转换为使用 Data
和 Swift 3?或者如何将NSRange
解析为Range
最佳答案
分离 Data
变小Data
实例
假设
这个答案是问题中代码的 Swift 3 & 4 等价物。它将产生相同的结果,dataFile
,给定相同的输入值:wav
, currentByte
和 wavDataSize
假设周围的代码没有任何变化。
我没有假设变量是什么:wav
, dataFile
, currentByte
或 wavDataSize
意思是。为避免变量名称暗示问题中未说明的内容,我将使用以下名称代替:sourceData
, subdata
, rangeStartByte
和 subdataLength
.我假设围绕它的代码(问题中未显示)将确保 rangeStartByte
和 subdataLength
在有效范围内以避免错误。
将 NSRange 转换为 Range
问题中的 Swift 2 实现使用 NSRange
由起点和长度定义,如下所示:
NSRange.init(location: rangeStartByte, length: subdataLength)
我建议的 Swift 3 & 4 实现创建了一个等效的 Range<Int>
由这样的起点和终点定义:
rangeStartByte ..< (rangeStartByte + subdataLength)
我将一个应用程序从 Swift 2.2 转换为 3,它使用类似的代码以较小的 block 上传照片。在转换过程中,我们忽略了这种细微差别,并使用 Swift 2 实现的长度代替 Swift 3 和 4 实现的端点。这导致了一个很难解决的缺陷。第一次迭代成功,但后续迭代失败。
另一个答案实现了我刚才描述为解决方案的问题。它使用 subdataLength
从 Swift 2 范围的长度作为 Swift 3 和 4 范围的终点。这将在 currentByte 为 0 且 subdataLength
的特殊情况下产生相同的结果。是 <= NSData 实例的长度(这就是我描述的问题中第一次迭代成功的原因)。该假设未在问题中明确说明,并且为其他人提供了不太灵活的解决方案。
Swift 3 & 4 等效
var subdata = sourceData.subdata(in: rangeStartByte ..< (rangeStartByte + subdataLength))
swift 2.2
(来自带有更新变量名的问题的代码)
var subdata: NSData = NSMutableData.init(data: sourceData.subdataWithRange(NSRange.init(location: rangeStartByte, length: subdataLength)))
可运行的示例代码
我提供了示例代码,您可以在 playground 中运行,演示如何使用这行代码来分隔 Data
实例变小 Data
实例。出处Data
实例是从字符串 "ABCDEFGHIJKL"
创建的.此实例被分成较小的 Data
长度为 5 的实例。
带上下文的 Swift 3 和 4
import UIKit
var sourceString = "ABCDEFGHIJKL"
let sourceData = sourceString.data(using: String.Encoding.utf8)! // sourceData is equivalent to "wav" from question
var rangeStartByte = 0 // rangeStartByte is equivalent to "currentByte" from question
let maxSubdataLength = 5
let dataLength = sourceString.lengthOfBytes(using: String.Encoding.utf8)
precondition(maxSubdataLength <= dataLength, "maxSubdataLength must be <= to dataLength")
while rangeStartByte < dataLength {
// subdataLength is equivalent to "wavDataSize" from question
let subdataLength = min(maxSubdataLength, dataLength - rangeStartByte)
// subdata is equivalent to "dataFile" from question
let subdata = Data(sourceData.subdata(in: rangeStartByte ..< (rangeStartByte + subdataLength)))
let subdataString = String(data: subdata, encoding: String.Encoding.utf8) ?? ""
print("'\(subdataString)'")
rangeStartByte += subdataLength
}
结果是:
'ABCDE'
'FGHIJ'
'KL'
带上下文的 Swift 2.2
import UIKit
var sourceString = "ABCDEFGHIJKL"
let sourceData = sourceString.dataUsingEncoding(NSUTF8StringEncoding)! // sourceData is equivalent to "wav" from question
var rangeStartByte = 0 // rangeStartByte is equivalent to "currentByte" from question
let maxSubdataLength = 5
let dataLength = sourceString.lengthOfBytesUsingEncoding(NSUTF8StringEncoding)
precondition(maxSubdataLength <= dataLength, "maxSubdataLength must be <= to dataLength")
while rangeStartByte < dataLength {
// subdataLength is equivalent to "wavDataSize" from question
let subdataLength = min(maxSubdataLength, dataLength - rangeStartByte)
// subdata is equivalent to "dataFile" from question
let subdata: NSData = NSMutableData.init(data: sourceData.subdataWithRange(NSRange.init(location: rangeStartByte, length: subdataLength)))
let subdataString = String(data: subdata, encoding: NSUTF8StringEncoding) ?? ""
print("'\(subdataString)'")
rangeStartByte += subdataLength
}
结果是:
'ABCDE'
'FGHIJ'
'KL'
使用 NSRange 的 Swift 3 和 4
pedrouan 的回答是这样使用 NSRange 的:
var subdata: Data = Data(sourceData.subdata(with: NSRange(location: rangeStartByte, length: subdataLength)))
我最初无法编译它,所以我忽略了它。现在我意识到如果 sourceData
它是有效的在 NSData
声明或转换而不是 Data
如果您想在上面的“Swift 3 & 4 with context”示例代码中运行此方法,请将该示例中的相应代码替换为:
// subdata is equivalent to "dataFile" from question
let sourceNSData = sourceData as NSData
let subdata = sourceNSData.subdata(with: NSRange(location: rangeStartByte, length: subdataLength))
我尽量不使用像 NSRange 这样的“NS”类,所以我更喜欢使用 Swift Range 的解决方案。
关于ios - NSData 到数据 swift 3,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39655472/