爬爬爬之路:OC语言(二) 类
来源:互联网 发布:悔创阿里杰克马 知乎 编辑:程序博客网 时间:2024/06/04 23:56
前言
类 是具有相同特征和行为的事物的抽象
万事万物皆对象
- 对象是类的实例
- 类是对象的类型
面向对象的特点分为:
- 封装
- 继承
- 多态
如何看懂OC中的一个类
在OC中
一个类中的成员, 分为实例变量和方法两种
写在.h文件中的方法在类外是可以调用的.
而实例变量虽然写在.h文件中, 但是它们能否在类外可以调用需要根据他们的修饰符定义的:
实例变量的可见度
@private 和@public 在实际开发中基本不使用
- 对于private和protected修饰的实例变量, 不能在类外进行调用 而public可以在类外对实例变量进行赋值, 但是这样破坏了面向对象的封装性.
如果我们需要对实例变量进行取值, 赋值操作. 可以在类内定义方法. 通过调用方法来获得实例变量的值.
注意: 类方法不能直接调用实例变量. 因为调用类方法的时候, 没有创建对象, 这时候没有在堆中开辟空间, 也没有进行赋值. 也就是说实例变量此时在内存中并不存在, 所以调用不了.
自定义初始化方法可以对实例变量进行赋值, 但是不能简单有效的对某个成员变量进行赋值.
要方便有效准确的对某一个实例变量进行取值, 赋值的操作, 通过以下方法:
先定义一个类:
@interface Person : NSObject{ @protected // 声明实例变量都是protected修饰 NSString *_name; // 姓名 NSString *_gender; // 性别 NSString *_age; // 年龄}
定义一个对象方法, 取出指定的实例变量的值(在类中封装一个取值方法)
- (NSString *)getName; // 在.h文件中声明- (NSString *)getName { // 在.m文件中实现 return _name;}
定义一个对象方法, 更改指定的实例变量的值(在类中封装一个赋值方法)
- (void)setName:(NSString *)name; // 在.h文件中声明- (void)setName:(NSString *)name { // 在.m文件中实现 _name = name;}
getName, setName方法称为实例变量name的setter getter方法
既然有专门的名称, 自然有专门的命名规范
setter方法: set+实例变量的名字(忽略下划线 首字母大写) 参数名=实例变量名(忽略下划线)
如: - (void)setName:(NSString *)name;
getter方法:方法名就等于实例变量名(忽略下划线)
如: - (NSString *)name;
关于方法名
例如自定义方法:
- (instancetype)initWithName:(NSString *)name age:(NSString *)age sex:(NSString *)sex;
它的方法名是: 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;}
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;}
附: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 <Foundation/Foundation.h>#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 <Foundation/Foundation.h>#import "Baby.h"@class Man; // 用@class 关键字声明Man是一个类 由于Man.h已经import “Woman.h”, 所以不能再在Woman.h中import ”Man.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 <Foundation/Foundation.h>@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 <Foundation/Foundation.h>#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:
- 在Woman.h中用
@class Man;
而不是import "Man.h"
.是为了避免循环引用.
要点2:
- Woman类中的description调用了Man对象,
NSLog(@"%@", man);
输出的是man对象, 而输出一个对象的内容是这个对象的description方法, 而若此时man的description又出现NSLog(@"%@", woman);
又调用了woman的description方法, 这样就发现两者无限循环调用, 直到内存满了, 程序崩溃. 所以两个对象不能同时输出对方的对象. 只有一方输出另一方的对象是可以接受的, 两者同时输出另一方, 是会崩溃的. 因而程序中, 输出的不是对象本身, 而是对象的实例变量, 这样就可以绕过循环调用.
- 爬爬爬之路:OC语言(二) 类
- iOS OC语言(二) 类
- 爬爬爬之路:OC语言(三) 继承
- OC语言(二)
- OC语言注意事项二
- OC语言(二)
- 爬爬爬之路:OC语言(一) 语法简介
- 爬爬爬之路:OC语言(六) Block语法简介
- 爬爬爬之路:OC语言(七) NSDate , 协议和代理
- 爬爬爬之路:OC语言(八) 属性 KVC简单介绍
- 爬爬爬之路:OC语言(九) 内存管理(初级)
- 我的OC升级之路(二)
- OC语言之block
- OC语言之协议
- OC语言之block
- OC语言之OC对象和方法
- OC语言之OC中的私有方法
- OC语言学习 (二) 创建OC对象类、成员/静态变量、对象/类方法
- SQL Server设置登录验证模式
- request属性 request.getAttribute()
- Android:控件AutoCompleteTextView
- jsp request 对象详解
- android实现开机自动播放音乐实例(Broadcast、Service)
- 爬爬爬之路:OC语言(二) 类
- android自定义圆形图像
- intValue、parseInt、valueOf 方法区别及其使
- InvertCopy
- JSP中Session的使用
- 电路分析中的参考方向
- Xcode7之 Scene Dock and Extra Views
- 虚函数在对象中的内存布局
- JSP中Session用法及其属性介绍