引用类型,Reference Type,多个实例可以共享统一份数据。
数值类型,Value Type,一个实例独享一份数据。这里的数据指的是内存中位置,并不是指具体的数值。
如果要将一个引用类型的的实例,赋值给一个变量,那么变量仅仅是共享内存中的这份数据,变量获得的是对这份数据的引用;如果是将一个数值类型的实例赋值给一个变量,那么将会在内存中创建一份相同的数据,赋值给变量。
两者的核心在于:
- 引用类型实例赋值给其他变量后,对变量的修改,原引用实例会一起被修改;
- 数值类型实例赋值给其他变量后,对变量的修改,不会影响原来的数值实例。
平时在使用和选择时,务必要记得这一点,否则可能会像我一样用踩坑来获取知识。(PS,我也不好说踩坑一定不好,我踩了一天的坑,对这个问题才会印象深刻。)
引用类型
一般的类,大多都是引用类型,自定义的类更是如此。
下面用代码说明下引用类型赋值后共享数据的情况:
class Person {
var name: String
init(_ name: String) {
self.name = name
}
}
let firstPerson = Person("Shen")
// 将引用类型的实例赋值给变量
var secondPerson = firstPerson
secondPerson.name = "Mike"
// 结果是Mike,因为引用类型共享同一份数据
print(firstPerson.name)
数值类型
在Swift里,基本数据类型,都是数值类型,比如:Int
、Bool
、Double
等。
另外,String
、Array
、Dictionary
这些基础的类也是数值类型。
最后,struct
、enum
、tuple
这些也都是数值类型。
下面用代码说明下,数据类型赋值后,独享数据互不干涉的情况:
struct SuperPerson {
var name: String
}
let firstSuper = SuperPerson(name: "Clark")
var secondSuper = firstSuper
secondSuper.name = "Bruce"
// 结果是Clark,因为Struct是数值类型,赋值给其他变量时,是直接复制给其他变量的。
print(firstSuper.name)
特殊情形:Array里的引用类型
Array是数值类型,就是说,将一个Array赋值给另一个变量时,变量持有的是不同的Array,在内存中的数据是不同的。比如下面的代码:
let firstPerson = Person("Huo")
let secondPerson = Person("Shi")
let persons = [firstPerson, secondPerson]
// 将数值类型的Array赋值给一个变量
var newPersons = persons
// 修改变量
newPersons[0] = Person("Mike")
// 结果是Huo,即newPersons和persons其实是两个不同的对象。
print("\(persons[0].name)")
对于一般的赋值,比如上面的var newPersons = persons
,因为persons
里面的元素本身是引用类型,而且这种默认执行的是浅复制,所以在新的newPersons
中包含的并不是拥有全新数据的元素,而仅仅是分享persons
中所包含的元素的内存数据而已,就像上面讨论的引用类型一样。请参考下面的代码:
// 获取newPersons中的第2个元素,并修改元素的name属性
let second = newPersons[1]
second.name = "Jim"
// 结果也是Jim,即虽然newPersons和persons是不同的数组。
// 但由于其生成方式是直接赋值,两个数组里其实是相同的元素。
print("\(persons[1].name)")
如果你在赋值时,关心里面的元素且需要对元素进行独立修改,那么就需要使用深复制,或者将元素替换为数值类型。
以下是对上面Person
类的深复制代码实例:
// 自定义一个深复制方法,其实就是重新创建一个实例,并且对其属性赋予相同的数值。
extension Person {
func copy() -> Person {
let person = Person(name)
return person
}
}
// 对每个元素进行深复制然后加入新的Array
var newerPersons = persons.map { $0.copy() }
let secondP = newerPersons[1]
secondP.name = "Jim"
// 结果不是Jim
print("\(persons[1].name)")
使用数据类型时的注意事项
务必记得,要修改数据变量,必须直接对其进行修改,而不能像对普通类的对象一样,先进行一次临时变量赋值,再对变量进行修改。
最后:
- 引用类型实例赋值给其他变量后,对变量的修改,原引用实例会一起被修改;
- 数值类型实例赋值给其他变量后,对变量的修改,不会影响原来的数值实例。