swift2.2当中的inout参数的使用
在Swift中,初次接触inout关键字以及它的用法,可能会让我们想起C/C++中的指针,但实际上Swift中inout只不过是按值传递,然后再写回原变量,而不是按引用传递:
An in-out parameter has a value that is passed in to the function, is modified by the function, and is passed back out of the function to replace the original value.
这样的好处在于它远比使用引用安全。首先举个最简单的例子看一看inout关键字怎么用:
func inc(inout i: Int) {
++i
}
var x = 0
inc(&x)
print(x) // 输出结果:“1”
参数x传入到inc函数中后,在函数内被修改为1,函数返回时这个值(1)覆盖了原来的x的值(0),所以x变成了1。
对比一下另一种同样能在函数内部改变变量值的实现方式——闭包:
func inc() -> () -> Int {
var i = 0 //在inc函数内定义变量i
return { ++i } // 闭包中截获变量i
}
let f = inc()
print(f()) // 输出结果:“1”
print(f()) // 输出结果:“2”
闭包是通过截获外部变量的引用从而实现对变量的修改的,我们通过闭包来证明,inout参数是按值传递的:
func inc(inout i: Int) -> () -> Int {
return { ++i } // 闭包中截获inout参数i
}
var x = 0
let f = inc(&x)
print(f()) // 输出结果:“1”
print(x) // 输出结果:“0”
如果inout参数是按引用传递,因为我们知道闭包会按引用截获变量,所以闭包内的++i语句实际上会影响到我们定义的变量x,因此最后一个输出的结果应该是1,但实际上运行结果是0。
这说明inout参数是按值传递的,我们梳理一下整个过程:
- 首先变量
x的值是0,它作为inout参数传入inc方法中,inc方法内有一个x的副本,闭包截获了这个副本的引用。 - 随后
inc方法方法返回,此时的副本值还是0,所以外部的变量x的值为0。 - 接下来我们调用闭包,副本值被改为1,但是外部的变量
x的值不会受到任何影响,所以它依然为0。
如果在inc方法中返回闭包之前就调用这个闭包,那么外部的变量x的值就会被修改为1,这是因为在函数返回前,副本的值变成了1:
func inc(inout i: Int) -> () -> Int {
let f = { ++i } // 闭包中截获inout参数i
f()
return f
}
var x = 0
let f = inc(&x)
print(x) // 输出结果:“1”
&并不总表示inout
如果在函数声明中,参数是一个UnsafeMutablePointer的指针,那么传递参数的时候也要加上&,这和inout参数看上去用法类似,但实际上这里是按引用传递而不是按值传递。我们可以改写一下之前的inc方法:
func inc(i: UnsafeMutablePointer<Int>) -> () -> Int {
//函数内存储指针i的副本,闭包截获这个副本
return {
i.memory++
return i.memory
}
}
这个方法的使用与之前类似。有兴趣的读者可以自行尝试。这里我们换一种调用方式,传入inc方法的参数不是整数地址,而是数组的地址:
let f: () -> Int
do {
var x = [0]
f = inc(&x)
}
print(f())
swift2.2当中的inout参数的使用的更多相关文章
- MySQL 存储过程传参之in, out, inout 参数用法
存储过程传参:存储过程的括号里,可以声明参数. 语法是 create procedure p([in/out/inout] 参数名 参数类型 ..) in :给参数传入值,定义的参数就得到了值 ou ...
- 替换 window.location当中的某个参数的值(而其它值不变)JS代码
在后台与前台的交互操作中,需要替换location当中的某个参数的值(而其它值不变)时,会用到以下函数: 说明: win:传入的窗口句柄,如,window或window.parent等forceAdd ...
- Swift - 31 - 常量参数, 变量参数和inout参数
//: Playground - noun: a place where people can play import UIKit // swift中默认情况下, 传入的参数是不可以修改的, 也就是l ...
- mysql存储过程中in、out、inout参数使用实际案例
1.参数in的使用(代表输入,意思说你的参数要传到存过过程的过程里面去)//为了避免存储过程中分号(";")结束语句,我们使用分隔符告诉mysql解释器,该段命令是否已经结束了./ ...
- MySQL 存储过程参数用法 in, out, inout
MySQL 存储过程参数有三种类型:in.out.inout.它们各有什么作用和特点呢? 一.MySQL 存储过程参数(in) MySQL 存储过程 “in” 参数:跟 C 语言的函数参数的值传递类似 ...
- java Servlet+mysql 调用带有输入参数和返回值的存储过程(原创)
这个数据访问的功能,我在.NET+Mysql .NET+Sqlserver PHP+Mysql上都实现过,并且都发布在了我博客园里面,因为我觉得这个功能实在是太重要,会让你少写很多SQL语句不说,还 ...
- Swift语言中为外部参数设置默认值可变参数常量参数变量参数输入输出参数
Swift语言中为外部参数设置默认值可变参数常量参数变量参数输入输出参数 7.4.4 为外部参数设置默认值 开发者也可以对外部参数设置默认值.这时,调用的时候,也可以省略参数传递本文选自Swift1 ...
- 笔记:MyBatis Mapper XML文件详解 - 映射和参数
MyBatis 的真正强大在于它的映射语句,也是它的魔力所在.由于它的异常强大,映射器的 XML 文件就显得相对简单.如果拿它跟具有相同功能的 JDBC 代码进行对比,你会立即发现省掉了将近 95% ...
- MySQL 存储过程参数
MySQL 存储过程参数 MySQL存储过程参数简介 在现实应用中,开发的存储过程几乎都需要参数.这些参数使存储过程更加灵活和有用. 在MySQL中,参数有三种模式:IN,OUT或INOUT. IN ...
随机推荐
- 理解并使用.NET 4.5中的HttpClient
HttpClient介绍 HttpClient是.NET4.5引入的一个HTTP客户端库,其命名空间为System.Net.Http..NET 4.5之前我们可能使用WebClient和HttpWeb ...
- Python - 多版本共存与虚拟独立环境
背景 利用pip直接在本地python环境安装包,基本上都安装到了site_package目录下面. 随着安装的包越来越多,依赖关系也就越来越复杂.不同的项目之间,依赖关系也会出现版本冲突的问题. 此 ...
- 利用栈实现算术表达式求值(Java语言描述)
利用栈实现算术表达式求值(Java语言描述) 算术表达式求值是栈的典型应用,自己写栈,实现Java栈算术表达式求值,涉及栈,编译原理方面的知识.声明:部分代码参考自茫茫大海的专栏. 链栈的实现: pa ...
- Mysql学习笔记(八)索引
PS:把昨天的学习内容补上...发一下昨天学的东西....五月三日...继续学习数据库... 学习内容: 索引.... 索引的优点: 1.通过创建唯一索引,可以保证数据库每行数据的唯一性... 2.使 ...
- [Latex]实现行内高亮
Latex的行内高亮 前两天想要在做的小操作系统实验指导书里使用行内高亮,一开始虽然有命令 \mint{Language}|contents| 但是无奈只能实现跳行高亮,即不能实现行内高亮.即代码高亮 ...
- jquery选择器(原创)<二>
jquery选择器,选择接着学: 前面学习了基本选择器中的CSS选择器,现在学层级选择器: 1.子元素选择器 子元素选择器,用于在给定的父元素下,查找这个父元素下面的所有的子元素,语法格式,如下: $ ...
- .net 中读取自定义Config文件
今天做一个windows插件式服务程序,插件有时要读取配置文件的设置,但是服务是动态加载到服务上的,没有办法作到动态修改服务的配置文件(app.config).在.net 2.0中有一个Configu ...
- 用UltraISO制作支持windows 7的U盘启动盘
用UltraISO制作U盘启动盘,有人写过,我也看过,不过依照网上的那些文章,成功的并不多,经过几次试验,在不同的主板环境下成功概率高的方法应该如下: 1. UltraISO建议9.3以上 2. ...
- C#的Raw Socket实现网络封包监视
同Winsock1相比,Winsock2最明显的就是支持了Raw Socket套接字类型,使用Raw Socket,可把网卡设置成混杂模式,在这种模式下,我们可以收到网络上的IP包,当然包括目的不是本 ...
- sql联合查询去除重复计算总和
1.首先来个联合查询 SELECT 字段1, 字段2, 字段3, 字段4 FROM 表1 INNER JOIN 表2 ON 表1.字段x = 表2.字段x x:代表随意的一个,只要在联合查询的两张表都 ...