09-Swift方法(Methods)

方法是与某些特定类型相关联的函数。类、结构体、枚举都可以定义实例方法;实例方法为特定类型的实例封装具体的任务与功能。类、结构体、枚举也可以定义类型方法;类型方法与类型本身相关联。类型方法与OC中的类方法类似。
 结构体和枚举能够定义方法是swift与C/OC主要区别之一。OC中,类是唯一能定义方法的类型。


一、实例方法


实例方法是属于某个特定类、结构体、枚举类型实例对象的方法。实例方法提供访问和修改实例属性的方法,另外实例方法的语法与函数完全一致(参考函数)。
 实例方法要写在它所属的类型的大括号内。实例方法能够隐式访问它所属类型的所有的实例方法以及属性。实例方法只能够被它所属类的实例对象调用。

  • 实例方法基本使用
// 定义一个学生类
class Student {
    // 学习次数
    var count = 1;
    // 实例方法1 - 说
    func speak() {
        print("hello");
    }
    // 实例方法2 - 学习
    func study() {
        // 每学习一次就加一
        count += 1;
    }
    // 实例方法3 - 重置
    func reset() {
        count = 0;
    }
}
// 调用实例方法,与调用属性一样,都是通过点语法
let liming = Student();
// 调用实例方法1
liming.speak();
// 调用实例方法2
liming.study();
// 调用实例方法3
liming.reset();
输出结果:
hello
count:2
count:0
  • 方法的局部参数名称和外部参数名称。swift的方法和OC中的方法类似,通常用一个介词指向方法的第一个参数,例如withforby等。swift中默认仅给方法的第一个参数是局部参数名称,默认同时给第二个和后续参数名称局部参数名称和外部参数名称:
class Student {
    // 实例方法 - 学习(具体学科,谁学习)
    func studyWith(subjectStr:String, studentName:String) {
        print("\(studentName)正在学习\(subjectStr)...");
    }
}
// 调用实例方法,与调用属性一样,都是通过点语法
let student = Student();
// 方法调用student.studyWith(subjectStr: String, studentName: String)
student.studyWith("英语", studentName: "张三");
输出结果:
张三正在学习英语...

注意: 不必为第一个参数值再定义一个外部变量名: 从方法名studyWith(_, studentName:)可看出其作用。但第二个参数,就要被一个外部参数名称所限定,以便在方法调用时明确作用。

  • self属性。类型的每个实例都隐含着一个selfself即表示是实例本身:
class Student {
    var name:String?;
    func studyWith(name:String) {
        // self表示实例本身,即访问实例中的name属性
        self.name = name;
        print("\(name)正在学习...");
    }
}
let student = Student();
student.studyWith("张三");
输出结果:
张三正在学习...
  • self属性消除方法参数和实例属性之间的歧义:
struct Point {
    var x=0.0, y=0.0;
    // 判断x是否在点(x,y)的右边
    func isToRightOfX(x:Double) -> Bool {
        // 传入参数与属性x相比较
        // self.x表示是实例属性,x表示是方法参数
        if self.x < x {
            // 表在右边
            return true;
        } else {
            // 表在左边
            return false;
        }
    }
}
// 实例化结构体
// 所有结构体都有一个自动生成的成员逐一构造器,用于初始化结构体实例中的成员属性
var somePoint = Point(x:1.0, y:1.0);
if (somePoint.isToRightOfX(5.0)) {
    print("右边");
} else {
    print("左边");
}
输出结果:
右边
  • 在实例方法中修改值类型。结构体和枚举是值类型,一般情况下,值类型的属性不能再它的实例方法中被修改。如果想要修改实例中的结构体或枚举的属性,可以选择变异这个方法,即方法就可以从方法内部改变它的属性,请求修改完结束后,还会保留在原始结构体中:
struct Point {
    var x=0.0, y=0.0;
    // 判断x是否在点(x,y)的右边
    mutating func moveToPoint(newX newX:Double, newY:Double) {
        // 修改结构体中的属性
        // 如果没有'mutating'关键字,则会报错
        x = newX;
        y = newY;
    }
}
// 注意不能再定义为常量,因为常量的属性不能被修改
var somePoint = Point();
print("x:\(somePoint.x)  y:\(somePoint.y)");
somePoint.moveToPoint(newX: 30, newY: 30);
print("x:\(somePoint.x)  y:\(somePoint.y)");
输出结果:
x:0.0  y:0.0
x:30.0  y:30.0
  • 结构体,在可变方法中给self赋值。可变方法是能够给隐含属性self赋值的:
struct Point {
    var x=0.0, y=0.0;
    // 判断x是否在点(x,y)的右边
    mutating func moveToPoint(newX newX:Double, newY:Double) {
        // 写法二,直接给本实例赋值
        self = Point(x:newX, y:newY);
    }
}
var somePoint = Point();
print("x:\(somePoint.x)  y:\(somePoint.y)");
somePoint.moveToPoint(newX: 30, newY: 30);
print("x:\(somePoint.x)  y:\(somePoint.y)");
输出结果:
x:0.0  y:0.0
x:30.0  y:30.0
  • 枚举的可变方法可以把self设置为相同的枚举类型中不同成员:
// 定义一个三态开关的枚举
enum StatuSwitch {
    // 关闭的、暗的、高亮的
    case Off, Low, High;
    
    // 实例方法
    mutating func next() {
        switch self {
        case Off:
            self = Low;
        case Low:
            self = High;
        case High:
            self = Off;
        }
    }
}
// 默认是关闭
var someStatu = StatuSwitch.Off;    // Off
someStatu.next();   // Low
someStatu.next();   // High


二、类型方法


通过类型本身调用的方法称为类型方法。声明结构体和枚举的类型方法,在方法的func关键字前面加上static。类可以用关键字class来允许子类重写父类方法。类型方法和实例方法一样用点语法调用。

  • 类型方法基本使用:
class someClass {
    // 类型方法
    static func speak() {
        print("hello Swift");
    }
}
// 调用someClass类型方法
someClass.speak();
输出结果:
hello Swift

在OC中,只能为类定义类型方法。在swift中,可以为类、结构体、枚举定义类型方法。

  • 类型方法,只能调用本类中另外的类型方法,直接就是类型方法名即可调用,不需要加上类型的名称。另外,可以直接通过静态属性名来访问静态属性,而不需要类型名称(结构体和枚举也同样):
// 该结构体用于检测玩家的等级,以及等级限制的解锁操作
struct LevelTracker {
    // 已经解锁的最高等级,静态属性
    // 默认所有玩家解锁的最高等级都是1
    static var highestUnlockLevel = 1;
    
    // 类型方法,改变最高等级
    static func unlockLevel(level:Int) {
        if level > highestUnlockLevel {
            highestUnlockLevel = level;
        }
    }
    
    // 类型方法,判断最高等级是否解锁
    static func unlockLevelStauts(level:Int) -> Bool {
        return highestUnlockLevel >= level;
    }
    
    // 玩家当前等级,默认是1级
    var currentLevel = 1;
    // 实例方法 - 更新当前等级
    mutating func advanceLevel(level:Int) -> Bool {
        // 检查新等级是否解锁成功
        if LevelTracker.unlockLevelStauts(level) {    // 类型方法的调用
            currentLevel = level;
            return true;    // 表当前等级更新成功
        } else {
            return false;
        }
    }
}
// 玩家类
class Player {
    // 检测和更新玩家等级
    var playerLevelTracker = LevelTracker();
    // 玩家名
    var playerName:String = "";
    // 实例方法 - 玩家完成任务,解锁下一个等级
    func completedTask(level:Int) {
        // 解锁最高等级
        LevelTracker.unlockLevel(level);
        // 更新当前等级
        playerLevelTracker.advanceLevel(level);
    }
    // 构造方法
    init(name:String) {
        playerName = name;
    }
}
// 实例化玩家1
var player1 = Player(name: "EndEvent");
// 玩家完成‘等级1’的任务
player1.completedTask(1);
// 玩家解锁的最高等级
print("玩家解锁的最高等级: \(LevelTracker.highestUnlockLevel)");
// 玩家2
var player2 = Player(name: "zhangsan");
// 尝试直接更新到"等级6"
if player2.playerLevelTracker.advanceLevel(6) {
    print("更新成功");
} else {
    print("更新失败");
}
输出结果:
玩家解锁的最高等级: 1
更新失败


注:xcode7.3环境

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容