iOS OC语言(二) 类

前言

类 是具有相同特征和行为的事物的抽象 
万事万物皆对象

  • 对象是类的实例
  • 类是对象的类型

面向对象的特点分为:

  • 封装
  • 继承
  • 多态 

如何看懂OC中的一个类

在OC中 
一个类中的成员, 分为实例变量和方法两种 
写在.h文件中的方法在类外是可以调用的. 
而实例变量虽然写在.h文件中, 但是它们能否在类外可以调用需要根据他们的修饰符定义的: 
实例变量的可见度

可见度 特点
@public 实例变量可以在类的外部和内部操作
@private 实例变量只能在该类内访问
@protected(默认的) 实例变量只能在该类和其子类内操作

@private 和@public 在实际开发中基本不使用

  • 对于private和protected修饰的实例变量, 不能在类外进行调用 而public可以在类外对实例变量进行赋值, 但是这样破坏了面向对象的封装性.

如果我们需要对实例变量进行取值, 赋值操作. 可以在类内定义方法. 通过调用方法来获得实例变量的值. 
注意: 类方法不能直接调用实例变量. 因为调用类方法的时候, 没有创建对象, 这时候没有在堆中开辟空间, 也没有进行赋值. 也就是说实例变量此时在内存中并不存在, 所以调用不了.

自定义初始化方法可以对实例变量进行赋值, 但是不能简单有效的对某个成员变量进行赋值.

要方便有效准确的对某一个实例变量进行取值, 赋值的操作, 通过以下方法:

  1. 先定义一个类:

    @interface Person : NSObject
    {
        @protected          // 声明实例变量都是protected修饰
        NSString *_name;    // 姓名
        NSString *_gender;  // 性别
        NSString *_age;     // 年龄
    }
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
  2. 定义一个对象方法, 取出指定的实例变量的值(在类中封装一个取值方法)

    - (NSString *)getName;   // 在.h文件中声明
    - (NSString *)getName {  // 在.m文件中实现
        return _name;
    }
    • 1
    • 2
    • 3
    • 4
  3. 定义一个对象方法, 更改指定的实例变量的值(在类中封装一个赋值方法)

    - (void)setName:(NSString *)name;  // 在.h文件中声明
    - (void)setName:(NSString *)name { // 在.m文件中实现
        _name = name;
    }   
    • 1
    • 2
    • 3
    • 4

getName, setName方法称为实例变量name的setter getter方法 
既然有专门的名称, 自然有专门的命名规范 
setter方法: set+实例变量的名字(忽略下划线 首字母大写) 参数名=实例变量名(忽略下划线) 
如: 
- (void)setName:(NSString *)name; 
getter方法:方法名就等于实例变量名(忽略下划线) 
如: 
- (NSString *)name;


关于方法名

例如自定义方法:

- (instancetype)initWithName:(NSString *)name age:(NSString *)age sex:(NSString *)sex;
  • 1

它的方法名是: 
initWithName:age:sex: 
值得注意的是, 冒号也是方法名的一部分, 不可缺少, 冒号是标识有参数. 
setter getter方法名: 
如以上定义的setter getter方法, 它们的名字分别是: 
setName: 和 name

同时修改两个成员变量的方法:

- (void)setName:(NSString *)name gender:(NSString *)gender; // 在.h文件中声明
- (void)setName:(NSString *)name gender:(NSString *)gender { // 在.m文件中实现
    _name = name;
    _gender = gender;
}
  • 1
  • 2
  • 3
  • 4
  • 5

p.s. OC是根据:来识别参数的个数的


关于创建对象

Person *p = [[Person alloc] init]; 
使用NSLog(@"%@", p); 打印p的信息 
通过占位符%@打印对象p的信息, 是调用了继承自父类的方法(继承随后的文章会介绍) 
- (NSString *)description; 
可以通过重写该方法, 使程序调用 
NSLog(@"%@", p);的时候打印出想要显示的结果

如想要根据自己的意愿打印Person类对象的所有信息, 可以重写description方法如下:

- (NSString *)description {
    NSString *str = [NSString stringWithFormat:@"姓名:%@,性别:%@,年龄:%@",_name, _gender, _age];
    return str;
}
  • 1
  • 2
  • 3
  • 4

附:stringWithFormat: 是格式化拼接字符串方法,是NSString类的类方法. 是OC中非常常用的方法


互相引用会导致错误

定义两个类 Man 和Woman 
如在Man类中 
#import "Woman.h" 
在Woman类中 
#import "Man.h"

这样互相引用会导致循环引用, 因而报错. 
原理是: 
在头文件A中import 头文件B, 而头文件B中又import了头文件A,这会导致A运行到import语句, 跳转到了头文件B, 头文件B运行到improt语句又跳回了头文件A, 而头文件A再次运行到import语句又跳到了B… 像这样导致编译器无限的在两个头文件中跳转来跳转去, 导致死循环, 直到崩溃.

解决方法是: 
在其中一个类中用@class 类名;的方法来解决 
如在Man类中 
#import "Woman.h" 
在Woman类中 
@class Man; 
此句的意思是声明Man是一个类 如果需要在.m文件中用到Man类中的方法, 还需要在#import “Man.h”(在.m文件中声明) 
此时在Man中可以定义一个Woman类的实例变量 
在Woman中也可以定义一个Man类的实例变量

在出现循环导入的时候, 注意初始化的时候会出现赋的值未初始化而无法进行赋值的情况.

引用语句中 
使用#include引头文件的时候不能重复导入 
#import可以重复导入 , 但是不能循环导入

重复导入是在同一个文件中 多次使用import “XXX.h” 语句调用同一个头文件. 
循环导入是在A文件中import B文件的头文件, 而B文件的头文件中又import了A文件的头文件, 导致编译器无限的在两个头文件中跳转来跳转去, 导致死循环, 直到崩溃.


例题如下: 
创建男人类: 
属性有: 姓名, 工作, 钱, 妻子 
方法有: 看篮球, 赚钱 
女人类: 姓名, 颜值, 丈夫, 孩子 
方法有: 购物 
宝宝类: 姓名, 性别

// Man.h
#import 
#import "Woman.h"
@interface Man : NSObject
{
    NSString *_name;    // 姓名
    NSString *_job;     // 工作
    NSString *_money;   // 钱
    // 复合: 在本类中 声明了一个其他类的对象作为本类的实例变量
    Woman *_wife;       // 妻子
}
- (instancetype)initWithName:(NSString *)name
                         job:(NSString *)job
                       money:(NSString *)money;

- (void)setName:(NSString *)name;
- (NSString *)name;

- (void)setJob:(NSString *)job;
- (NSString *)job;

- (void)setMoney:(NSString *)money;
- (NSString *)money;

- (void)setWife:(Woman *)wife;
- (Woman *)wife;


- (void)watchBasketBall;
- (void)makeMoney;


- (NSString *)description;
@end

// Man.m
#import "Man.h"

@implementation Man
// 可能在初始化Man的对象的时候, Woman对象尚未创建, 所以先不给Man里的实例变量Woman进行赋值
- (instancetype)initWithName:(NSString *)name
                         job:(NSString *)job
                       money:(NSString *)money {
    _name = name;
    _job = job;
    _money = money;

    return self;
}

- (void)setName:(NSString *)name {
    _name = name;
}
- (NSString *)name {
    return _name;
}

- (void)setJob:(NSString *)job {
    _job = job;
}
- (NSString *)job {
    return _job;
}

- (void)setMoney:(NSString *)money {
    _money = money;
}
- (NSString *)money {
    return _money;
}

- (void)setWife:(Woman *)wife {
    _wife = wife;
}
- (Woman *)wife {
    return _wife;
}


- (void)watchBasketBall {
    NSLog(@"看篮球");
}
- (void)makeMoney {
    NSLog(@"挣钱");
}

- (NSString *)description {                         // 输出妻子的姓名, 而不是妻子的全部信息. 请注意
    return [NSString stringWithFormat:@"姓名:%@,工作:%@,钱:%@,妻子:%@", _name, _job, _money, [_wife name]];
}
@end

// Woman.h
#import 

#import "Baby.h"
@class Man; // 用@class 关键字声明Man是一个类 由于Man.h已经importWoman.h”, 所以不能再在Woman.himportMan.h"
@interface Woman : NSObject
{
    NSString *_name;            // 姓名
    NSString *_beautifulValue;  // 颜值
    Man *_husband;              // 丈夫 在这里将Man仅仅当成一个类名来使用, 在这里虽然husbend声明的是Man的对象 但它没有保存Man类中的实例变量和方法, 若要调用到Man类里的方法, 只能通过import ”Man.h”的形式
    Baby *_baby;                // 孩子
}
        // 由于Woman对象初始化的时候 Man对象和Baby对象可能还未创建, 无法给Woman对象中的实例变量_husband, _baby赋值, 所以先不在初始化中对这两个变量进行初始化.
- (instancetype)initWithName:(NSString *)name  
              beautifulValue:(NSString *)beautifulValue;

- (void)setName:(NSString *)name;
- (NSString *)name;

- (void)setBeautifulValue:(NSString *)beautifulValue;
- (NSString *)beautifulValue;

- (void)setHusBand:(Man *)husband;
- (Man *)husband;

- (void)setBaby:(Baby *)baby;
- (Baby *)baby;

- (void)bayBayBay;


- (NSString *)description;
@end

// Woman.m
#import "Woman.h"
#import “Man.h” // 在这里import “Man.h” 可以通过.h中声明的Man变量 调用Man类里的方法.
@implementation Woman
- (instancetype)initWithName:(NSString *)name
              beautifulValue:(NSString *)beautifulValue {
    _name = name;
    _beautifulValue = beautifulValue;

    return self;
}

- (void)setName:(NSString *)name {
    _name = name;
}
- (NSString *)name {
    return _name;
}

- (void)setBeautifulValue:(NSString *)beautifulValue {
    _beautifulValue = beautifulValue;
}
- (NSString *)beautifulValue {
    return _beautifulValue;
}

- (void)setHusBand:(Man *)husband {
    _husband = husband;
}
- (Man *)husband {
    return _husband;
}

- (void)setBaby:(Baby *)baby {
    _baby = baby;
}
- (Baby *)baby {
    return _baby;
}

- (void)bayBayBay {
    NSLog(@"买东西");
}


- (NSString *)description {
    return [NSString stringWithFormat:@"姓名:%@,颜值:%@,丈夫:%@,孩子:%@",_name, _beautifulValue, [_husband name], [_baby name]]; // 在这里显示的是男人的名字和孩子的名字 为的是避免出现循环调用 导致死循环, 下文会进行详细解释
}

@end

// Baby.h
#import 

@interface Baby : NSObject
{
    NSString *_name; // 姓名
    NSString *_sex;  // 性别
}
- (instancetype)initWihtName:(NSString *)name
                         sex:(NSString *)sex;

- (void)setName:(NSString *)name;
- (NSString *)name;

- (void)setSex:(NSString *)sex;
- (NSString *)sex;


- (NSString *)description;
@end

// Baby.m
#import "Baby.h"

@implementation Baby
- (instancetype)initWihtName:(NSString *)name
                         sex:(NSString *)sex {
    _name = name;
    _sex = sex;
    return self;
}
- (void)setName:(NSString *)name {
    _name = name;
}
- (NSString *)name {
    return _name;
}

- (void)setSex:(NSString *)sex {
    _sex = sex;
}
- (NSString *)sex {
    return _sex;
}

- (NSString *)description {
    return [NSString stringWithFormat:@"姓名:%@,性别:%@",_name,_sex];
}
@end


// main.m
#import 
#import "Man.h"
#import "Woman.h"
#import "Baby.h"
int main(int argc, const char * argv[]) {

    Man *man = [[Man alloc] initWithName:@"l" job:@"ios" money:@"12k/m"];
    Woman *women = [[Woman alloc] initWithName:@"y" beautifulValue:@"10"];
    Baby *baby = [[Baby alloc] initWihtName:@"ly" sex:@"女"];
    [women setBaby:baby];
    [women setHusBend:man];
    [man setWife:women];

    NSLog(@"%@", man);

    return 0;
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134
  • 135
  • 136
  • 137
  • 138
  • 139
  • 140
  • 141
  • 142
  • 143
  • 144
  • 145
  • 146
  • 147
  • 148
  • 149
  • 150
  • 151
  • 152
  • 153
  • 154
  • 155
  • 156
  • 157
  • 158
  • 159
  • 160
  • 161
  • 162
  • 163
  • 164
  • 165
  • 166
  • 167
  • 168
  • 169
  • 170
  • 171
  • 172
  • 173
  • 174
  • 175
  • 176
  • 177
  • 178
  • 179
  • 180
  • 181
  • 182
  • 183
  • 184
  • 185
  • 186
  • 187
  • 188
  • 189
  • 190
  • 191
  • 192
  • 193
  • 194
  • 195
  • 196
  • 197
  • 198
  • 199
  • 200
  • 201
  • 202
  • 203
  • 204
  • 205
  • 206
  • 207
  • 208
  • 209
  • 210
  • 211
  • 212
  • 213
  • 214
  • 215
  • 216
  • 217
  • 218
  • 219
  • 220
  • 221
  • 222
  • 223
  • 224
  • 225
  • 226
  • 227
  • 228
  • 229
  • 230
  • 231
  • 232
  • 233
  • 234
  • 235
  • 236
  • 237
  • 238
  • 239
  • 240
  • 241
  • 242
  • 243
  • 244
  • 245

要点1:

  • 在Woman.h中用@class Man; 而不是import "Man.h".是为了避免循环引用.

要点2:

  • Woman类中的description调用了Man对象, NSLog(@"%@", man);输出的是man对象, 而输出一个对象的内容是这个对象的description方法, 而若此时man的description又出现NSLog(@"%@", woman);又调用了woman的description方法, 这样就发现两者无限循环调用, 直到内存满了, 程序崩溃. 所以两个对象不能同时输出对方的对象. 只有一方输出另一方的对象是可以接受的, 两者同时输出另一方, 是会崩溃的. 因而程序中, 输出的不是对象本身, 而是对象的实例变量, 这样就可以绕过循环调用.

你可能感兴趣的:(OC语言)