Swift - 可选项(Optional)

文章目录

  • Swift - 可选项(Optional)
    • 1. 可选项(Optional)
    • 2. 强制解包(Forced Unwrapping)
    • 3. 判断可选项是否包含值
    • 4. 可选项绑定(Optional Binding)
    • 5. 等价写法
    • 6. while循环中使用可选项绑定
    • 7. 空合并运算符 ??(Nil-Coalescing Operator)
      • 7.1 API
      • 7.2 示例
    • 8. 多个??一起使用
    • ??跟if let配合使用
    • if语句实现登陆
    • guard语句
    • 9. 隐式解包(Implicitly Unwrapped Optional)
    • 10. 字符串插值
    • 11. 多重可选项

Swift - 可选项(Optional)

1. 可选项(Optional)

可选项,一般也叫可选类型,它允许将值设置为nil
在类型名称后面加个问号 ? 来定义一个可选项

无法赋值为nil

使用?

没设置初始值时,默认是nil

var age: Int? //默认就是nil
age = 10
age = nil

函数可以返回nil

var array = [1, 15, 40 , 29]
func get(_ index: Int) -> Int? {if index < 0 || index >= array.count {return nil}return array[index]
}
print(get(1))  // Optional(15)
print(get(-1))  // nil
print(get(4))  // nil

通过打印结果可以看出是否是可选类型

var age: Int = 15
print(age)var age1: Int? = 15
print(age1)

2. 强制解包(Forced Unwrapping)

可选项是对其他类型的一层包装,可以将它理解为一个盒子

如果为nil,那么它是个空盒子

如果不为nil,那么盒子里装的是:被包装类型的数据

拿这个例子来说:

var age: Int? //默认就是nil
age = 10
age = nil

var age: Int?相当于是空盒子

age = 10相当于把数据装到盒子里去

age = nil相当于把数据从盒子里面拿掉

如果要从可选项取出被包装的数据(将盒子里装的东西取出来),需要使用感叹号 ! 进行强制解包

var age: Int? = 10
let ageInt: Int = age!
print(age)  // Optional(10)
print(ageInt)  // 10

解包后打印结果就不是Optional

如果对值为nil的可选项(空盒子)进行强制解包,将会产生运行时错误

3. 判断可选项是否包含值

我们可以先判断可选项是否包含值,再进行强制解包

let number = Int("123")
if number != nil {
print("字符串转换整数成功:\(number!)")
} else {
print("字符串转换整数失败")
}
// 字符串转换整数成功:123

4. 可选项绑定(Optional Binding)

可以使用可选项绑定来判断可选项是否包含值
如果包含就自动解包,把值赋给一个临时的常量(let)或者变量(var),并返回true,否则返回false

简单用法

if let number = Int("123") {print("字符串转整数成功:\(number)")// number是强制解包后的Int值// number的作用域仅限这个大括号内
}
else {print("字符串转整数失败")
}
// 字符串转整数成功:123

季节 示例

enum Season: Int {case spring = 1, summer, autumn, winter
}
if let season = Season(rawValue: 6) {switch season {case .spring:print("the season is spring")default:print("th season is other")}
} else {print("no such season")
}
// no such season

5. 等价写法

if let first = Int("4") {if let second = Int("42") {if first < second && second < 100 {print("\(first) < \(second) < 100")}}
}
// 4 < 42 < 100

等价于:

if let first = Int("4"),let second = Int("42"),first < second && second < 100 {print("\(first) < \(second) < 100")
}
// 4 < 42 < 100

6. while循环中使用可选项绑定

遍历数组,将遇到的整数都加起来,如果遇到负数或者非数字,停止遍历

var strs = ["10", "20", "abc", "-20", "30"]var index = 0
var sum = 0
while let num = Int(strs[index]), num > 0 {sum += numindex += 1
}
print(sum)

7. 空合并运算符 ??(Nil-Coalescing Operator)

7.1 API

public func ?? <T>(optional: T?, defaultValue: @autoclosure () throws -> T?) rethrows -> T?public func ?? <T>(optional: T?, defaultValue: @autoclosure () throws -> T) rethrows -> T

7.2 示例

a ?? b

  • a 是可选项
  • b 是可选项 或者 不是可选项
  • b 跟 a 的存储类型必须相同

如果 a 不为nil,就返回 a

如果 anil,就返回 b

如果 b 不是可选项,返回 a 时会自动解包


a、b都是可选项

let a: Int? = 1
let b: Int? = 2
let c = a ?? b

a不为nil,所以cInt?Optional(1)

a是nil,b是可选项

let a: Int? = nil
let b: Int? = 2
let c = a ?? b

anil,返回b,所以cInt?Optional(2)

ab都是nil

let a: Int? = nil
let b: Int? = nil
let c = a ?? b

anil,返回bb也是nil,所以cInt?nil

a是可选项,b不是可选项

let a: Int? = 1
let b: Int = 2
let c = a ?? b

a不为nil,返回a,但是因为b不是可选项,返回a时会自动解包,所以cInt1

anilb不是可选项

let a: Int? = 1
let b: Int = 2
let c = a ?? b

anil,返回bbInt,所以cInt2

如果不使用??运算符

let a: Int? = nil
let b: Int = 2
let c: Int
if let tmp = a {c = tmp
}
else {c = b
}
// c = 2

8. 多个??一起使用

a、b都是可选项

let a: Int? = 1
let b: Int? = 2
let c = a ?? b ?? 3

a不为nil,返回a,得到 => a ?? 3,3Int,所以cInt3

a为nil、b都可选项

let a: Int? = nil
let b: Int? = 2
let c = a ?? b ?? 3

anil,返回b,得到 => b ?? 3,b不为nil,返回b3Int,所以cInt3

a、b都为nil

let a: Int? = nil
let b: Int? = nil
let c = a ?? b ?? 3

anil,返回b,得到 => b ?? 3,bnil,返回3,所以cInt3

??跟if let配合使用

let a: Int? = nil
let b: Int? = 2
if let c = a ?? b {print(c)
}

类似于if a != nil || b != nil

if let c = a, let d = b {print(c)print(d)
}

类似于if a != nil && b != nil

if语句实现登陆

func login(_ info: [String : String]) {var username: Stringif let tmp = info["username"] {username = tmp}else {print("请输入用户名")return}var password: Stringif let tmp = info["password"] {password = tmp}else {print("请输入密码")return}// if username ...// if password ...print("用户名:\(username)", "密码:\(password)", "登录ing")
}
login(["username" : "jack", "password" : "123456"])  // 用户名:jack 密码:123456 登录ing
login(["username" : "jack"])  // 请输入密码

guard语句

语法

guard 条件 else {
// do something....
退出当前作用域
// return、break、continue、throw error
}
  • guard语句的条件为false时,就会执行大括号里面的代码
  • guard语句的条件为true时,就会跳过guard语句
  • guard语句特别适合用来“提前退出”
  • 当使用guard语句进行可选项绑定时,绑定的常量(let)、变量var)也能在外层作用域中使用

使用guard语句改造上面登录代码

func login(_ info: [String : String]) {guard let username = info["username"] else {print("请输入用户名")return}guard let password = info["password"] else {print("请输入密码")return}// if username ...// if password ...print("用户名:\(username)", "密码:\(password)", "登录ing")
}
login(["username" : "jack", "password" : "123456"])  // 用户名:jack 密码:123456 登录ing
login(["username" : "jack"])  // 请输入密码

9. 隐式解包(Implicitly Unwrapped Optional)

  • 在某些情况下,可选项一旦被设定值之后,就会一直拥有值
  • 在这种情况下,可以去掉检查,也不必每次访问的时候都进行解包,因为它能确定每次访问的时候都有值
  • 可以在类型后面加个感叹号 ! 定义一个隐式解包的可选项

前面我们使用如下方式进行强制解包

let num1: Int? = 10
let num2: Int = num1!

这种方式在每次使用的时候,需要使用!来强制解包

我们可以在可选项类型后面加个感叹号 !,是其成为一个隐式解包的可选项,在后续使用过程中它将会自动解包

let num1: Int! = 10
let num2: Int = num1

如果隐式解包nil解包时会报错

我们可以对其进行判空处理

if num1 != nil {print(num1 + 6)  // 16
}
if let num3 = num1 {print(num3)  // 10
}

10. 字符串插值

可选项在字符串插值或者直接打印时,编译器会发出警告

至少有3种方法消除警告
强制解包:

print("my age is \(age!)")  // my age is 10

使用Stringdescribing

print("my age is \(String(describing: age))")  // my age is Optional(10)

空合并运算符??

print("my age is \(age ?? 0)")  // my age is 10

11. 多重可选项

多重可选项,相当于装了多层盒子

var num1: Int? = 10
var num2: Int?? = num1
var num3: Int?? = 10print(num2 == num3) // true


可以使用lldb指令 frame variable –R 或者 fr v -R 查看区别

fr v -R num1
fr v -R num2
fr v -R num3


num1、num3为nil的情况

var num1: Int? = nil
var num2: Int?? = num1
var num3: Int?? = nil

fr v -R num1
fr v -R num2
fr v -R num3

@oubijiexi

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/318025.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CogAgent:开创性的VLM在GUI理解和自动化任务中的突破

尽管LLMs如ChatGPT在撰写电子邮件等任务上能够提供帮助&#xff0c;它们在理解和与GUIs交互方面存在挑战&#xff0c;这限制了它们在提高自动化水平方面的潜力。数字世界中的自主代理是许多现代人梦寐以求的理想助手。这些代理能够根据用户输入的任务描述自动完成如在线预订票务…

GraspNet-1Billion 论文阅读

文章目录 GraspNet-1Billion总体数据集评价指标网络pointnet&#xff1a;Approach Network:Operation Network&#xff1a;Tolerance Network 摘要相关工作基于深度学习的抓取预测算法抓取数据集点云深度学习 GraspNet-1Billion CVPR2020 上海交大 论文和数据集地址&#xff1…

翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深度学习五

合集 ChatGPT 通过图形化的方式来理解 Transformer 架构 翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深度学习一翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深度学习二翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深…

Linux下启动jenkins报错问题解决

jenkins端口报错 java.io.IOException: Failed to start Jettyat winstone.Launcher.<init>(Launcher.java:209)at winstone.Launcher.main(Launcher.java:496)at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)at java.base/jdk.int…

《QT实用小工具·四十八》趣味开关

1、概述 源码放在文章末尾 该项目实现了各种样式的趣味开关&#xff1a; 1、爱心形状的switch开关&#xff0c;支持手势拖动、按压效果 2、线条样式的3种开关 项目demo演示如下所示&#xff1a; 使用方式&#xff1a; 1、sapid_switch文件夹加入工程&#xff0c;.pro文件中…

Go 语言(三)【面向对象编程】

1、OOP 首先&#xff0c;Go 语言并不是面向对象的语言&#xff0c;只是可以通过一些方法来模拟面向对象。 1.1、封装 Go 语言是通过结构体&#xff08;struct&#xff09;来实现封装的。 1.2、继承 继承主要由下面这三种方式实现&#xff1a; 1.2.1、嵌套匿名字段 //Add…

InfluxDB安装使用介绍

1.介绍 InfluxDB是一个由InfluxData开发的开源时序型数据。它由Go写成&#xff0c;着力于高性能地查询与存储时序型数据。InfluxDB被广泛应用于存储系统的监控数据&#xff0c;IoT行业的实时数据等场景。 2.对常见关系型数据库&#xff08;MySQL&#xff09;的基础概念对比 1…

ubuntu22.04 修改内核源码教程

1. 确认当前内核版本 uname -a 2. 去ubuntu官网下载对应版本内核源码 6.5.0-28.29 : linux package : Ubuntu (launchpad.net) 3. 准备编译环境 sudo apt-get install libncurses5-dev libssl-dev build-essential openssl flex bison libelf-dev tar -xzvf linux_6.5.…

8_手眼标定总结_auboi5机械臂与海康平面相机

经过不断地学习与调试&#xff0c;不断地学习网络上其他同志分享的资料&#xff0c;opencv手眼标定迎来了阶段性结束。实际测试结果在机械臂坐标系中X方向差5mm左右。 代码参考《https://blog.csdn.net/wanggao_1990/article/details/81435660》 注意事项&#xff1a; ①标定…

nacos-redis-springboot

新项目 准备工作 nacos 版本 2.0.3 redis 最终版本说明 springcloud-alibaba&#xff1a;2.2.7RELEASE springcloud&#xff1a;Hoxton.SR12 springboot&#xff1a;2.3.12.RELEASE Nacos&#xff1a;2.0.3 步骤 启动nacos和redis 准备nacos配置文件 server: port…

Node私库Verdaccio使用记录,包的构建,推送和拉取

Node私库Verdaccio使用记录&#xff0c;包的构建&#xff0c;推送和拉取 Verdaccio是一个轻量级的私有npm代理注册中心&#xff0c;它可以帮助你在本地搭建一个npm仓库&#xff0c;非常适合企业内部使用。通过使用Verdaccio&#xff0c;你可以控制和缓存依赖包&#xff0c;提高…

基于Pytorch深度学习——多层感知机

本文章来源于对李沐动手深度学习代码以及原理的理解&#xff0c;并且由于李沐老师的代码能力很强&#xff0c;以及视频中讲解代码的部分较少&#xff0c;所以这里将代码进行尽量逐行详细解释 并且由于pytorch的语法有些小伙伴可能并不熟悉&#xff0c;所以我们会采用逐行解释小…

Upload-labs 靶场通关解析(上)

前言 文件上传漏洞是一种常见的网络安全漏洞&#xff0c;存在于许多Web应用程序中。攻击者利用这个漏洞可以上传恶意文件到目标服务器&#xff0c;从而执行各种恶意操作&#xff0c;如执行恶意代码、获取敏感信息、控制服务器等。 文件上传漏洞的原理是&#xff0c;Web应用程…

【精选文献】JAG|基于时序Sentinel-1 SAR影像小农耕作区烟草空间分布制图

目录 文章简介 01 文章摘要 02 研究背景、目标及创新点 03 研究区域与数据集 04 研究方法 05 研究结果 06 研究讨论 07 研究结论 08 文章引用 文章简介 论文名称&#xff1a;Mapping tobacco planting areas in smallholder farmlands using Phenological-Spatial-Te…

jenkins汉化不完全问题解决

jenkins安装完Localization:Chinese(Simplified)中文语言包后&#xff0c;发现是出现汉化不完全或者部分汉化的情况&#xff0c;如下图&#xff1a; 解决方法&#xff1a; 启动命令中指定语言 -Duser.languageen_US.UTF-8 或者 -Duser.languageC.UTF-8原因分析&#xff1a;安…

HarmonyOS Next从入门到精通实战精品课

第一阶段&#xff1a;HarmonyOS Next星河版从入门到精通该阶段由HarmonyOS Next星河版本出发&#xff0c;介绍HarmonyOS Next版本应用开发基础概念&#xff0c;辅助学员快速上手新版本开发范式&#xff0c;共计42课时 第一天鸿蒙NEXT Mac版、Windows版【编辑器】和【模拟器】&a…

限流--4种经典限流算法讲解--单机限流和分布式限流的实现

为什么需要限流 系统的维护使用是需要成本的&#xff0c;用户可能使用科技疯狂刷量&#xff0c;消耗系统资源&#xff0c;出现额外的经济开销问题&#xff1a; 控制成本>限制用户的调用次数用户在短时间内疯狂使用&#xff0c;导致服务器资源被占满&#xff0c;其他用户无…

【Python的魅力】:利用Pygame实现游戏坦克大战——含完整源码

文章目录 一、游戏运行效果二、代码实现2.1 项目搭建2.2 加载我方坦克2.3 加载敌方坦克2.4 添加爆炸效果2.5 坦克大战之音效处理 三、完整代码 一、游戏运行效果 二、代码实现 坦克大战游戏 2.1 项目搭建 本游戏主要分为两个对象&#xff0c;分别是我方坦克和敌方坦克。用户可…

【大模型系列】大模型的上下文长度解释与拓展

文章目录 1 什么是大模型的上下文长度&#xff1f;2 拓展大模型上下文长度的方式参考资料 1 什么是大模型的上下文长度&#xff1f; 大模型的上下文长度&#xff08;Context Length&#xff09;是指在自然语言处理&#xff08;NLP&#xff09;的大型语言模型&#xff08;Large…

Qt Creator导入第三方so库和jar包——Qt For Android

前言 之前了解了在Android Studio下导入so库和jar包&#xff0c;现在实现如何在Qt上导入so库和jar包。 实现 下面是我安卓开发&#xff08;需调用安卓接口的代码&#xff09;的目录&#xff08;图1&#xff09;&#xff0c;此目录结构和原生态环境&#xff08;Android Studi…