在我们实际开发的过程中,类和结构体是人们创建代码时所用的一种通用且灵活构造体,为了在类和结构体中实现各种功能,我们必须要严格按照对于常量,变量以及函数所规定的语法规则来定义属性和添加方法.

但是在Swift中,并不要求你为自定义类和结构去创建独立的接口和实现文件.

PS: 通常一个类的实例被称为对象,然而在Swift 中,类和结构体的关系要比在其他语言中更加的密切,本章中所讨论的大部分功能都可以用在类和结构体上,因此,我们会主要使用实例而不是对象.

1.类和结构体的对比

类和结构体的共同点:
1.定义属性用于储存值
2.定义方法用于提供功能
3.定义下标用于通过下标语法访问值
4.定义初始化器用于生成初始化值
5.通过扩展以增加默认实现的功能
6.符合协议以对某类提供标准功能

与结构体相比,类还有如下的附加功能:
1.继承允许一个类继承另一个类的特征
2.类型转换允许在运行时检查和解释一个类实例的类型
3.取消初始化器允许一个类实例释放任何其所被分配的资源
4.引用计数允许对一个类的多次引用

PS: 结构体总是通过被复制的方式在代码中传递,因此请不要使用引用计数.

2.类和结构体的定义以及使用

struct Resolution {
    var width = 0
    var height = 0
}

class VideoMode {
    var resolution = Resolution()
    var interlaced = false
    var frameRate = 0.0
    var name: String?
}

let someResolution = Resolution()
let someVideoMode = VideoMode()

println("The width of someResolution is \(someResolution.width)")
// 打印出来的结果: The width of someResolution is 0

println("The width of someVideoMode is \(someVideoMode.resolution.width)")
// 打印出来的结果: The width of someVideoMode is 0

如果我们要修改结构体或者类里面的值,我们可以使用点语法来进行修改,比如:

someVideoMode.resolution.width = 1280

println("The width of someVideoMode is \(someVideoMode.resolution.width)")
// 打印出来的结果: The width of someVideoMode is 1280

如果我们需要给结构体里面的成员赋值的话,我们需要通过某些方法来间接赋值,比如:

let vga = Resolution(width: 540,height: 680)
println(vga.width)
// 打印出来的结果: 540

关于结构体的实例化,后面会继续讲解.

3.结构体和枚举是值类型

在 Swift 中,所有的基本类型: 整数(Integer),浮点数(floating-point),布尔值(Booleans),字符串(string),数组(array)和字典(dictionaries),都是值类型,并且都是以结构体的形式在后台所实现,所有的结构体和枚举都是值类型,这意味着它们的实例,以及实例中所包含的任何值类型属性,在代码中传递的时候都会被复制,让我们来看一个例子:

let hd = Resolution(width: 1920,height: 1080)

var cinema = hd

cinema.width = 2048

println("cinema is Now \(cinema.width) pixels wide")
println("hd is still \(hd.width ) pixels wide")
// 打印出来的结果: 
// cinema is Now 2048 pixels wide
// hd is still 1920 pixels wide

同样的,枚举也是如此:

enum Compasspoint {
    case north,South,East,West
}

var currentDirection = Compasspoint.West
let rememberedDirection = currentDirection

currentDirection = .East

if rememberedDirection == .West {
    println("The remembered direction is still .West")
}
// 打印出来的结果: The remembered direction is still .West

PS: 例子中rememberedDirection 被赋予了 currentDirection 的值(value),实际上它被赋予的是值(value)的一个拷贝,赋值过程结束后再修改 currentDirection 的值并不影响rememberedDirection 所储存的原始值(value)的拷贝.

4.类是引用类型

为什么说类是引用类型呢,其实在我们定义类的时候,会在里面设置一些参数,根据个人的需求来决定里面的参数是否需要修改,如果该类在后期还要继续使用的话,那么里面的参数就要设置成变量,比如:

struct Resolution {
    var width = 0
    var height = 0
}

class VideoMode {
    var resolution = Resolution()
    var interlaced = false
    var frameRate = 0.0
    var name: String?
}

let hd = Resolution(width: 1920,height: 1080)
let tenEighty = VideoMode()
tenEighty.resolution = hd
tenEighty.interlaced = true
tenEighty.name = "1080i"
tenEighty.frameRate = 25.0

同样的,如果后续还要继续使用,那么还可以继续引用,并且修改里面的参数值,比如:

let alsoTenEighty = tenEighty
alsoTenEighty.frameRate = 30.0

println("The frameRate property of tenEighty is Now \(tenEighty.frameRate)")
// 打印出来的结果: The frameRate property of tenEighty is Now 30.0

5.恒等于运算符

因为类是引用类型,有可能有多个常量和变量在后台同时引用某一个类实例,但对于结构体或者枚举类型来说的话,是不存在这个的,因为它们作值类型,在被赋予到常量,变量或者传递到函数时,总是会被拷贝.

为了能够判定两个常量或者变量是否引用同一个类,我们可以使用Swift中的两个内置运算符,那就是等价于(===),不等价于(!==),来看看例子:

if tenEighty === alsoTenEighty {
    println("tenTighty and alsoTenEighty refer to the same Resolution instance.")
}
// 打印出来的结果: alsoTenEighty and alsoTenEighty refer to the same Resolution instance.

6.指针

这里稍微补充一下,一个 Swift 常量或者变量引用一个引用类型的实例与 C 语言中的指针类似,不同的是并不直接指向内存中的某个地址,而且也不要求你使用星号(*)来表明你 在创建一个引用,Swift 中这些引用与其它的常量或变量的定义方式相同.

7.类和结构体的选择

在Swift编程中,如果我们要自定义数据类型,我们有两种方式可选,一种可以选择类,另一种就是结构体,虽然他们同样是可以存储数据,但他们两种互不相同,类是通过引用传递,而结构体是通过值来传递,所以两者可以满足不同的需求,按照官方的说法就是当符合一条或多条以下条件时,请考虑构建结构体.

原因:
1.结构体的主要目的是用来封装少量相关简单数据值。
2.有理由预计一个结构体实例在赋值或传递时,封装的数据将会被拷贝而不是被引用。
3.任何在结构体中储存的值类型属性,也将会被拷贝,而不是被引用.
4.结构体不需要去继承另一个已存在类型的属性或者行为.

合适的结构体候选者包括:
1.几何形状的大小,封装一个 width 属性和 height 属性,两者均为 Double 类型.
2.一定范围内的路径,封装一个 start 属性和 length 属性,两者均为 Int 类型.
3.三维坐标系内一点,封装 x,y 和 z 属性,三者均为 Double 类型.

在所有其它案例中,定义一个类,生成一个它的实例,并通过引用来管理和传递,实际中,这意味着绝大部分的自定义数据构造都应该是类,而非结构体

8.集合类型的赋值和拷贝行为

Swift的字符串,数组和字典类型作为结构实现,这意味着当他们被分配到一个新的常量或变量,或将它们传递给一个函数或方法,也会复制字符串,数组和字典.

这种行为是不同于作为不是结构的类实现的 Nsstring,NSArray 和在基础上,NSDictionary. Nsstring,NSArray 和 NSDictionary 实例总是分配,流传到现有实例的引用,而不是一个副本.

好了,这次我们就讲到这里,下次我们继续~

Swift基础语法: 25 - Swift的类和结构体的更多相关文章

  1. ios – 声明NSDictionary并在Swift中添加键值对?

    我一直在尝试使用类类型键和值来声明一个NSDictionary,如下所示:这里,“Category”和“SubCategory”是全局类.我知道我不能将类类型用于关键字段.但是,无论如何,我应该做到这一点.有没有办法做到这一点?如何声明专门的NSDictionary或类似的东西来做到这一点?

  2. ios – Swift相当于`[NSDictionary initWithObjects:forKeys:]`

    Swift的原生字典是否与[NSDictionaryinitWithObjects:forKeys:]相当?假设我有两个带键和值的数组,并希望将它们放在字典中.在Objective-C中,我这样做:当然我可以通过两个数组迭代一个计数器,使用vardict:[String:Int]并逐步添加东西.但这似乎不是一个好的解决方案.使用zip和enumerate可能是同时迭代两者的更好方法.然而,这种方法

  3. ios – 无法识别的选择器发送到实例NSTimer Swift

    解决方法让updateTime成为一个类方法.如果它是在一个纯粹的Swift类中,你需要在@objc前面说明该方法的声明,如:

  4. ios – 在Swift中获取Cocoa Touch Framework项目版本字符串

    有谁知道这是否是我的项目设置中的缺陷,Xcode中的一个错误,或者是否有一种方法可以将Swift中的框架版本作为String或数组获取,这样我可以提供比major.minor更精细的版本控制?

  5. ios – 搜索数组swift中的对象

    我正在尝试使用UISearchController创建搜索功能.但是,我似乎无法使其与我的团队对象一起工作.我首先创建了一个包含id,name和shortname的TeamObject.然后我从一个url中检索teamData,并将TeamObjects添加到一个填充到tableView中的数组中.这个tableView包含一个searchController,它假设过滤数据,但没有任何反应.阵列

  6. ios – 类型推断(自动类型检测)如何在swift中工作?

    LLVM如何检测变量是一个字符串?

  7. ios – Swift可选项:语言问题,还是做错了什么?

    应该有可选的类型;type是但是,如果我这样做,它的工作原理:它似乎是基本的替代,但我可能会遗漏一些语言的细微差别.谁能对此有所了解?之后就像暧昧一样,更多,这是我的解决方案:这适用于所有非对象Swift对象,包括Swift字符串,数字等.感谢Viktor提醒我String不是Swift中的对象.如果您知道值的类型,您可以替换任何?使用适当的可选类型,如String?

  8. ios – 覆盖Swift中的超类委托

    我正在开发一个包含两个UIViews的Swift(v1.2)项目.MyView和MyViewSubclass.MyView有一个委托,我想在MyViewSubclass中覆盖它作为一个子协议,类似于UITableViews有一个UITableViewDelegate,它也符合超级uiscrollviewdelegate.我的第一个想法是覆盖超类属性,但这会导致编译器错误,因为子类不能覆盖具有不同类

  9. ios – Swift – 使用字典数组从字典访问数据时出错

    我有一个非常简单的例子,说明我想做什么基本上,我有一个字典,其值包含[String:String]字典数组.我把数据填入其中,但当我去访问数据时,我收到此错误:Cannotsubscriptavalueoftype‘[([String:String])]?’withanindexoftype‘Int’请让我知道我做错了什么.解决方法您的常量数组是可选的.订阅字典总是返回一个可选项.你必须打开它.更

  10. ios – 在Swift中使用“Map”创建两个数组的超集

    假设我有两个数组:我想组合两个数组,以便我得到一个输出我该怎么做呢?

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部