指針類型
是不安全的
1.UnsafePointer<Pointee> 類似于const Pointee * 只可以用來訪問內(nèi)存 不可以修改
2.UnsafeMutablePointer<Pointee> 類似于Pointee * 可以用來訪問內(nèi)存 可以修改
3.UnsafeRawPointer 類似于const void * 不支持泛型
4.UnsafeMutableRawPointer 類似于void * 不支持泛型
var age = 10
func test1(_ ptr: UnsafeMutablePointer<Int>){
ptr.pointee += 10
}
func test2(_ ptr: UnsafePointer<Int>){
print(ptr.pointee)
//ptr.pointee 取出指針指向的數(shù)據(jù)
}
test1(&age)
test2(&age)
print(age)
20
20
ptr.pointee 取出指針指向的數(shù)據(jù)
func test3(_ ptr: UnsafeRawPointer){
let d = ptr.load(as: Int.self)
print("test3",d)
}
func test4(_ ptr: UnsafeMutableRawPointer){
//不支持泛型 那么就要自己設(shè)置是什么類型 傳入什么值
ptr.storeBytes(of: 30, as: Int.self)
}
test3(&age)
test4(&age)
print(age)
ptr.load(as: Int.self) 指針指向的Int數(shù)據(jù)
ptr.storeBytes(of: 30, as: Int.self) 把數(shù)據(jù)存儲(chǔ)到指針指向的空間
指針的應(yīng)用
var arr = NSArray(objects: 11,22,33,44)
arr.enumerateObjects { (element, idx, stop) in
print(idx,element)
if idx == 2 {
stop.pointee = true
}
}
stop.pointee 代表執(zhí)行完大括號(hào)之后再判斷條件是否滿足 是否繼續(xù)執(zhí)行
可讀行更好
var arr = NSArray(objects: 11,22,33,44)
for (idx,element) in arr.enumerated() {
print(idx,element)
if idx == 2 {
break
}
}
獲得指向某個(gè)變量的指針
var age = 10
var ptr = withUnsafePointer(to: &age) { $0 }
var ptr2 = withUnsafeMutablePointer(to: &age) { $0
}
ptr2.pointee = 30
print(age)
var age = 10
var ptr = withUnsafePointer(to: &age) { UnsafeRawPointer($0)
}
var ptr2 = withUnsafeMutablePointer(to: &age) { UnsafeMutableRawPointer($0)
}
ptr2.storeBytes(of: 30, as: Int.self)
print(ptr.load(as: Int.self))
print(age)
獲得指向堆空間實(shí)例的指針
class Person {
var age: Int
init(age: Int) {
self.age = age
}
}
var person = Person(age: 21)
var ptr = withUnsafePointer(to: &person) { UnsafeRawPointer($0)
}
//ptr 指向person這個(gè)指針變量 存放person這個(gè)指針變量的地址值
//ptr.pointee 等價(jià)于person
print(ptr)
var ptr2 = UnsafeMutableRawPointer(bitPattern: ptr.load(as: UInt.self))
print(ptr2 ?? 2)
//ptr2 堆空間地址值 Person的地址值
創(chuàng)建指針
var ptr = malloc(16)
//往堆空間里面放入數(shù)據(jù)
ptr?.storeBytes(of: 10, as: Int.self)
ptr?.storeBytes(of: 20, toByteOffset: 8, as: Int.self)
print(ptr!)
free(ptr)
var ptr = UnsafeMutableRawPointer.allocate(byteCount: 16, alignment: 1)
//指針偏移 8個(gè)字節(jié)
ptr.advanced(by: 8)
//把22存放在后8個(gè)字節(jié)
ptr.advanced(by: 8).storeBytes(of: 22, as: Int.self)
//釋放
ptr.deallocate()
//capacity 是容量 相當(dāng)于Int的8個(gè)字節(jié) * 3 = 24個(gè)字節(jié)
var ptr = UnsafeMutablePointer<Int>.allocate(capacity: 2)
ptr.pointee = 10
ptr.advanced(by: 8).pointee = 20
//初始化內(nèi)存 用10初始化前8個(gè)字節(jié)
ptr.initialize(to: 10)
//連續(xù)的是2個(gè)8個(gè)字節(jié)的10 初始化內(nèi)存
ptr.initialize(repeating: 10, count: 2)
//successor 后記 跳8個(gè)字節(jié)(如果是Int) 下一個(gè)8個(gè)字節(jié)初始化10
ptr.successor().initialize(to: 10)
//取出下一個(gè)字節(jié)的內(nèi)容
print(ptr.successor().pointee)
print((ptr + 1).pointee)
print(ptr[0])
print(ptr[1])
//反初始化 防止內(nèi)存泄露
ptr.deinitialize(count: 3)
//釋放
ptr.deallocate()
class Person {
var age: Int
var name: String
init(age: Int, name: String) {
self.age = age
self.name = name
}
deinit {
print(name,"deinit")
}
}
var ptr = UnsafeMutablePointer<Person>.allocate(capacity: 3)
ptr.initialize(to: Person(age: 20, name: "jack"))
(ptr + 1).initialize(to: Person(age: 21, name: "Rose"))
(ptr + 2).initialize(to: Person(age: 22, name: "Kate"))
//ptr[2].name
ptr.deinitialize(count: 3)
ptr.deallocate()
初始化多少內(nèi)存 就要反初始化多少內(nèi)存 否則 內(nèi)存泄露
指針之間的轉(zhuǎn)換
var ptr = UnsafeMutableRawPointer.allocate(byteCount: 16, alignment: 1)
//轉(zhuǎn)換為UnsafeMutablePointer指針
var ptr2 = ptr.assumingMemoryBound(to: Int.self)
//ptr2 是泛型的 ptr2 + 1 就是移動(dòng)8個(gè)字節(jié)
var ptr4 = ptr2 + 1
//ptr 不是泛型的 就是移動(dòng)8個(gè)字節(jié)
var ptr3 = (ptr + 8).assumingMemoryBound(to: Int.self)
//強(qiáng)制轉(zhuǎn)換
var ptr = UnsafeMutableRawPointer.allocate(byteCount: 16, alignment: 1)
var ptr5 = unsafeBitCast(ptr, to: Int.self)
unsafeBitCast 是忽略數(shù)據(jù)類型的強(qiáng)制轉(zhuǎn)換 不會(huì)因?yàn)閿?shù)據(jù)類型的變化而改變?cè)瓉淼膬?nèi)存數(shù)據(jù)
unsafeBitCast
var person = Person()
//把person里面的內(nèi)容放入ptr 內(nèi)存數(shù)據(jù)是一樣的 類型不同
var ptr = unsafeBitCast(person, to: UnsafeRawPointer.self)
print(ptr)