Objective-C (temp)

2017-02-23  本文已影响0人  WesleyLien

基本概念

基本数据类型

iOS 应避免使用基本类型,建议使用 Foundation 数据类型,这是为了基于64-bit 适配考虑,对应关系如下:

基本类型 Foundation 类型
int NSInteger
float CGFloat
unsigned NSUInteger
动画时间 NSTimeInterval

一般文件形式

// 引用
// #import 与 C 的 #include 区别是 #import 可确保不会重复导入同一个文件
#import <UIKit/UIKit.h>
// 一般用于引用系统框架
//@import UserNotifications;
// @class 只是对类进行声明
//@class TestItem;

// Objective-C 中定义一个类并不是使用 @class , 而是使用 @interface 。 interface 直译过来就是接口的意思, Objective-C 使用这个单词定义类,意思是说”外部能用的接口都在这里了“
// : NSObject -- 表示继承
@interface TestObject : NSObject
//{
//    //放置实例变量,但一般不会写在.h文件中,因为一般默认实例变量是私有的
//    //实例变量,一般约定实例变量采用 _ 开头命名
//    NSString *_str1;
//}
// 存取方法
//- (void)setStr1:(NSString *)str;
//- (NSString *)str1;

// 属性 = 实例变量 + getter + setter
// 属性的特征:多线程特征、读写特征、内存管理特征
@property (nonatomic, readwrite, strong) NSString *str2;
@property (nonatomic, readwrite, assign) CGFloat floatValue;

// 初始化方法 ( initialization method ),一般约定实例初始化方法格式为 initXXX ,类初始化方法格式为 类名 + XXX
// 返回值为什么是 instancetype ?涉及到继承问题
// 实例方法 ( instance method ),以 - 开头
- (instancetype)initWithParam:(NSString *)param1 param2:(NSInteger)param2;
// 类方法 ( class method ),以 + 开头
+ (instancetype)testObjectWithoutParam;

// 返回值类型 函数名(包含:) 参数类型 参数名
- (void)method1;
- (void)method2:(NSString *)str;
+ (NSString *)sayHello;

@end
#import "TestObject.h"

// 类扩展( class extension )
@interface TestObject()
{
    // 放置实例变量

}
// 放置属性

@end

// 实现程序段 ( implementation block )从 @implementation 指令开始, @end 指令结束
@implementation TestObject
{
    // 放置实例变量

}
// 自定义属性的合成。对应也有 @dynamic ,@dynamic 告诉编译器:属性的setter与getter方法由用户自己实现,不自动生成
@synthesize str2 = _str2;
// 如果自定义了属性的存取方法,编译器不会自动创建相应的实例变量,则需要进行属性合成
- (void)setStr2:(NSString *)str2 {
    _str2 = str2;
}
- (NSString *)str2 {
    return _str2;
}
//在初始化方法中尽量避免使用存取方法访问实例变量,而是直接访问实例变量
- (instancetype)initWithParam:(NSString *)param1 param2:(NSInteger)param2 {
    self = [super init];
    if (self) {

    }
    return self;
}
+ (instancetype)testObjectWithoutParam {
    TestObject *testObject = [[TestObject alloc] initWithParam:@"defaultStr" param2:0];
    return testObject;
}

@end

NOTE1:

NOTE2: 类方法和实例方法的区别和联系

OC对象的内存布局

属性

property 在 Runtime 中是 objc_property_t ,定义如下:

typedef struct objc_property *objc_property_t;

而 objc_property 是一个结构体,包括 name 和 attributes ,定义如下:

struct property_t {
    const char *name;
    const char *attributes;
};

而 attributes 本质是 objc_property_attribute_t ,定义了 property 的一些属性,定义如下:

/// Defines a property attribute
typedef struct {
    const char *name;           /**< The name of the attribute */
    const char *value;          /**< The value of the attribute (usually empty) */
} objc_property_attribute_t;

而attributes的具体内容包括:类型,原子性,内存语义和对应的实例变量。
例如:我们定义一个 string 的 property @property (nonatomic, copy) NSString *string; ,通过 property_getAttributes(property) 获取到 attributes 并打印出来之后的结果为T@"NSString",C,N,V_string
其中T就代表类型,可参阅Type Encodings,C就代表Copy,N代表nonatomic,V就代表对于的实例变量。

属性的特征

默认修饰符

weak的使用场景

copy的使用场景

weak 与 assign 区别

//输出为:
//A dealloc
//b.a: 0x0
//B dealloc
// 因为 A 是强引用 B,所以先 dealloc A,然后 B 中属性指向的 A 指针指向 nil,最后 dealloc B

@interface A : NSObject
@property B *b;
@end
@implementation A
- (void)dealloc {
    NSLog(@"A dealloc");
}
@end

@interface B : NSObject
@property (weak)A *a;
@end
@implementation B
- (void)dealloc {
    NSLog(@"B dealloc");
}
@end

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        B *b = [[B alloc] init];
        {
            A *a = [[A alloc] init];
            a.b = b;
            b.a = a;
        }
        NSLog(@"b.a: %p", b.a);
    }
    return 0;
}
//输出为:
//A dealloc
//b.a: 0x1002025a0  //这里是有值的
//B dealloc

@interface A : NSObject
@property B *b;
@end
@implementation A
- (void)dealloc {
    NSLog(@"A dealloc");
}
@end

@interface B : NSObject
@property (assign)A *a;
@end
@implementation B
- (void)dealloc {
    NSLog(@"B dealloc");
}
@end

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        B *b = [[B alloc] init];
        {
            A *a = [[A alloc] init];
            a.b = b;
            b.a = a;
        }
        NSLog(@"b.a: %p", b.a);
    }
    return 0;
}

Runtime 如何实现 weak 属性

Runtime 对注册的类,会进行布局,对于 weak 对象会放入一个 hash 表中。 用 weak 指向的对象内存地址作为 key,wesk 修饰的属性变量的内存地址为 value 。当此对象的引用计数为 0 的时候会 dealloc,假如 weak 指向的对象内存地址是 a,那么就会以 a 为 key , 在这个 weak 表中搜索,找到所有以 a 为 key 的 weak 对象,从而设置为 nil。

属性所指向的对象使用 copy 修饰符的条件

要使属性所指向的对象能使用 copy 修饰符,对象必需实现 <NSCopying> 协议的 - copyWithZone: 方法

- (id)copyWithZone:(NSZone *)zone {
    CYLUser *copy = [[[self class] allocWithZone:zone]
                     initWithName:_name
                     age:_age
                     sex:_sex];
    copy->_friends = [_friends mutableCopy];
    return copy;
}
//在上述例子中,存放朋友对象的 set 是用 “copyWithZone:” 方法来拷贝的,这种浅拷贝方式不会逐个复制 set 中的元素。若需要深拷贝的话,则可像下面这样,编写一个专供深拷贝所用的方法:
- (id)deepCopy {
    CYLUser *copy = [[[self class] allocWithZone:zone]
                     initWithName:_name
                     age:_age
                     sex:_sex];
    copy->_friends = [[NSMutableSet alloc] initWithSet:_friends
                                             copyItems:YES];
    return copy;
}

NSObject 与 <NSObject>

NSObject 是所有类的根类,<NSObject> 是 NSObject 实现的协议( protocol ),集合了对所有OC对象基本的方法

实例的创建、初始化和销毁

// 返回接收类的新的实例,但这个实例并不能使用,因为所有关于这个实例的实例变量的内存地址都指向0
+ alloc
// 一般的初始化方法
- init
+ new
// 当实例被回收前系统发送该通知
- dealloc
// 返回一个对象,使用此方法必须实现 <NSCopying> 协议的- copyWithZone:方法
- copy
// 返回一个对象,使用此方法必须实现 <NSMutableCopying> 协议的- mutableCopyWithZone:方法
- mutableCopy

浅复制与深复制

浅复制:指针复制
深复制:内容复制。但对于集合类来说,内容复制是复制集合对象本身,对于集合对象内的元素仍是指针复制,即单层深复制

immutable 对象 mutable 对象
copy 浅复制 深复制
mutableCopy 深复制 深复制
immutable 对象 mutable 对象
copy 浅复制 单层深复制
mutableCopy 单层深复制 单层深复制

判断对象类、继承、行为和一致性

// 返回类对象
- class

// 判断实例是否某一类对象或继承与该类的子类的对象
- isKindOfClass:
// 判断实例是否某一类对象
- isMemberOfClass:
// 判断实例是否能够响应选择器 ( selector )
- respondsToSelector:
- instanceRespondsToSelector:
// 判断实例是否实现协议内容
- conformsToProtocol:

判断两个实例是否相等

// 1.先判断hash值:hash == NO,对象不等;hash == YES ==> isEqual:
// 2.isEqual: == NO,对象不等,hash == YES,对象相等

.hash
- isEqual:

发送消息

- performSelector:
- performSelector:withObject:
- performSelector:withObject:withObject:
- performSelector:withObject:afterDelay:

对象的描述

//可重写属性 getter 方法返回对象的描述
.description

MRC 相关

- retain
- release
- autorelease
- retainCount

KVC 相关

- setValue:forKey:
- setValue:forUndefinedKey:
- setValue:forKeyPath:

- valueForKey:
- valueForUndefinedKey:
- valueForKeyPath:

KVO相关

- addObserver:forKeyPath:options:context:
- removeObserver:forKeyPath:
// oberver 的方法回调
- observeValueForKeyPath:ofObject:change:context:

其他

- dictionaryWithValuesForKeys:
- awakeFromNib

内存管理

OC 通过引用计数来管理内存,决定对象是否需要释放。检查对象的引用计数( retainCount ),如果为 0 ,则释放掉

对象的内存销毁步骤

  1. 对象的引用计数变为零
  1. 父类对象调用 - dealloc
  1. 根类对象 NSObject 调用 - dealloc
  1. 调用 object_dispose()

MRC

在OC 1.0采用的是手动引用计数 (MRC) 来管理内存

- release
- retain
- retainCount

@interface Test : NSObject
@end
@implementation Test
- (void)dealloc {
    NSLog(@"Test: dealloc");
}
@end

Test *a = [[Test alloc] init];
Test *b = nil;  
b = a;
[a release];
//输出 Test: dealloc
//因为 b 并未拥有 Test 类型的对象,所以并不影响这个对象的回收,此时如果执行 [b release] 会报错提示 overreleased
Test *a = [[Test alloc] init];
Test *b = nil;  
b = a;
[a retain];
[a release];
//输出 Test: dealloc
[a release]; // 或 [b release];

//谁持有谁释放
Test *a = [[Test alloc] init];
Test *b = nil;  
b = a;
[b retain];
[a release];
//输出 Test: dealloc
[b release];
@interface Test : NSObject
@end
@implementation Test
- (void)dealloc {
    NSLog(@"Test: dealloc");
}
//重写 retain 和 release 方法
- (instancetype)retain {
    NSLog(@"Test before retain: %@", @(self.retainCount));
    id result = [super retain];
    NSLog(@"Test after retain: %@", @(self.retainCount));
    return result;
}
- (oneway void)release
{
    NSLog(@"Test before release: %@", @(self.retainCount));
    [super release];
    NSLog(@"Test after release: %@", @(self.retainCount));
}
@end

Test *a = [[Test alloc] init];
//Test before release: 1
//Test: dealloc
//Test after release: 1
[a release];
//Test after release: 1 是因为都已经释放了,也没必要将引用计数置为 0
@interface Test : NSObject
+ (Test *)getInstance;
@end
@implementation Test
- (void)dealloc {
    NSLog(@"Test: dealloc");
}
+ (Test *)getInstance {
    Test *result = [[Test alloc] init];
    return result;
}
@end

Test *a = [Test getInstance];
[a release];
// 这里我们明确知道要调用 [a release],但如果 [Test getInstance] 是个黑箱,又如何知道是否该 release  
// 函数创建对象,但无法release它;函数调用者使用对象,且不应去 release 它  
//即函数调用的时候不是释放对象的好时机  
// autorelease == 延后调用一次release ,关于延后,具体是什么时机 ==> NSAutoreleasePool 相关  
//对象标记 autorelease 相当于 放到 NSAutoreleasePool 对象里,当 NSAutoreleasePool 对象调用 - drain 时,它会对里面所有对象都调用 release  

@interface Test : NSObject
+ (Test *)getInstance;
@end
@implementation Test
- (void)dealloc {
    NSLog(@"Test: dealloc");
}
+ (Test *)getInstance {
    // OC 约定放到最近的一个 NSAutoreleasePool 对象
    Test *result = [[[Test alloc] init] autorelease];
    return result;
}
@end

NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
Test *a = [Test getInstance];
//输出 Test: dealloc
[pool drain];

ARC

ARC(自动引用计数)相对于 MRC ,不仅仅在编译时自动添加 retain / release / autorelease ,应该是编译期和运行期两部分帮助开发者管理内存

ARC 下 不能使用 NSAutoreleasePool ,对于 autorelease 需求,有 NSAutoreleasePool ==> @autoreleasepool { … }, ARC 下 autorelease 的释放时机:

ARC下的dealloc

//创建B并且当回收时,先调用B的dealloc,再调用A的dealloc  

@interface A : NSObject
@end
@implementation A
- (void)dealloc {
    NSLog(@"A dealloc");
}
@end

@interface B : NSObject
@property A *a;
@end
@implementation B
- (void)dealloc {
    NSLog(@"B dealloc");
}
- (instancetype)init {
    self = [super init];
    if (self) {
        _a = [[A alloc] init];
    }
    return self;
}
@end

BAD_ACCESS

BAD_ACCESS 出现的情况分为两种:

CoreFoundation 管理内存

CoreFoundation 没有自动内存管理,需要手动释放内存

CFStringRef string = CFStringCreateWithCString(NULL, "苹果", kCFStringEncodingUTF8);  
CFRelease(string);

因此,当一个对象在 Foundation 或 CoreFoundation 中创建而需要在另一个系统中使用时,需要确定对象的管理权,分三种情况:

  1. 不改变
  2. 在 Foundation 中创建str,在CoreFoundation中手动管理
  3. 在 CoreFoundation 中创建str,在Foundation中自动管理

原则:明确告知由哪个系统来管理内存资源

  1. __bridge —— 跨层级使用,但不更改对象的拥有权
  2. __bridge_retained —— 内存归属修改,需手动release
  3. __bridge_transfer —— 内存归属修改,由ARC管理,不需手动release
NSString *str = @"苹果";
CFStringRef str2 = (__bridge CFStringRef)str;

NSString *str3 = @"苹果";
CFStringRef str4 = (__bridge_retained CFStringRef)str3;
CFRelease(str4);


CFStringRef string = CFStringCreateWithCString(NULL, "苹果", kCFStringEncodingUTF8);
NSString *string2 = (__bridge NSString *)string;
//...
CFRelease(string);

CFStringRef string3 = CFStringCreateWithCString(NULL, "苹果", kCFStringEncodingUTF8);
NSString *string4 = (__bridge_transfer NSString *)string3;

KVC & KVO

KVC

KVC ( Key Value Coding ),它把整个 object 看成 key 和 value 的对应关系
- setValue:forKey:
// 可重写用于处理没有对应key的情况
- setValue:forUndefinedKey:
// 嵌套使用
- setValue:forKeyPath:

- valueForKey:
- valueForUndefinedKey:
- valueForKeyPath:

KVC Collection Operators

@avg.
@sum.
@max.
@min.
@count

NSMutableArray* array = [NSMutableArray array];
NSDictionary* dic = @{@"name" : @"苹果", @"price" : @1.5};
Fruit* fruit = [[Fruit alloc] initWithDict:dict];
[array addObject:fruit];
NSDictionary* dic = @{@"name" : @"苹果", @"price" : @1};
Fruit* fruit = [[Fruit alloc] initWithDict:dict];
[array addObject:fruit];
NSDictionary* dic = @{@"name" : @"苹果", @"price" : @2.5};
Fruit* fruit = [[Fruit alloc] initWithDict:dict];
[array addObject:fruit];

CGFloat avg = [[array valueForKeyPath:@"@avg.price"] floatValue];
CGFloat max = [[array valueForKeyPath:@"@max.price"] floatValue];

KVO

KVO ( Key Value Observing )
- addObserver:forKeyPath:options:context:
- removeObserver:forKeyPath:
//观察的回调
- observeValueForKeyPath:ofObject:change:context:

Runtime

要使用Runtime相关方法,需导入Runtime框架#import <objc/runtime.h>
OC 可以转成 C 的 API,通过这些API可以达到高级的功能,这个C API也称为 Runtime

在运行时,类(Class)维护了一个消息分发列表来解决消息的正确发送。每一个消息列表的入口是一个方法(Method),这个方法映射了一对键值对,其中键值是这个方法的名字 selector(SEL),值是指向这个方法实现的函数指针 implementation(IMP)

OC函数调用本质

@interface Fruit : NSObject
@property CGFloat price;
@end
@implementation Fruit
@end

@interface Ferrari : NSObject
@property CGFloat price;
@end
@implementation Ferrari
@end

@interface Util : NSObject
@end
@implementation Util
+(CGFloat)getPriceForFruit:(Fruit *)f {
    return f.price;
}
@end


Ferrari *f = [[Ferrari alloc] init];
f.price = 5000000;
//输出5000000
NSLog(@"price: %@", @([Util getPriceForFruit:f]));

上面的代码Xcode会警告但编译运行仍会输出结果,对于 Util 的类方法 getPriceForFruit ,它接收的是 Fruit 类型的参数,但在代码中实际我们给的是 Ferrari 类型是实例对象
再看下面这一句代码:

CGFloat result = [cashier checkout];   

这句代码可以有两种说法:

objc_msgSend

首先用终端切换到项目目录下,输入 clang -rewrite-objc xxx.m ,把OC程序重写成C代码
简化代码可知每一个对象的消息发送,都是调用到

// id -- self,指发送消息的对象本身
// SEL -- _cmd,指消息名称
objc_msgSend(id, SEL, …)

即在 OC 动态编译时,方法在运行时会被动态转为消息发送,即: objc_msgSend()

@interface TestModel : NSObject
- (void)hahahaInstance;
+ (void)hahahaClass;
@end
@implementation TestModel
- (void)instanceMethodDealWithHahaha {
    NSLog(@"hahaha");
}
+ (void)classMethodDealWithHahaha {
    NSLog(@"hahaha");
}
// Method resolution
+ (BOOL)resolveClassMethod:(SEL)sel {
    NSLog(@"resolveClassMethod");
    return [super resolveClassMethod:sel];
}
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    NSLog(@"resolveInstanceMethod");
    if (sel == @selector(hahahaInstance)) {
        Method method = class_getInstanceMethod([self class], @selector(instanceMethodDealWithHahaha));
        IMP methodImp = method_getImplementation(method);
        const char *methodType = method_getTypeEncoding(method);
        class_addMethod([self class], sel, methodImp, methodType);
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}
// Fast forwarding
- (id)forwardingTargetForSelector:(SEL)aSelector {
    NSLog(@"forwardingTargetForSelector");
    if (aSelector == @selector(hahahaInstance)) {
        TestModel2 *testModel2 = [[TestModel2 alloc] init];
        return testModel2;
    }
    return [super forwardingTargetForSelector:aSelector];
}
// Normal forwarding
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    NSMethodSignature *signature = [super methodSignatureForSelector:aSelector];
    if (!signature) {
        Method m = class_getInstanceMethod([TestModel2 class], aSelector);
        const char *type = method_getTypeEncoding(m);
        signature = [NSMethodSignature signatureWithObjCTypes:type];
    }
    return signature;
}
- (void)doesNotRecognizeSelector:(SEL)aSelector {
    NSLog(@"doesNotRecognizeSelector");
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
    NSLog(@"forwardInvocation");
    TestModel2 *aaa = [[TestModel2 alloc] init];
    [anInvocation invokeWithTarget:aaa];
}
@end
@interface TestModel2 : NSObject
- (void)hahahaInstance;
@end
@implementation TestModel2
- (void)hahahaInstance {
    NSLog(@"TestModel2's hahaha");
}
@end

OC <=> C

要使用Runtime相关方法,需包含头文件 #include <objc/runtime.h>
Runtime C API 可做到:

获取一个类对象的所有属性名

unsigned int count = 0;
//objc_property_t 是一个C的数据结构,为属性的所有描述
objc_property_t *properties = class_copyPropertyList([XXX class], &count);
for (int i = 0; i < count; i++) {
    objc_property_t peoperty = properties[i];
    const char *propertyName = property_getName(properties[i]);
    NSString *name = [NSString stringWithUTF8String:propertyName];
    NSLog(@"%@", name);
}
free(properties);

Category添加属性(关联对象)

//.h 文件
#import <Foundation/Foundation.h>

@interface NSObject (ExtentNSObject)

@property (nonatomic, copy)NSString *name;

@end

//.m 文件
#import "NSObject+ExtentNSObject.h"
#include <objc/runtime.h>
//取key值一般有三种推荐的方式
//static 的地址是唯一的
static char kAssociatedObjectNameKey1;
//
static void *kAssociatedObjectNameKey2 = &kAssociatedObjectNameKey2;

@implementation NSObject (ExtentNSObject)

- (NSString *)name {
//    return objc_getAssociatedObject(self, &kAssociatedObjectNameKey1);
//    return objc_getAssociatedObject(self, kAssociatedObjectNameKey2);
    return objc_getAssociatedObject(self, _cmd);
}

- (void)setName:(NSString *)name {
//    objc_setAssociatedObject(self, &kAssociatedObjectNameKey1, name, OBJC_ASSOCIATION_COPY_NONATOMIC);
//    objc_setAssociatedObject(self, kAssociatedObjectNameKey2, name, OBJC_ASSOCIATION_COPY_NONATOMIC);
    objc_setAssociatedObject(self, @selector(name), name, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

@end

添加方法

在 runtime 中为类在运行时添加一个方法
class_addMethod()会为类添加一个其父类的实现方法的重写,但不会替换掉原来的方法的实现。也就是说,如果类没有重写父类的实现方法, class_addMethod 会用指定的实现方法重写父类的实现方法(相当于重写),但如果子类重写了父类的方法,则消息发送的还是重写的方法

// SEL -- 方法名
// IMP -- 方法必须带至少两个参数:self 和 _cmd  
// typeEncoding -- 用来描述消息实现体的参数和返回值类型的顺序:返回值 + 参数  
class_addMethod(class,SEL, IMP, typeEncoding)
//TestModel 没有重写 NSObject 的 description 方法
TestModel *testModel = [[TestModel alloc] init];
//输出为 <TestModel: 0x100202fa0>
NSLog(@"%@", [testModel description]);
NSObject *obj = [[NSObject alloc] init];
//输出为 <NSObject: 0x100202500>
NSLog(@"%@", [obj description]);

class_addMethod([TestModel class], @selector(description), imp_implementationWithBlock(^NSString*(TestModel* self){
    return @"hahaha";
}), "@@:");

//输出为 hahaha
NSLog(@"%@", [testModel description]);
//输出为 <NSObject: 0x100202500>
NSLog(@"%@", [obj description]);
typeEncoding

- (NSString *)description; ==> id id SEL
id OC对象在运行时都是OC对象
id 所有的事件都有一个隐含的参数self
SEL 所有的事件都有一个隐含的参数_cmd

方法替代

如果类中有对应的方法实现(不是其父类),则可以用 class_replaceMethod()

#import <Foundation/Foundation.h>
@interface NSObject (ExtentNSObject)
@end
#import "NSObject+ExtentNSObject.h"
#include <objc/runtime.h>
@implementation NSObject (ExtentNSObject)
- (NSString *)myDescription {
    return @"hahaha";
}
@end

TestModel *testModel = [[TestModel alloc] init];
//输出为 <TestModel: 0x100400800>
NSLog(@"%@", [testModel description]);

Method method = class_getInstanceMethod([NSObject class], @selector(myDescription));
IMP imp = method_getImplementation(method);
const char *typeEncoding = method_getTypeEncoding(method);  

class_replaceMethod([NSObject class], @selector(description), imp, typeEncoding);
//输出为 hahaha
NSLog(@"%@", [testModel description]);

这里用 category 扩展的方法实现 myDescription 代替了 NSObject 原本的 description
但是原本的 description 就被丢弃了

方法交换

method_exchangeImplementations() 交换了两个方法的实现

#import <Foundation/Foundation.h>
@interface NSObject (ExtentNSObject)
@end
#import "NSObject+ExtentNSObject.h"
#include <objc/runtime.h>
@implementation NSObject (ExtentNSObject)
- (NSString *)myDescription {
    //输出为 selector: description
    NSLog(@"selector: %@", NSStringFromSelector(_cmd));
    //调用了 myDescription 的实现
    return [NSString stringWithFormat:@"hahaha, %@", [self myDescription]];
}
@end

TestModel *testModel = [[TestModel alloc] init];
//输出为 <TestModel: 0x100400800>
NSLog(@"%@", [testModel description]);

Method method = class_getInstanceMethod([NSObject class], @selector(myDescription));
Method m = class_getInstanceMethod([NSObject class], @selector(description));
method_exchangeImplementations(method, m);
//输出为 hahaha, <TestModel: 0x100300d70>
NSLog(@"%@", [testModel description]);
Method Swizzle

Method Swizzle 要尽早的发生,在方法调用之前

#import <Foundation/Foundation.h>
@interface NSObject (ExtentNSObject)
@end
#import "NSObject+ExtentNSObject.h"
#include <objc/runtime.h>
@implementation NSObject (ExtentNSObject)
+ (void)load {
    Method method = class_getInstanceMethod([NSObject class], @selector(myDescription));
    Method m = class_getInstanceMethod([NSObject class], @selector(description));
    method_exchangeImplementations(method, m);
}
- (NSString *)myDescription {
    //输出为 selector: description
    NSLog(@"selector: %@", NSStringFromSelector(_cmd));
    //调用了 myDescription 的实现
    return [NSString stringWithFormat:@"hahaha, %@", [self myDescription]];
}
@end

TestModel *testModel = [[TestModel alloc] init];
//输出为 hahaha, <TestModel: 0x100400180>
NSLog(@"%@", [testModel description]);

Method Swizzle 最佳实践

+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Method newMethod = class_getInstanceMethod([NSObject class], @selector(myDescription));
        IMP newImp = method_getImplementation(newMethod);
        const char *newType = method_getTypeEncoding(newMethod);
        Method oldMethod = class_getInstanceMethod([NSObject class], @selector(description));
        IMP oldImp = method_getImplementation(oldMethod);
        const char *oldType = method_getTypeEncoding(oldMethod);
        BOOL addMethod = class_addMethod([self class], @selector(description), newImp, newType);
        if (addMethod) {
            NSLog(@"method add");
            class_replaceMethod([self class], @selector(myDescription), oldImp, oldType);
        } else {
            NSLog(@"method did not add");
            method_exchangeImplementations(newMethod, oldMethod);
        }
    });
}
- (NSString *)myDescription {
//    NSLog(@"selector: %@", NSStringFromSelector(_cmd));
    return [NSString stringWithFormat:@"hahaha, %@", [self myDescription]];
//    return @"hahaha";
}

NSString

初始化方法

//语法糖
NSString *str = @“xxoo”;  
NSString *str = [[NSString alloc] initWithUTF8String:"xxoo"];  
NSString *str = [NSString stringWithFormat:@"哈哈哈 %@", @"xxoo"] ;  

字符串比较

- isEqualToString:
// 返回值为NSComparisonResult类型的枚举值
- compare:
- compare:options:
- compare:options:range:

BOOL result = [str1 compare:str2 options:NSCaseInsensitiveSearch] == 0;//大小写无关相等

获取长度

.length

查找和替换

// 返回值NSRange不是一个OC对象,它是一个结构体,包含两个属性.location.length,当查找结果不存在时,.location的值为NSNotFound
- rangeOfString:

NSRange range = [str1 rangeOfString:@"xxoo"];

可修改子类 NSMutableString

//str不能修改原文,- stringByReplacingOccurrencesOfString: 是重新新建一个NSString对象并赋予实例变量
str1 = [str1 stringByReplacingOccurrencesOfString:@"xx" withString:@"ss"];
//NSMutableString可修改原文
\- replaceOccurrencesOfString:withString:options:range:

格式化字符串

//在OC输出只要带*的都可以用%@输出OC变量
%@
%d
%f
%p
_cmd
__FUNCTION__
__FILE__ //文件名
__LINE__ // 行数
__PRETTY_FUNCTION__ // 类名与方法名

其它

NSStringFromClass
NSStringFromSelector

NSArray

创建

NSArray *d = @[@“a”, @“b”];
// 等同于
NSString *raw[] = {@“a”, @“b”};
NSArray *d = [NSArray arrayWithObjects:raw count:2];

NSArray *d = [[NSArray alloc] initWithObjects:@1, @2, nil];

//指明ObjectType
NSArray<NSString *> d = @[@“a”, @“b”];

放置非OC对象

NSArray只接受OC对象,因此非OC对象需要包装

基本类型变量的包装

//语法糖
@1
//等同于
[NSNumber numberWithInteger:1]
//解包
[number integerValue];
const char *str = “haha”;
NSString *oc_str = @(str);

放置空对象

//Log打印输出为1, 2, 3
NSArray *d = [[NSArray alloc] initWithObjects:@1, @2, @3, nil, @4, nil];
//NSNull也是一个类
//Log打印输出为1, 2, 3, "<null>", 4
NSArray *d = [[NSArray alloc] initWithObjects:@1, @2, @3, [NSNull null], @4, nil];

取出元素

//语法糖
NSNumber *v = d[0];
// 等同于
NSNumber *v = [d objectAtIndex:0];

NOTE1: 如果取出的元素不存在(超出范围),会导致崩溃
但使用 .firstObject.lastObject 访问则不会,为空则返回null

迭代

for-in
- enumerateObjectsUsingBlock:

查找

//如果不存在,返回NSNotFound, NSNotFound为最大值
- indexOfObject:

可修改子类 NSMutableArray

- addObject:
- addObjectsFromArray:
- insertObject:atIndex:
- insertObjects:atIndexes:

- removeObject:
- removeObjectAtIndex:
- removeObject:inRange:
- removeLastObject
- removeAllObjects

- replaceObjectAtIndex:withObject:

- sortUsingDescriptors:
- sortUsingComparator:

NSArray *sortedArray = [array sortedArrayUsingComparator: ^(id obj1, id obj2) {
    if ([obj1 integerValue] > [obj2 integerValue]) {
        return (NSComparisonResult)NSOrderedDescending;
    }
    if ([obj1 integerValue] < [obj2 integerValue]) {
        return (NSComparisonResult)NSOrderedAscending;
    }
    return (NSComparisonResult)NSOrderedSame;
}];

NSIndexSet & NSMutableIndexSet

NSIndexSet 表示一个索引的集合(NSRange & NSInteger),一组index

NSMutableArray *d = [[NSMutableArray alloc] initWithObjects:@1, @2, @3, @4, @5, nil];
NSMutableIndexSet *indexSet = [NSMutableIndexSet indexSetWithIndex:1];
[indexSet addIndexesInRange:NSMakeRange(3, 1)];
//等同于
//NSRange range;
//range.location = 3;
//range.length = 1;
//[indexSet addIndexesInRange:range];
//结果为:1, 3, 5
[d removeObjectsAtIndexes:indexSet];

NSDictionary

NSDictionary的 key 需实现<NSCoping>协议的 - copyWithZone: 方法,value必须为oc对象

创建

// 语法糖
NSDictionary *fruitToPrice = @{@"苹果" : @1.5, @"草莓" : @1.2};
// 等同于
NSNumber *nums[] = {@1.5, @1.2};
NSString *strs[] = {@"苹果", @"梨子"};
NSDictionary *dic = [NSDictionary dictionaryWithObjects:(id *)nums forKeys:(id *)str count:2];

NSArray *nums = @[@1.5, @1.2];
NSArray *strs = @[@"苹果", @"梨子"];
NSDictionary *dic = [NSDictionary dictionaryWithObjects:nums forKeys:strs];

NSDictionary *dic = [NSDictionary dictionaryWithObjectsAndKeys:@1.5, @"苹果", @1.2, @"梨子", nil];

// copyItems 为 NO 时为浅复制,为 YES 时为深复制,集合里的每个对象都会收到 copyWithZone: 消息。如果集合里的对象遵循 NSCopying 协议,那么对象就会被深复制到新的集合。如果对象没有遵循 NSCopying 协议,而尝试用这种方法进行深复制,会在运行时出错。
- initWithDictionary:copyItems:

迭代

// NSDictionary 是无序的
for-in

for (id key in dic.allKeys) {
    id value = [dic objectForKey:key];
}

- enumerateKeysAndObjectsUsingBlock:

可修改子类 NSMutableDictionary

- setObject:forKey:

NSMutableDictionary *fruitToPrice = [NSMutableDictionary dictionaryWithObjectsAndKeys:@1.5, @"苹果", nil];

[fruitToPrice setObject:@1.2 forKey:@"草莓"];
// 等同于
fruitToPrice[@"草莓"] = @1.2;

NSSet

NSSet 无序且各个元素互不相等

//.count == 4
NSSet *set = [NSSet setWithOnject:@1, @2, @3, @4, @1, nil];

NSString *str = @“a”;
NSString *str2 = @“a”;
// set.count == 1
// isEqual:和 hash 是NSSet用来判断两个对象是不是同一个对象
NSSet *set = [NSSet setWithObjects:str, str2, nil];

// 深复制 copyItems每一项都会执行 - copyWithZone:
- initWithSet:copyItems:

可修改子类 NSMutableSet

NSString / NSNumber / NSArray / NSSet / NSDictionary 的相互转换

NSString <=> NSNumber

NSString *str = @"18";
NSNumber *num = @([str integerValue]);

NSString *str = @"18abc”;
//18
NSNumber *num = @([str integerValue]);

NSString *str = @"abc”;
//0
NSNumber *num = @([str integerValue]);

NSString *str = nil;
//0
NSNumber *num = @([str integerValue]);

NSNumber *num2 = @18;
NSString *str2 = [num2 stringValue];

NSString <=> NSArray

NSString *str = @"a, b, c";
NSArray *array = [str componentsSeparatedByString:@","];

NSArray *array2 = @[@"a", @"b", @"c"];
NSString *str2 = [array2 componentsJoinedByString:@", "];

NSArray <=> NSSet

//去重
NSArray *array = @[@"a", @"b", @"c", @"a"];
NSSet *set = [[NSSet alloc] initWithArray:array];

NSSet *set2 = [[NSSet alloc] initWithObjects:@"a", @"b", @"c", nil];
NSMutableArray *array2 = [NSMutableArray array];
for (id value in set2) {
     [array2 addObject:value];
}

NSArray <=> NSDicytionary

NSArray *array = @[@"苹果", @"草莓"];
NSArray *array2 = @[@1.5, @1.2];
NSDictionary *dic = [NSDictionary dictionaryWithObjects:array2 forKeys:array];

NSArray *keys = dic.allKeys;
NSArray *values = dic.allValues;

Block

Block 的声明和使用

Block 的声明
int (^myBlock)(int) = ^(int num) {
    return num * multiplier;
};

//形参
- (CGFloat)calcPriceWithDiscountHandler:(CGFloat(^)(Fruit *fruit))handler;  // CGFloat(^)(Fruit *fruit) 指block的类型
//实参
[fruit calcPriceWithDiscountHandler:^CGFloat(Fruit *fruit) {
    return 1.0;
}];

// typedef Block
typedef int(^myBlock)(NSString *name);
@property (nonatomic, copy) myBlock mBlock;

__wesk & __block

NSData

NSData 表示一块内存区域

NSString *str = @"abc";
NSData *data = [str dataUsingEncoding:NSUTF8StringEncoding];
NSString *str2 = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];

NSError

NSError 包含

/+ errorWithDomain:code:userInfo:

NS_ENUM & NS_OPTIONS

typedef NS_ENUM(NSInteger, CYLSex) {
    CYLSexMan,
    CYLSexWoman
};

参考

  1. iOS 集合的深复制与浅复制
  2. Objective-C Associated Objects 的实现原理
  3. Method Swizzling
  4. 《招聘一个靠谱的iOS》面试题参考答案(上)
  5. 《招聘一个靠谱的iOS》面试题参考答案(下)
上一篇 下一篇

猜你喜欢

热点阅读