Swift Memory Safty Swift 内存安全机理(乱译)

前言

乱译:不是规规矩矩的翻译,主要目的是为了学知识。但也是无奈之举,水平有限,我不会啊。有些地方我加入了自己的理解,关于新的知识点我会翻译的详细些,旧知识我就概括翻译,不懂的就不翻译。
官网 : Memory Safety

Beginning

默认情况下,Swift 会防止不安全的编码行为发生。例如:
1 .确保变量在初始化之后才能使用
2 .内存(对象)释放 之后,是不可访问的
3 .数组下标会做越界检测

Swift 可以确保同一内存多个访问不发生冲突。是这样做到的:
把一个内存地址改到一个唯一的地址。 啥,啥,啥!不会翻译,好像是说,每一个访问的代码在修改对象之前,都会修改访问对象的地址到一个只有它自己可以访问的唯一的地址。 这样就不冲突了(后面发现它这是针对单线程的,一开始,误以为是多线程读到这吓一跳)

Swift 的这些内存管理都是自动的,如果我们的代码包含了冲突,会有编译时错误或运行时错误,但是我们应知其然知其所以然。这就是这篇文章的目的。

Understanding Conflicting Access to Memory 理解内存访问冲突

当我们给变量赋值或向函数传参时,就会发生内存访问。如下代码包含了内存读访问和写访问

// one 被写入内存
var one = 1
// one 从内存中读出来
print("We're number \(one)!")

下面举个了账单的读写冲突的例子,旧知识,老生常谈的,不翻译了,。。。

note
这里讨论的是单线程的情况,不包含并发编程和多线程情况。
如果你是在单线程时,写了内存访问冲突的代码,Swift会自动的在编译或运行时报错。
如果你多线程编程请使用 Thread Sanitizer 帮助你检测线程冲突。

Characteristics of Memory Access 内存访问的特征

相当于介绍了内存访问:问自己什么是内存访问,便于理解。

  • 至少一个在读
  • 多个同时写
  • 他们期间,发生重叠。读的期间,写的期间。

想一想,有些读写是瞬间的有些并不是瞬间的,不能理解看完后面的部分,就会明白

访问有瞬间访问和长期访问之分,当然,瞬间访问是不会冲突。大部分内存访问都是瞬间访问,例如下面的例子

func oneMore(than number: Int) -> Int {
    return number + 1
}

var myNumber = 1
myNumber = oneMore(than: myNumber)
print(myNumber)
// Prints "2"

但是,重点!重点!重点!有几种方式的内存访问是长期访问,在这个期间,其他代码可以执行。

和瞬间访问不同,对长期访问来说,其他代码是可以在长期访问开始但没有结束期间执行的。我们把这叫做 overlap 覆盖

长期访问最基本的一个例子是在使用 in-out 参数时。

Conflicting Access to In-Out Paramters In-Out 参数访问冲突

函数对它所有的 in-out 参数 持有长期写的访问,持有开始于非 in-out 参数的 evaluated (啥啥啥,不会翻译了,就记住从函数开始到函数结束)(这是Swift语言决定的,我说的哈哈)

长期写访问的一个结论是 :你不能访问通过 in-out 修饰传过来的参数,即使它本来可以访问。例如下面的例子,你一运行就 crash 报错: Simultaneous accesses to 0x100006728, but modification requires exclusive access.

var stepSize = 1

func increment(_ number: inout Int) {
    number += stepSize
}

increment(&stepSize)
// Error: conflicting accesses to stepSize

虽然 stepSize is a global variable, 本来可以访问,但这里不能访问了!看下图很容易理解


1547091660951.jpg

一种解决方案是:对 setpSize 显示 copy

// Make an explicit copy.
var copyOfStepSize = stepSize
increment(&copyOfStepSize)

// Update the original.
stepSize = copyOfStepSize
// stepSize is now 2

长期写访问的另一个关于 in-out 参数的结论是:不可以对有多个 in-out 参数的函数传递一个变量。看例子就明白意思了

func balance(_ x: inout Int, _ y: inout Int) {
    let sum = x + y
    x = sum / 2
    y = sum - x
}
var playerOneScore = 42
var playerTwoScore = 30
balance(&playerOneScore, &playerTwoScore)  // OK
balance(&playerOneScore, &playerOneScore)
// Error: conflicting accesses to playerOneScore

Conflicting Access to self in Methods 方法中 self 的访问冲突
结构体的可变方法在被调用期间对 self 持有读的访问。

extension Player {
    mutating func shareHealth(with teammate: inout Player) {
        balance(&teammate.health, &health)
    }
}

var oscar = Player(name: "Oscar", health: 10, energy: 10)
var maria = Player(name: "Maria", health: 5, energy: 10)
oscar.shareHealth(with: &maria)  // OK

如果,你把 oscar 作为参数传给 sheareHealth(with:),会有冲突:

oscar.shareHealth(with: &oscar)
// Error: conflicting accesses to oscar

shareHealth 对 self 有读的访问,in-out 对 teammate 有读的访问。self 和 teammate 都是 oscar 产生 冲突!这一点原文的图好像画的有点问题,文字表述是对的。

Conflicting Access to Properties 属性访问冲突
对于值类型像 structures,tuple,enumerations 他们是由单个的值组成。任何一个属性的修改,都会对整个值持有读或写的访问。例如下面会产生些访问冲突

var playerInformation = (health: 10, energy: 20)
balance(&playerInformation.health, &playerInformation.energy)
// Error: conflicting access to properties of playerInformation

同时持有 playerInfomation


var holly = Player(name: "Holly", health: 10, energy: 10)
balance(&holly.health, &holly.energy)  // Error

同时持有 holly

但是在实际开发中,有时结构体属性的覆盖访问是安全的。如下例子:

func someFunction() {
    var oscar = Player(name: "Oscar", health: 10, energy: 10)
    balance(&oscar.health, &oscar.energy)  // OK
}

上面这个例子中,oscar 是局部变量 health 和 energy 这两个储属性没有在任何地方交互,所以编译器判断这个内存覆盖访问是安全的。

Memory safety 的目的是为了保证内存访问的安全,但是唯一访问的要求更加的严格。Swift 允许非唯一访问内存,只要它能够证明内存是安全的。
对于结构体,满足下面条件的就可以证明是安全的

  • 你正在访问的的只有实例的存储属性,不包括计算属性和类属性
  • 结构体是局部变量
  • 结构体没有被闭包捕获,或者被非逃逸闭包捕获。 (捕获就是放在闭包你面,我说的哈哈)

如果编译器不能证明访问是安全的,它就不允许访问,就会报错!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 193,968评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,682评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,254评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,074评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,964评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,055评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,484评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,170评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,433评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,512评论 2 308
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,296评论 1 325
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,184评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,545评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,150评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,437评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,630评论 2 335