如何制作数组的精确副本?

时间:2015-01-07 04:31:40

标签: arrays swift reference copy

如何精确复制数组?

我很难找到有关在Swift中复制数组的信息。

我尝试使用.copy()

var originalArray = [1, 2, 3, 4]
var duplicateArray = originalArray.copy()

5 个答案:

答案 0 :(得分:149)

数组在Swift中具有完整的价值语义,因此不需要任何花哨的东西。

var duplicateArray = originalArray就是你所需要的。


如果数组的内容是引用类型,那么是的,这只会复制指向对象的指针。要执行内容的深层复制,您可以使用map并执行每个实例的副本。对于符合NSCopying协议的Foundation类,您可以使用copy()方法:

let x = [NSMutableArray(), NSMutableArray(), NSMutableArray()]
let y = x
let z = x.map { $0.copy() }

x[0] === y[0]   // true
x[0] === z[0]   // false

请注意,这里存在一些陷阱,Swift的值语义正在努力保护您 - 例如,由于NSArray表示不可变数组,因此copy方法只返回对自身的引用,因此上面的测试会产生意想不到的结果。

答案 1 :(得分:26)

Nate是对的。如果您正在使用原始数组,那么您需要做的就是将duplicateArray分配给originalArray。

为了完整起见,如果您正在使用NSArray对象,则可以执行以下操作来执行NSArray的完整副本:

var originalArray = [1, 2, 3, 4] as NSArray

var duplicateArray = NSArray(array:originalArray, copyItems: true)

答案 2 :(得分:15)

Nate的答案还有第三种选择:

let z = x.map { $0 }  // different array with same objects

*已编辑* 编辑从此处开始

上面与下面基本相同,实际上使用下面的相等运算符会表现得更好,因为除非更改数组(这是设计),否则不会复制数组。

let z = x

在此处阅读更多内容:https://developer.apple.com/swift/blog/?id=10

*已编辑* 编辑在此处结束

添加或删除此数组不会影响原始数组。但是,更改任何对象'数组保存的任何属性都将在原始数组中看到。因为数组中的对象不是副本(假设数组保存对象,而不是原始数字)。

答案 3 :(得分:13)

对于普通对象,可以做的是实现支持复制的协议,并使对象类实现这样的协议:

protocol Copying {
    init(original: Self)
}

extension Copying {
    func copy() -> Self {
        return Self.init(original: self)
    }
}

然后是用于克隆的Array扩展:

extension Array where Element: Copying {
    func clone() -> Array {
        var copiedArray = Array<Element>()
        for element in self {
            copiedArray.append(element.copy())
        }
        return copiedArray
    }
}

就是这样,查看代码和示例检查这个gist

答案 4 :(得分:0)

如果要复制某个类对象数组的项目。 然后,您可以在不使用NSCopying协议的情况下遵循以下代码,但是您需要具有一个初始化方法,该方法应采用对象所需的所有参数。 这是在操场上测试的示例代码。

class ABC {
    
    var a = 0
    func myCopy() -> ABC {
        
        return ABC(value: self.a)
    }
    
    init(value: Int) {
        
        self.a = value
    }
}

var arrayA: [ABC] = [ABC(value: 1)]
var arrayB: [ABC] = arrayA.map { $0.myCopy() }

arrayB.first?.a = 2
print(arrayA.first?.a)//Prints 1
print(arrayB.first?.a)//Prints 2