03

2017-07-31  本文已影响0人  小的小碰撞

1、成员变量

2、@property是一个编译器指令

3、 @synthesize

//@synthesize age = _age;

/*
 - (void)setAge:(int)age
 {
    _number = age;
 }
 - (int)age
 {
    return _number
 ;
 }
 */
//@synthesize age = _number;


// 如果在@synthesize后面没有告诉系统将传入的值赋值给谁, 系统默认会赋值给和@synthesize后面写得名称相同的成员变量
// _age? age;
@synthesize age;

/*
- (void)setAge:(int)age
{
    _age = age;
}

- (int)age
{
    return _age;
}
 */

4、id 类型

id == NSObject * 万能指针
id和NSObject *的区别:
NSObject *是一个静态数据类型
id 是一个动态数据类型

弊端: 由于动态数据类型可以调用任意方法, 所以有可能调用到不属于自己的方法, 而编译时又不会报错, 所以可能导致运行时的错误

为了避免动态数据类型引发的运行时的错误, 一般情况下如果使用动态数据类型定义一个变量, 在调用这个对象的方法之前会进行一次判断, 判断当前对象是否能够调用这个方法

5 new方法的实现和原理

1.开辟了存储空间+alloc 方法
2.初始化所有的属性(成员变量)- init方法
3.返回对象的地址

1 开辟了存储空间
2 将所有的属性设置为0
3 返回当前实例对象的地址
注意:1.初始化成员变量, 但是默认情况下init的实现是什么都没有做
2.返回 初始化后的实例对象地址
注意: alloc返回的地址, 和init返回的地址是同一个地址
建议大家以后创建一个对象都使用 alloc init, 这样可以统一编码格式

6 构造方法

在oc中init开头的方法,我们称之为构造方法
构造方法的用途:用于初始化一个对象,让某个对象一创建出来就拥有某些属性和值

重写init方法,在init方法中初始化成员变量
注意:重写init方法必须按照苹果规定的格式重写,如果不按照规定会引发一些未知的错误

1.必须先初始化父类, 再初始化子类
// 2.必须判断父类是否初始化成功, 只有父类初始化成功才能继续初始化子类
// 3.返回当前对象的地址
- (instancetype)init
{
    // 1.初始化父类
    // 只要父类初始化成功 , 就会返回对应的地址, 如果初始化失败, 就会返回nil
    // nil == 0 == 假 == 没有初始化成功
    self = [super init];
    // 2.判断父类是否初始化成功
    if (self != nil) {
        // 3.初始化子类
        // 设置属性的值
        _age = 6;
        
    }
    // 4.返回地址
    return self;
}

7 instancetype和id的区别

如果init方法的返回值是instancetype, 那么将返回值赋值给一个其它的对象会报一个警告
如果是在以前, init的返回值是id, 那么将init返回的对象地址赋值给其它对象是不会报错的

instancetype == id == 万能指针 == 指向一个对象
id在编译的时候不能判断对象的真实类型
instancetype在编译的时候可以判断对象的真实类型

id和instancetype除了一个在编译时不知道真实类型, 一个在编译时知道真实类型以外, 还有一个区别
id可以用来定义变量, 可以作为返回值, 可以作为形参
instancetype只能用于作为返回值

注意: 以后但凡自定义构造方法, 返回值尽量使用instancetype, 不要使用id

其实就是自定义一个init方法
1.一定是对象方法
2.一定返回id/instancetype
3.方法名称一定以init开头

8 自定义类工厂方法

自定义类工厂方法是苹果的一个规范, 一般情况下, 我们会给一个类提供自定义构造方法和自定义类工厂方法用于创建一个对象

什么是类工厂方法:

用于快速创建对象的类方法, 我们称之为类工厂方法
类工厂方法中主要用于 给对象分配存储空间和初始化这块存储空间

规范:

1.一定是类方法 +
2.方法名称以类的名称开头, 首字母小写
3.一定有返回值, 返回值是id/instancetype

eg:

+ (instancetype)person;

+ (instancetype)person
{
//    return [[Person alloc] init];
    // 注意: 以后但凡自定义类工厂方法, 在类工厂方法中创建对象一定不要使用类名来创建
    // 一定要使用self来创建
    // self在类方法中就代表类对象, 到底代表哪一个类对象呢?
    // 谁调用当前方法, self就代表谁
    return [[self alloc] init];
}

9 类的本质

类的本质:

 类其实也是一个对象, 这个对象会在这个类第一次被使用的时候创建
 只要有了类对象, 将来就可以通过类对象来创建实例对象
 实例对象中有一个isa指针, 指向创建自己的类对象
 
 类对象中保存了当前对象所有的对象方法
 当给一个实例对象发送消息的时候, 会根据实例对象中的isa指针去对应的类对象中查找

类的启动过程

只要程序启动就会将所有类的代码加载到内存中, 放到代码区
load方法会在当前类被加载到内存的时候调用, 有且仅会调用一次
如果存在继承关系, 会先调用父类的load方法, 再调用子类的load方法

当当前类第一次被使用的时候就会调用(创建类对象的时候)
initialize方法在整个程序的运行过程中只会被调用一次, 无论你使用多少次这个类都只会调用一次
initialize用于对某一个类进行一次性的初始化
initialize和load一样, 如果存在继承关系, 会先调用父类的initialize再调用子类的initialize

10 SEL类型

// 1.SEL类型的第一个作用, 配合对象/类来检查对象/类中有没有实现某一个方法
    /*
    SEL sel = @selector(setAge:);
    Person *p = [Person new];
    // 判断p对象中有没有实现-号开头的setAge:方法
    // 如果P对象实现了setAge:方法那么就会返回YES
    // 如果P对象没有实现setAge:方法那么就会返回NO
    BOOL flag = [p respondsToSelector:sel];
    NSLog(@"flag = %i", flag);
    
    // respondsToSelector注意点: 如果是通过一个对象来调用该方法那么会判断该对象有没有实现-号开头的方法
    // 如果是通过类来调用该方法, 那么会判断该类有没有实现+号开头的方法
    SEL sel1 = @selector(test);
    flag = [p respondsToSelector:sel1];
    NSLog(@"flag = %i", flag);
    
    flag = [Person respondsToSelector:sel1];
    NSLog(@"flag = %i", flag);
     */
    
    
    // 2.SEL类型的第二个作用, 配合对象/类来调用某一个SEL方法
    /*
    SEL sel = @selector(demo);
    Person *p = [Person new];
    // 调用p对象中sel类型对应的方法
    [p performSelector:sel];
    
    SEL sel1 = @selector(signalWithNumber:);
    // withObject: 需要传递的参数
    // 注意: 如果通过performSelector调用有参数的方法, 那么参数必须是对象类型,
    // 也就是说方法的形参必须接受的是一个对象, 因为withObject只能传递一个对象
    [p performSelector:sel1 withObject:@"13838383438"];
    
    SEL sel2 = @selector(setAge:);
    [p performSelector:sel2 withObject:@(5)];
    NSLog(@"age = %i", p.age);
    
    // 注意:performSelector最多只能传递2个参数
    SEL sel3 = @selector(sendMessageWithNumber:andContent:);
    [p performSelector:sel3 withObject:@"138383438" withObject:@"abcdefg"];
     */
上一篇 下一篇

猜你喜欢

热点阅读