输出的结果为:
Element(name: “A”, number: 1)
Element(name: “B”, number: 2)
当引用或者传递 引用类型 变量的时候,新的变量指针指向的仍是原先的内存地址。此时原先的变量值改变的话,也会导致新变量值的变化。
比如:
class Element {
var name : String
var number : Int
required init(name : String, number : Int) {
self.name = name
self.number = number
}
}
extension Element : CustomStringConvertible {
var description : String {
return "Element(name: (name), number: (number))"
}
}
var firstElement = Element(name: "A", number: 1)
var secondElement = firstElement
secondElement.name = "B"
secondElement.number = 2
print(firstElement)
print(secondElement)
此时的输出结果为:
Element(name: B, number: 2)
Element(name: B, number: 2)
我们为什么在此讨论值类型和引用类型呢?
回到之前book和pages的例子。我们用struct代替class:
struct Book {
private var pages = [Page]()
mutating func add(_ page : Page) {
pages.append(page)
}
}
struct Page {
private var book : Book
init(book : Book) {
self.book = book
}
}
var book = Book()
let page = Page(book: book)
book.add(page)
此时,便不会发生循环引用的情况。
如果仍想使用class的话,可以使用weak来避免循环引用:
class Book {
private var pages = [Page]()
func add(_ page : Page) {
pages.append(page)
}
}
class Page {
private weak var book : Book?
required init(book : Book) {
self.book = book
}
}
let book = Book()
let page = Page(book: book)
book.add(page)
Protocols
Protocols在swift中使用的很广泛。class,struct 和 enum 都可以使用Protocol。但是如果使用不当的话,同样会引起循环引用。
比如:
protocol ListViewControllerDelegate {
func configure(with list : [Any])
}
class ListViewController : UIViewController {
var delegate : ListViewControllerDelegate?
override func viewDidLoad() {
super.viewDidLoad()
}
}
ListViewController 中的delegate变量是strong类型的,可以引用任何实现它protocol的变量。假如实现其protocol的变量对该 view controller 同样有强引用的话会怎么样? 声明delegate为weak可能会避免这种情况,但是这样的话会引起编译错误,因为structs和enums不能引用weak变量。
该如何解决呢?当声明protocol的时候,我们可以指定只有class类型的变量可以代理它,这样的话就可以使用weak来修饰了。
protocol ListViewControllerDelegate : class {
func configure(with list : [Any])
}
class ListViewController : UIViewController {
weak var delegate : ListViewControllerDelegate?
override func viewDidLoad() {
super.viewDidLoad()
}
}








