tips-2

Xcode release notes更新

定义了泛型取出非id,而是定义的泛型类型可以用点语法(id不可用点语法)

...表示可以传任意个参数
NSLog(@“”,…);
objc_msgSet(id,SEL,…)

------------------------------RUNTIME----------------------
// 使用运行时的第一步:导入
// 第二步:Build Setting -> 搜索msg -> 设置属性为No
// OC:运行时机制,消息机制是运行时机制最重要的机制
// 消息机制:任何方法调用,本质都是发送消息

// SEL:方法编号,根据方法编号就可以找到对应方法实现
// [p performSelector:@selector(eat)];

// 运行时,发送消息,谁做事情就那谁
// xcode5之后,苹果不建议使用底层方法
// xcode5之后,使用运行时.

// 让p发送消息
// objc_msgSend(p, @selector(eat));
// objc_msgSend(p, @selector(run:),10);

// 类名调用类方法,本质类名转换成类对象
// [Person eat];
// 获取类对象
Class personClass = [Person class];

// [personClass performSelector:@selector(eat)];

// 运行时
objc_msgSend(personClass, @selector(eat));

_OBJC_MESSAGE_H

OBJC_EXPORT id objc_msgSend(id self, SEL op, ...)
__OSX_AVAILABLE_STARTING(__MAC_10_0, __IPHONE_2_0);

遍历模型所有成员属性

// ivar:成员属性
// class_copyIvarList:把成员属性列表复制一份给你
// Ivar *:指向Ivar(成员属性类型)指针
// ivarList :指向一个成员变量数组
// class:获取哪个类的成员属性列表
// count:成员属性总数
unsigned int count = 0;
Ivar *ivarList = class_copyIvarList(self, &count);
for ()
{
Ivar one=ivarList[i]//取出每一个成员属性(包括公开和私有)
}
// 获取成员名
NSString *propertyName = [NSString stringWithUTF8String:ivar_getName(ivar)];

// 成员属性类型
NSString *propertyType = [NSString stringWithUTF8String:ivar_getTypeEncoding(ivar)];

/**

  • Returns the name of an instance variable.
  • @param v The instance variable you want to enquire about.
  • @return A C string containing the instance variable's name.
    */
    OBJC_EXPORT const char *ivar_getName(Ivar v)
    __OSX_AVAILABLE_STARTING(__MAC_10_5, __IPHONE_2_0);

/**

  • Returns the type string of an instance variable.
  • @param v The instance variable you want to enquire about.
  • @return A C string containing the instance variable's type encoding.
  • @note For possible values, see Objective-C Runtime Programming Guide > Type Encodings.
    */
    OBJC_EXPORT const char *ivar_getTypeEncoding(Ivar v)
    __OSX_AVAILABLE_STARTING(__MAC_10_5, __IPHONE_2_0);

// 解决KVC报错

//kvc没有找到和"key"对应的属性key时会来这个方法

  • (void)setValue:(id)value forUndefinedKey:(NSString *)key
    {
    if ([key isEqualToString:@"id"]) {
    _ID = [value integerValue];
    }
    // key:没有找到key
    // value:没有找到key对应的值
    NSLog(@"%@ %@",key,value);
    }

分类中.h用property

@interface NSObject (Objc)
/**

  • 分类中用property,只作用到.h,也就是只会声明getter和setter方法不会在.m中生成对应的成员属性和方法实现.
    */
    @property (nonatomic, strong) NSString *name;

@end

分类中.h用property,.m实现联系,类似增加属性

@implementation NSObject (Objc)

  • (void)setName:(NSString *)name
    {
    // 添加属性,跟对象
    // 给某个对象产生关联,添加属性
    // object:给哪个对象添加属性
    // key:属性名,根据key去获取关联的对象 ,void * == id
    // value:关联的值
    // policy:策略
    objc_setAssociatedObject(self, @"name", name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }

  • (NSString *)name
    {
    return objc_getAssociatedObject(self, @"name");
    }

@end

/**

  • Sets an associated value for a given object using a given key and association policy.
  • @param object The source object for the association.
  • @param key The key for the association.
  • @param value The value to associate with the key key for object. Pass nil to clear an existing association.
  • @param policy The policy for the association. For possible values, see “Associative Object Behaviors.”
  • @see objc_setAssociatedObject
  • @see objc_removeAssociatedObjects
    */
    OBJC_EXPORT void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
    __OSX_AVAILABLE_STARTING(__MAC_10_6, __IPHONE_3_1);

/**

  • Returns the value associated with a given object for a given key.
  • @param object The source object for the association.
  • @param key The key for the association.
  • @return The value associated with the key \e key for \e object.
  • @see objc_setAssociatedObject
    */
    OBJC_EXPORT id objc_getAssociatedObject(id object, const void *key)
    __OSX_AVAILABLE_STARTING(__MAC_10_6, __IPHONE_3_1);

当调用了没有实现的方法没有实现就会调用resolveInstanceMethod

@implementation Person

// 定义函数
// 没有返回值,参数(id,SEL)
// void(id,SEL)
void aaaa(id self, SEL _cmd, id param1)
{

NSLog(@"调用eat %@ %@ %@",self,NSStringFromSelector(_cmd),param1);

}

// 默认一个方法都有两个参数,self,_cmd,隐式参数
// self:方法调用者
// _cmd:调用方法的编号

// 动态添加方法,首先实现这个resolveInstanceMethod
// resolveInstanceMethod调用:当调用了没有实现的方法没有实现就会调用resolveInstanceMethod
// resolveInstanceMethod作用:就知道哪些方法没有实现,从而动态添加方法
// sel:没有实现方法

  • (BOOL)resolveInstanceMethod:(SEL)sel
    {
    // NSLog(@"%@",NSStringFromSelector(sel));

    // 动态添加eat方法

    if (sel == @selector(eat:)) {

      /*
       cls:给哪个类添加方法
       SEL:添加方法的方法编号是什么
       IMP:方法实现,函数入口,函数名
       types:方法类型
       */
      // @:对象 :SEL
      class_addMethod(self, sel, (IMP)aaaa, "v@:@");
      
      // 处理完
      return YES;
    

    }

return [super resolveInstanceMethod:sel];

}
@end

交换方法实现,方法都是定义在类里面

// 加载这个分类的时候调用

  • (void)load
    {

    // 交换方法实现,方法都是定义在类里面
    // class_getMethodImplementation:获取方法实现
    // class_getInstanceMethod:获取对象
    // class_getClassMethod:获取类方法
    // IMP:方法实现

    // imageNamed
    // Class:获取哪个类方法
    // SEL:获取方法编号,根据SEL就能去对应的类找方法
    Method imageNameMethod = class_getClassMethod([UIImage class], @selector(imageNamed:));

    // xmg_imageNamed
    Method xmg_imageNamedMethod = class_getClassMethod([UIImage class], @selector(xmg_imageNamed:));

    // 交换方法实现
    method_exchangeImplementations(imageNameMethod, xmg_imageNamedMethod);

}

------------------------------RUNTIME----------------------

A中static修饰全局变量,只能在A本文件中使用,没有修饰的全局变量,在B外可访问,访问时要用B中extern(外部的)声明,一下A中定义的全局变量,,如int a=2; extern只能声明不能定义,不能初始化.extern int a =2;//错的.extern只是声明a,说明a是其他文件的全局变量.extern翻译来自外部的.static修饰全局变量,缩小其使用范围,定义局部变量,扩展其生命周期

A: int a=5;//A中定义全局变量
B: extern int a;//B中声明a是来自外部的全局变量,然后就可以用了

super 一个修饰标志,不是一个指针,实际上方法的调用者还是self.有点类似runtime中的交换方法实现

这个是UIViewControl里面的属性

@property(nonatomic,assign) BOOL automaticallyAdjustsScrollViewInsets NS_AVAILABLE_IOS(7_0); // Defaults to YES

信号的绑定(其他的许多也是类似加工信号的原理,或拦截过滤)就是信号加工场
要 接收原始信号,加工信号,发送加工信号

define XMGLog(...) NSLog(VA_ARGS)

…表示参数个数不定
VA_ARGS表示要替代的参数.arguments

你可能感兴趣的:(tips-2)