在面向过程的编程语言(如C语言)中,结构体用得比较多,但是面向对象之后,如在C++和Objective-C中,结构体已经很少使用了。这是因为结构体能够做的事情,类完全可以取而代之。
而Swift语言却非常重视结构体,把结构体作为实现面向对象的重要手段。Swift中的结构体与C++和Objective-C中的结构体有很大的差别,C++和Objective-C中的结构体只能定义一组相关的成员变量,而Swift中的结构体不仅可以定义成员变量(属性),还可以定义成员方法。因此,我们可以把结构体看做是一种轻量级的类。
Swift中的类和结构体非常类似,都具有定义和使用属性、方法、下标和构造器等面向对象特性,但是结构体不具有继承性,也不具备运行时强制类型转换、使用析构器和使用引用计等能力。
一、类和结构体定义
Swift中的类和结构体定义的语法也是非常相似的。我们可以使用class关键词定义类,使用struct关键词定义结构体,它们的语法格式如下:
class 类名 {
    定义类的成员
}
struct 结构体名 {
    定义结构体的成员
}
从语法格式上看,Swift中的类和结构体的定义更类似于Java语法,不需要像C++和Objective-C那样把接口部分和实现部分放到不同的文件中。
类名、结构体名的命名规范与枚举类型的要求是一样的。下面我们来看一个示例:
class Employee { //定义员工类
    var no : Int = 0 //定义员工编号属性
    var name : String = "" //定义员工姓名属性
    var job : String? //定义工作属性
    var salary : Double = 0 //定义薪资属性

var dept : Department?  //定义所在部门属性
}

struct Department { //定义部门结构体
    var no : Int = 0 //定义部门编号属性
    var name : String = "" //定义部门名称属性
}
Employee是我们定义的类,Department是我们定义的结构体。在Employee和Department中我们只定义了一些属性。关于属性的内容我们将在下一章介绍。
Employee和Department是有关联关系的,Employee所在部门的属性dept与Department关联起来,它们的类图如下图所示。
 
我们可以通过下列语句实例化:
var emp = Employee()
var dept = Department()
Employee()和Department()是调用它们的构造器实现实例化,关于构造器我们会在14.1节介绍。
提示 实例化之后会开辟内存空间,emp和dept被称为“实例”,但只有类实例化的“实例”才能被称为“对象”。事实上,不仅仅是结构体和类可以实例化,枚举、函数类型和闭包开辟内存空间的过程也可以称为实例化,结果也可以叫“实例”,但不能叫“对象”。
二、再谈值类型和引用类型
数据类型可以分为:值类型和引用类型,这是由赋值或参数传递方式决定的。值类型就是在赋值或给函数传递参数时候,创建一个副本,把副本传递过去,这样在函数的调用过程中不会影响原始数据。引用类型就是在赋值或给函数传递参数的时候,把本身数据传递过去,这样在函数的调用过程中会影响原始数据。
在众多的数据类型中,我们只需记住:只有类是引用类型,其他类型全部是值类型。即便结构体与类非常相似,它也是值类型。值类型还包括整型、浮点型、布尔型、字符串、元组、集合和枚举。
Swift中的引用类型与Java中的引用类型是一样的,Java中的类也是引用类型。如果你没有Java经验,可以把引用类型理解为C、C++和Objective-C语言中的指针类型,只不过不需要在引用类型变量或常量前面加星号(*)。
下面我们看一个示例:

[html] view plaincopy
  1. var dept = Department() ①
  2. dept.no = 10
  3. dept.name = "Sales"         ②
  4. var emp = Employee()        ③
  5. emp.no = 1000
  6. emp.name = "Martin"
  7. emp.job = "Salesman"
  8. emp.salary = 1250
  9. emp.dept = dept             ④
  10. func updateDept (dept : Department) {   ⑤
  11. dept.name = "Research"  ⑥
  12. }
  13. println("Department更新前:\(dept.name)")   ⑦
  14. updateDept(dept)                ⑧
  15. println("Department更新后:\(dept.name)")   ⑨
  16. func updateEmp (emp : Employee) {   ⑩
  17. emp.job = "Clerk"       ⑪
  18. }
  19. println("Employee更新前:\(emp.job)")   ⑫
  20. updateEmp(emp)              ⑬
  21. println("Employee更新后:\(emp.job)")   ⑭

上述代码第①~②行创建Department结构体实例,并设置它的属性。代码第③~④行创建Employee类实例,并设置它的属性。
为了测试结构体是否是值类型,我们在第⑤行代码定义了updateDept函数,它的参数是Department结构体实例。第⑥行代码dept.name = "Research"是改变dept实例。然后在第⑦行打印更新前的部门名称属性,在第⑧行进行更新,在第⑨行打印更新后的部门名称属性。如果更新前和更新后的结果一致,则说明结构体是值类型,反之则为引用类型。事实上第⑥行代码会有编译错误,错误信息如下。
Playground execution failed: error: <REPL>:34:15: error: cannot assign to 'name' in 'dept'
    dept.name = "Research"
    ~~~~~~~~~ ^
这个错误提示dept.name = "Research"是不能赋值的,这说明了dept结构体不能修改,因为它是值类型。其实有另外一种办法可以使值类型参数能够以引用类型传递,我们在第9章介绍过使用inout声明的输入输出类型参数,这里需要修改一下代码:
func updateDept (inout dept : Department) {
    dept.name = "Research"
}

println("Department更新前:\(dept.name)")
updateDept(&dept)
println("Department更新后:\(dept.name)")
我们不仅要将参数声明为inout,而且要在使用实例前加上&符号。这样修改后输出结果如下:
Department更新前:Sales
Department更新后:Research
相比之下,第⑩行代码是定义updateEmp函数,它的参数是Employee类的实例,我们不需要将参数声明为inout类型。在第⑪行修改emp没有编译错误,这说明Employee类是引用类型,在调用的时候不用在变量前面添加&符号,见代码第 行。输出结果如下:
Employee更新前:Salesman
Employee更新后:Clerk
这个结果再次说明了类是引用类。
三、引用类型的比较
我们在第4章介绍了基本运算符,提到了恒等于(===)和不恒等于(!===)关系运算符。===用于比较两个引用是否为同一个实例,!===则恰恰相反,它只能用于引用类型,也就是类的实例。
下面我们看一个示例:

[html] view plaincopy
  1. var emp1 = Employee()       ①
  2. emp1.no = 1000
  3. emp1.name = "Martin"
  4. emp1.job = "Salesman"
  5. emp1.salary = 1250
  6. var emp2 = Employee()       ②
  7. emp2.no = 1000
  8. emp2.name = "Martin"
  9. emp2.job = "Salesman"
  10. emp2.salary = 1250
  11. if emp1 === emp2                ③
  12. {
  13. println("emp1 === emp2")
  14. }
  15. if emp1 === emp1                ④
  16. {
  17. println("emp1 === emp1")
  18. }
  19. var dept1 = Department()    ⑤
  20. dept1.no = 10
  21. dept1.name = "Sales"
  22. var dept2 = Department()    ⑥
  23. dept2.no = 10
  24. dept2.name = "Sales"
  25. if dept1 == dept2   //编译失败  ⑦
  26. {
  27. println("dept1 === dept2")
  28. }

上述代码第①行和第②行分别创建了emp1和emp2两个Employee实例。在代码第③行比较emp1和emp2两个引用是否为一个实例。可以看到,比较结果为False,也就是emp1和emp2两个引用不是一个实例,即便是它们内容完全一样,结果也是False,而第④行的比较结果为True。如果我们采用==比较,结果会如何呢?代码如下:
if emp1 == emp2 
{
    println("emp1 === emp2")
}
答案是有如下编译错误。==比较要求两个实例的类型(类、结构体、枚举等)必须要在该类型中重写==运算符,定义相等规则。同样的错误也会发生在第⑦行代码。
Playground execution failed: error: <REPL>:42:9: error: could not find an overload for '==' that accepts the supplied arguments
if emp1 == emp2
   ~~~~~^~~~~~~
代码第⑤行和第⑥行分别创建了dept1和dept2两个Department实例。在代码第⑦行使用==比较dept1和dept2两个值是否相等,不仅不能比较,而且还会发生编译错误,这在上面已经解释过了。
如果我们采用恒等于===比较dept1和dept2,结果会如何呢?代码如下:
if dept1 === dept2
{
    println("dept1 === dept2")
}

我们发现会有编译错误。===不能比较值类型,而Department结构体是值类型,因此不能使用===比较。

更多内容请关注国内第一本Swift图书《Swift开发指南》
本书交流讨论网站:http://www.51work6.com/swift.php
欢迎加入Swift技术讨论群:362298485

欢迎关注智捷iOS课堂微信公共平台

转载于:https://www.cnblogs.com/iOS-Blog/p/3946812.html

Swift结构体与类相关推荐

  1. Swift 结构体和类的最大区别

    2019独角兽企业重金招聘Python工程师标准>>> 结构体类型的赋值是值类型 类的赋值是引用类型 来自为知笔记(Wiz) 转载于:https://my.oschina.net/t ...

  2. swift string转int_swift中结构体和类的区别(值类型和引用类型的区别)

    在swift中结构体和类有着更多的相同之处,在一般的使用中能够做到互相替换.我们可以先看看官方文档的描述: Unlike other programming languages, Swift does ...

  3. Swift 中枚举、结构体、类(enum、struct、class)

    Swift 中枚举.结构体.类(enum.struct.class) Swift中的枚举与OC相比不会自动分配初始值,值的类型不会限定为整数,可以给定关联值类型和具体值(整型.浮点型.字符型(Swif ...

  4. Swift学习笔记-005结构体和类(疯狂swift讲义第二版)

    1.定义类及类相关的一切 定义类的一般语法 [修饰符] class 类名{ //类的语句 } 定义结构体的一般语法 [修饰符] class 类名{ //结构体的语句 } 定义存储属性的一般语法 [修饰 ...

  5. Swift学习笔记 (十八) 结构体和类

    结构体和类作为一种通用而又灵活的结构,成为了人们构建​代码的​基础.你可以使用定义常量.变量和函数的语法,为你的结构 体和类定义属性.添加方法. 与其他编程语⾔所不同的是,Swift 并不要求你为自定 ...

  6. swift学习笔记(9)-结构体和类

    结构体和类对比 Swift 中类和结构体有很多共同点.共同处在于: * 定义属性用于存储值 * 定义方法用于提供功能 * 定义下标操作使得可以通过下标语法来访问实例所包含的值 * 定义构造器用于生成初 ...

  7. 结构体、类的成员对齐

    自然对齐 为了保证CPU的运算稳定和效率,要求基本数据类型在内存中的存储地址必须自然对齐.所谓自然对齐,就是基本数据类型的变量不能简单的存储于内存中的任意地址处,它们的起始地址必须能够被它们的大小整除 ...

  8. 详解结构体、类等内存字节对齐

    先说个题外话:早些年我学C程序设计时,写过一段解释硬盘MBR分区表的代码,对着磁盘编辑器怎么看,怎么对,可一执行,结果就错了.当时调试也不太会,又根本没听过结构体对齐这一说,所以,问题解决不了,好几天 ...

  9. C++:C++语言入门级基础知识考察点回顾之函数、结构体和类

    C++:C++语言入门级基础知识考察点回顾之函数.结构体和类 目录 C++的函数 1.函数的定义.调用,全局变量局部变量 1.1.自定义函数求其倒数

最新文章

  1. linux显示不在sudoers文件中,Ubuntu无法使用sudo提权提示当前用户不在sudoers文件中...
  2. ul弹性怎么一行显示_css3系列之弹性盒子 flex
  3. 修改图片背景_用P图软件将图片背景更改
  4. Mongodb数据库连接
  5. 思维模型篇:四大战略分析工具
  6. RecyclerView(滚动控件)的用法
  7. torch学习笔记--tensor介绍2,对tensor的结构
  8. 论信息化投标低于1元中标值吗?
  9. js进栈出栈_JavaScript调用栈
  10. TensorFlow---(1)开源软件库TensorFlow最全教程和项目列表
  11. 10代i5主频为什么这么低_i5-10210U 笔记本推荐
  12. Codeforces 1132E (看题解)
  13. Excel怎么忽略位置对比两列数据是否相同
  14. [自制]python批量压缩图像
  15. 2022 年 React Native 的全新架构更新
  16. 三进制小数转换C语言
  17. 操作系统实验一 添加内核模块
  18. 计算机安全的基本概念,计算机安全包括哪几个方面
  19. Win10家庭版组策略打不开怎么办
  20. 网课查题公众号 对接查题题库

热门文章

  1. Scala进阶之路-面向对象编程之类的成员详解
  2. [ASP.NET MVC] 利用动态注入HTML的方式来设计复杂页面
  3. SVN工具的使用 和在Eclipse中安装GPD插件:(多步审批流,因此选择使用工作流(JBPM)来实现)...
  4. 十一、Android学习第十天——项目开始(转)
  5. 在linux下配置oracle的远程访问
  6. 【转】算法中时间复杂度概括——o(1)、o(n)、o(logn)、o(nlogn)
  7. 如何接入虹软免费人脸识别SDK
  8. Windows 环境 Tomcat 的 HTTPS 单向认证和双向认证的配置
  9. shell脚本编程《linux下kvm虚拟机的创建、开启、显示、停止、重置》
  10. 在一个字符串寻找另一个字符串,并且输出短字符串头字母在长字符串的下标...