继承
方法重写
在子类中实现与父类中同名的方法,称之为方法重写;
重写以后当给子类发送这个消息的时候,执行的是在子类中重写的那个方法,而不是父类中的方法。
如果想在子类中调用被子类重写的父类的方法,可以通过super关键字
使用场景:当从父类继承的某个方法不适合子类,可以在子类中重写父类的这个方法。
继承中方法调用顺序
1、在自己类中找
2、如果没有,去父类中找
3、如果父类中没有,就去父类的父类中
4、如果父类的父类也没有,就还往上找,直到找到基类(NSObject)
5、如果NSObject都没有就报错了
- 如果找到了就执行这个方法,就不再往后查找了
OC中的继承关系
B类继承A类,那么B类将拥有A类的所有属性和方法,此时我们说A类是B类的父类,B类是A类的子类
C类继承B类,那么C类将拥有B类中的所有属性和方法,包括B类从A类中继承过来的属性和方法,此时我们说B类是C类的父类,C类是B类的子类
注意:
基类的私有属性能被继承,不能在子类中访问。
OC中的继承是单继承:也就是说一个类只能一个父类,不能继承多个父类
子类与父类的关系也称为isA(是一个)关系,我们说 子类isA父类,也就是子类是一个父类,比如狗类继承动物类,那么我们说狗isA动物,也就是狗是一个动物。在如汽车继承交通工具,那么们说汽车isA交工工具,也就是汽车是一个交通工具
继承的合理性:引用《大话西游》里的一句话来描述继承的。“人是人他妈生的,妖是妖他妈生的!”
继承的注意事项
子类不能定义和父类同名的成员变量,私有成员变量也不可以;因为子类继承父类,子类将会拥有父类的所有成员变量,若在子类中定义父类同名成员变量 属于重复定义。
OC类支持单一继承,不支持多继承;也就是说一个类只能有一个直接父类
多态基本概念
什么是多态
多态就是某一类事物的多种形态
程序中的多态:父类指针指向子类对象
多态的条件
有继承关系
子类重写父类方法
父类指针指向子类对象
表现:当父类指针指向不同的对象的时候,通过父类指针调用被重写的方法的时候,会执行该指针所指向的那个对象的方法
多态的主要好处就是简化了编程接口.它允许在类和类之间重用一些习惯性的命名,而不用为每一个新加的函数命名一个新名字.这样,编程接口就是一些抽象的行为的集合,从而和实现接口的类的区分开来.
多态也使得代码可以分散在不同的对象中而不用 试图在一个函数中考虑到所有可能的对象这样使得您的代码扩展性和复用性更好一些.当一个新的情景出现时,您无须对现有的代码进行改动,而只需要增加一个新的类和新的同名方法.
私有变量和私有方法
私有变量
- 实例变量(成员变量)既可以在@interface中定义, 也可以在@implementation中定义
写在@implementation中的成员变量, 默认就是私有的成员变量, 并且和利用@private修饰的不太一样, 在@implementation中定义的成员变量在其它类中无法查看, 也无法访问
在@implementation中定义的私有变量只能在本类中访问,在其他类中无法查看,也无法访问
- 无论使用什么成员变量修饰符修饰成员变量,我们都可以在其他类中看到这个变量,只不过有的成员变量修饰符修饰的变量我们不能操作而已.
私有方法
如果只有方法的实现,没有方法的声明,那么该方法就是私有方法
在OC中没有真正的私有方法,因为OC是消息机制
@property编译器指令
什么是编译器指令? 编译器指令就是用来告诉编译器要做什么
@property会让编译器做什么呢?
@property 用在声明文件中告诉编译器声明成员变量的的访问器(getter/setter)方法
这样的好处是:免去我们手工书写getter和setter方法繁琐的代码
@property编写步骤
1.在@inteface和@end之间写上@property
2.在@property后面写上需要生成getter/setter方法声明的属性名称, 注意因为getter/setter方法名称中得属性不需要_, 所以@property后的属性也不需要_.并且@property和属性名称之间要用空格隔开
3.在@property和属性名字之间告诉需要生成的属性的数据类型, 注意两边都需要加上空格隔开
synthesize基本使用
什么是@synthesianze
@synthesize是编译器的指令
什么是编译器的指令 ?
- 编译器指令就是用来告诉编译器要做什么!
- @synthesize会让编译器做什么呢?
@synthesize 用在实现文件中告诉编译器实现成员变量的的访问器(getter/setter)方法
这样的好处是:免去我们手工书写getterr和setter方法繁琐的代码
@synthesize基本使用
- 在@implementation中, 用来自动生成setter和getter的实现
用@synthesize age = _age;就可以代替
- (int)age{
return _age;
}
- (void)setAge:(int)age{
_age = age;
}
- @synthesize编写步骤
1.在@implementation和@end之间写上@synthesize
2.在@synthesize后面写上和@property中一样的属性名称, 这样@synthesize就会将@property生成的什么拷贝到@implementation中
3.由于getter/setter方法实现是要将传入的形参 给属性和获取属性的值,所以在@synthesize的属性后面写上要将传入的值赋值给谁和要返回哪个属性的值, 并用等号连接
@synthesize注意点
- @synthesize age = _age;
setter和getter实现中会访问成员变量_age
如果成员变量_age不存在,就会自动生成一个@private的成员变量_age
- @synthesize age;
setter和getter实现中会访问@synthesize后同名成员变量age
如果成员变量age不存在,就会自动生成一个@private的成员变量age
- 多个属性可以通过一行@synthesize搞定,多个属性之间用逗号连接
@synthesize age = _age, number = _number, name = _name;
@property增强
- 自从Xcode 4.x后,@property可以同时生成setter和getter的声明和实现
@interface Person : NSObject
{
int _age;
}
@property int age;
@end
@property增强注意点
- 默认情况下,setter和getter方法中的实现,会去访问下划线 _ 开头的成员变量。
@interface Person : NSObject
{
@public
int _age;
int age;
}
@property int age;
@end
int main(int argc, const char * argv[]) {
Person *p = [Person new];
[p setAge:30];
NSLog(@"age = %i, _age = %i", p->age, p->_age);
return 0;
}
如果没有会自动生成一个_开头的成员变量,自动生成的成员变量是私有变量, 声明在.m中,在其它文件中无法查看,但当可以在本类中查看
@property只会生成最简单的getter/setter方法,而不会进行数据判断
Person *p = [Person new];
[p setAge:-10];
NSLog(@"age = %i", [p age]);
- 如果需要对数据进行判断需要我们之间重写getter/setter方法
若手动实现了setter方法,编译器就只会自动生成getter方法
若手动实现了getter方法,编译器就只会自动生成setter方法
若同时手动实现了setter和getter方法,编译器就不会自动生成不存在的成员变量
@property修饰符
- 【掌握】@property修饰符
@property修饰符
- 修饰是否生成getter方法的
readonly 只生成getter方法,不生成setter方法
readwrite 既生成getter 又生成setter方法(默认)
@property (readonly) int age;
- 指定所生成的方法的方法名称
getter=你定制的getter方法名称
setter=你定义的setter方法名称(注意setter方法必须要有 :)
@property (getter=isMarried) BOOL married;
说明,通常BOOL类型的属性的getter方法要以is开头
new方法实现原理
完整的创建一个可用的对象:Person *p=[Person new];
new方法的内部会分别调用两个方法来完成3件事情:
(1)使用alloc方法来分配存储空间(返回分配的对象);
(2)使用init方法来对对象进行初始化。
(3)返回对象的首地址
- 可以把new方法拆开如下:
(1)调用类方法+alloc分配存储空间,返回未经初始化的对象
Person *p1=[person alloc];
(2)调用对象方法-init进行初始化,返回对象本身
Person *p2=[p1 init];
- (3)以上两个过程整合为一句:
Person *p=[[Person alloc] init];
- 说明:
alloc 与 init合起来称为构造方法,表示构造一个对象
alloc 方法为对象分配存储空间,并将所分配这一块区域全部清0.
The isa instance variable of the new instance is initialized to a data structure that describes the class; memory for all other instance variables is set to 0.
- init方法是初始化方法(构造方法),用来对象成员变量进行初始化,默认实现是一个空方法。
An object isn’t ready to be used until it has been initialized. The init method defined in the NSObject class does no initialization; it simply returns self.
- 所以下面两句的作用是等价的
Person *p1 = [Person new];
Person *p = [[Person alloc] init];
- iOS 程序通常使用[[类名 alloc] init] 的方式创建对象,因为这个可以与其他initWithXX:...的初始化方法,统一来。代码更加统一