swift 中“?”和“!”区别以及相关用法

2024-09-03 20:58
文章标签 用法 区别 相关 swift

本文主要是介绍swift 中“?”和“!”区别以及相关用法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近也在学习swift的只是,已经是学完了闭包的用法,但回头想想之前学的一些知识感觉又望的差不多了。现在自己总结一些swift中?、!和as的相关用法。废话不多说了直接来吧!

一、swift语言使用var定义变量,但和别的语言不同,swift里不会自动给变量赋初始值,也就是说变量不会有默认值,所以要求使用变量之前必须要对其初始化。如果在使用之前不进行初始化的就会报错:

      

   其实Optional是一个enum,里面有None和Some两种类型。所谓的nil就是Optional.None,非nil就是Optional.Some,然后会通过Some (T)包装(wrap)原始值,这也是为什么在使用Optional的时候要拆包(从enum里取出来原始值)的原因,也是PlayGround会把Optional值显示为类似{Some “hello world”}的原因,这里是enum Optional的定义:

enum Optional<</span>T> : LogicValue, Reflectable {

            case None

            case Some(T)

            init()

            init(_ some: T)

            

            /// Allow use in a Boolean context.

            func getLogicValue() -> Bool

            

            /// Haskell's fmap, which was mis-named

            func map<</span>U>(f: (T) -> U) -> U?

            func getMirror() -> Mirror

        }

声明为Optional只需要在类型后面紧跟一个?即可。eg:

  var strValue: String?

上面这个Optional的声明,意思不是“我声明了一个Optional的String值”,而是“我声明了一个Optional类型值,它可能包含一个String值,也可能什么都不包含”,也就是说实际上我们声明的Optional类型,而不是声明一个String类型,这一点很重要。

一旦声明为Optional的,如果不显示的赋值就会有个默认值nil。判断一个Optional的值是否有值,可以用if来判断:

if strValue {

   //do sth with strValue

}

使用Optional值得时候需要在具体的操作,比如调用方法、属性、下标索引等前面加上一个?,如果是nil指,也就是Optional.None, 会跳转过后面的操作不执行,如果有值,就是Optional.Some,可能就会拆包(unwrap),然后对拆包后的指执行后面的操作,来保证执行这个操作的安全性,eg:

let hashValue = strValue?.hsahValue

strValue是Optional的字符串,如果strValue是nil,则hashValue也为nil,如果strValue不为nil,hsahValue就是strValue字符串的哈希值(Optional wrap后的值),另外,?还可以用在安全地调用protocol类型方法上,eg:

        @objc protocol Downloadable {

            @optional func download(toPath: String) -> Bool;

        }

        

        @objc class Content: Downloadable {

            //download method not be implemented

        }

        

        var delegate: Downloadable = Downloadable()

        delegate.download?("some path")


因为上面的delegate是Downloadable类型的,它的download方法是optional,所以它的具体实现有没有download方法是不确定的。swift提供了一种在参数括号前面加上一个?的方式安全地调用protocol的optional方法。另外如果你需要向下面这样向下转型(Downcast),可能会用到as?:

        if let dataSource = object as? UITableViewDataSource {

            let rowsInFirstSection  = dataSource.tableView(tableView, numberOfRowsInSection: 0)

        }

到这里我们看到了?的几种使用场景:

1.声明Optional值变量

2.用在对Optional值操作中,用来判断是否能相应后面的操作

3.用于安全调用protocol的optional方法

4.使用as?向下转型(Downcast)

另外,对于Optional值,不能直接进行操作,否则会报错:

        //error: 'String?' does not have a member named 'hashValue'

        //let hashValue = strValue.hashValue

        //                ^        ~~~~~~~~~

        

        let hashValue = strValue.hashValue

上面提到的Optional值需要拆包(unwrap)后才能得到原来值,然后才能对其操作,那怎么来拆包呢?拆包提到了几种方法,一种是Optional Binding,eg:

        if let str = strValue {

            let hashValue = str.hashValue

        }

二、还有一种是在具体的操作前添加!符号。很诡异  eg:

let hashValue = strValue!.hashValue   

这里的!表示“我确定这里的strValue一定是非nil的, 尽管用吧”, 比如这种情况:

        if strValue {

            let hashValue = strValue!.hashValue

        }

{}里的strValue一定是非nil的,所以就能直接上!,强制拆包(unwrap)并执行后面的操作。当然如果不加判断,strValue不小心为nil的话,就会报错,crash掉。

场景再现:我们有一个自定义的MyViewController类,类中有个一属性是myLabel,myLabel是在viewdidLoad中进行初始化。因为是在viewDidLoad中初始化,所以能直接声明为普通值:var myLabel : UILabel, 因为非Optional的变量必须在声明是或者构造器中进行初始化,但我们是想在viewDidLoad中初始化,所以就只能声明为

Optional:var myLabel : UILabel?,虽然我们确定在viewDidLoad中会初始化,并且在viewDidLoad的声明周期内不会置为nil,但是在对myLabel操作时,每次依然要加上!来强制拆包(在读取值得时候,也可以用?) eg:

        myLabel!.text = "text"

        myLabel!.frame = CGRectMake(0, 0, 10, 10)

            ...

对于这种类型的值,我们可以直接这么声明: var myLabel : UILabel !, 这就是特殊的Optional,称为Implicitly Unwrapped Optionals, 译为隐式拆包的Optional,就等于说你每次对这种类型操作时,都会在操作前不上一个!进行拆包,然后在执行后面的操作,当然如果该值是nil,也一样会报错crash掉。

        var myLabel: UILabel//!相当于下面这种写法的语法糖

        var myLabel: ImplicitlyUnwrappedOptional<</span>UILabel>

那么!大概就这么个情况:

1.强制对Optional值进行拆包(unwrap)

2.声明Implicitly Unwrapped Optionals值,一般用于类中的属性

对于swift的学习还有待提高,欢迎正在学习swift的同学进行交流

这篇关于swift 中“?”和“!”区别以及相关用法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1134029

相关文章

sqlite3 相关知识

WAL 模式 VS 回滚模式 特性WAL 模式回滚模式(Rollback Journal)定义使用写前日志来记录变更。使用回滚日志来记录事务的所有修改。特点更高的并发性和性能;支持多读者和单写者。支持安全的事务回滚,但并发性较低。性能写入性能更好,尤其是读多写少的场景。写操作会造成较大的性能开销,尤其是在事务开始时。写入流程数据首先写入 WAL 文件,然后才从 WAL 刷新到主数据库。数据在开始

native和static native区别

本文基于Hello JNI  如有疑惑,请看之前几篇文章。 native 与 static native java中 public native String helloJni();public native static String helloJniStatic();1212 JNI中 JNIEXPORT jstring JNICALL Java_com_test_g

bytes.split的用法和注意事项

当然,我很乐意详细介绍 bytes.Split 的用法和注意事项。这个函数是 Go 标准库中 bytes 包的一个重要组成部分,用于分割字节切片。 基本用法 bytes.Split 的函数签名如下: func Split(s, sep []byte) [][]byte s 是要分割的字节切片sep 是用作分隔符的字节切片返回值是一个二维字节切片,包含分割后的结果 基本使用示例: pa

两个月冲刺软考——访问位与修改位的题型(淘汰哪一页);内聚的类型;关于码制的知识点;地址映射的相关内容

1.访问位与修改位的题型(淘汰哪一页) 访问位:为1时表示在内存期间被访问过,为0时表示未被访问;修改位:为1时表示该页面自从被装入内存后被修改过,为0时表示未修改过。 置换页面时,最先置换访问位和修改位为00的,其次是01(没被访问但被修改过)的,之后是10(被访问了但没被修改过),最后是11。 2.内聚的类型 功能内聚:完成一个单一功能,各个部分协同工作,缺一不可。 顺序内聚:

log4j2相关配置说明以及${sys:catalina.home}应用

${sys:catalina.home} 等价于 System.getProperty("catalina.home") 就是Tomcat的根目录:  C:\apache-tomcat-7.0.77 <PatternLayout pattern="%d{yyyy-MM-dd HH:mm:ss} [%t] %-5p %c{1}:%L - %msg%n" /> 2017-08-10

Node Linux相关安装

下载经编译好的文件cd /optwget https://nodejs.org/dist/v10.15.3/node-v10.15.3-linux-x64.tar.gztar -xvf node-v10.15.3-linux-x64.tar.gzln -s /opt/node-v10.15.3-linux-x64/bin/npm /usr/local/bin/ln -s /opt/nod

git ssh key相关

step1、进入.ssh文件夹   (windows下 下载git客户端)   cd ~/.ssh(windows mkdir ~/.ssh) step2、配置name和email git config --global user.name "你的名称"git config --global user.email "你的邮箱" step3、生成key ssh-keygen

Android fill_parent、match_parent、wrap_content三者的作用及区别

这三个属性都是用来适应视图的水平或者垂直大小,以视图的内容或尺寸为基础的布局,比精确的指定视图的范围更加方便。 1、fill_parent 设置一个视图的布局为fill_parent将强制性的使视图扩展至它父元素的大小 2、match_parent 和fill_parent一样,从字面上的意思match_parent更贴切一些,于是从2.2开始,两个属性都可以使用,但2.3版本以后的建议使

Collection List Set Map的区别和联系

Collection List Set Map的区别和联系 这些都代表了Java中的集合,这里主要从其元素是否有序,是否可重复来进行区别记忆,以便恰当地使用,当然还存在同步方面的差异,见上一篇相关文章。 有序否 允许元素重复否 Collection 否 是 List 是 是 Set AbstractSet 否

javascript中break与continue的区别

在javascript中,break是结束整个循环,break下面的语句不再执行了 for(let i=1;i<=5;i++){if(i===3){break}document.write(i) } 上面的代码中,当i=1时,执行打印输出语句,当i=2时,执行打印输出语句,当i=3时,遇到break了,整个循环就结束了。 执行结果是12 continue语句是停止当前循环,返回从头开始。