类和结构体


最使人疲惫的往往不是道路的遥远,而是你心中的郁闷;最使人颓废的往往不是前途的坎坷,而是你自信的丧失;最使人痛苦的往往不是生活的不幸,而是你希望的破灭;最使人绝望的往往不是挫折的打击,而是心灵的死亡


前言

Swift语言同时支持类和结构体两种数据结构,类是面向对象编程中非常重要的一个概念,而结构体是从C语言中遗留下来的一种组合数据结构。而Swift语言中结构体的功能相对于C语言有了一定的提升,以至于在Swift语言中二者非常相似,常常令人难以取舍,今天我们就来揭示二者异同以及在实际开发中的取舍问题。

相同点

在Swift语言中,结构体和类都可以:

  • 都可以定义方法,属性
  • 都能继承或是实现某个协议
  • 都有自己的初始化方法
  • 都支持下标访问语法
  • 都支持扩展

二者的定义方式是一样的:

class A {
    var p1:String?
    var p2:String?
    
    func f() {
      print("这是类的定义")
    }
}

struct A1 {
    var p1:String?
    var p2:String?
    
    func f() {
        print("这是结构体的定义")
    }
}

不同点

尽管有很多相似的地方,但类和结构体还是有很多不同的地方,下面就让笔者为你一一道来!

  • 类支持继承
  • 支持deInit()方法(C++语言中叫析构函数)释放资源
  • 引用计数器允许对一个类的多次引用
class B1 {
    var x:Int?
    var y:Int?
    
    init() {
        print("初始化方法,初始化时调用")
    }
    
    deinit {
        print("反初始化方法,释放资源时调用")
    }
    
    func f() {
        print("B1->f()")
    }
}

class B2 : B1 {
    override func f() {
        print("B2->f()")
    }
}

let b1:B1 = B2()
b1.f()

struct B11 {
    var x:Int?
    var y:Int?
    
    init() {
        print("初始化方法,初始化时调用")
    }
    
    // 没有deInit方法
//    deinit {
//        
//    }
}

// 不支持继承
//struct B12 : B11 {
//
//}

除了上面的不同点之外,类和结构体还有几处很重要的不同点:

  • 编译器为类自动生成不带参数的初始化方法,而为结构体生成带所有属性的初始化方法
  • 结构体实例是值类型,类实例是引用类型
// 编译器为二者自动生成初始化函数
class C1 {
    var a:String?
    var b:String?
}

struct C11 {
    var a:String?
    var b:String?
}

let c1:C1 = C1()
c1.a = "a"
let c11:C11 = C11(a: "a", b: "b")

// 类实例是引用类型
// 改变c2实例的值也会改变c1的值
let c2:C1 = c1
print("a=\(c1.a!)")
c2.a = "aaa"
print("a=\(c1.a!)")
// 结构体实例是值类型
// 改变c22的值不会改变c11的值
var c22:C11 = c11
print("a=\(c11.a!)")
c22.a = "aaa"
print("a=\(c11.a!)")

下面,我们用一张表格来概括类和结构体的不同:

不同点 结构体
默认初始化函数 带全部参数 不带参数
实例类型 值类型 引用类型
继承 不支持 支持

如何取舍

通过上面的分析,我们可以知道,结构体在设计上更偏向于数据的封装。而类除了数据的封装之外,还偏向于针对数据的处理。在类和结构体不同点中,最重要的一点是:类实例是引用类型,而结构体实例是值类型,如果是数据的传递过程中,不希望影响原数据的值,请考虑结构体,而如果本身考虑是对同一个实例进行操作的话,则请考虑使用类。还有一个非常重要的点是:结构体不能实现继承,多态等面向对象特性,因此,如果考虑扩展以及面向对象特性的话,也请使用类型。

概括来讲:如果仅仅是针对数据的封装的话,使用结构体即可。而如果考虑运行时特性以及类型的扩展的话请使用类。

如果你喜欢我的这篇文章,请点击文章右上方的添加关注。如果你想和更多的人一起讨论Swift语言,请加入我的Swift交流群,我们等着你。
也欢迎你fork这篇文章的源码仓库:https://github.com/yuanhoujun/Swift.git。更多的意见和讨论请在文章下方的评论里面告诉我

你可能感兴趣的:(类和结构体)