一、OC的包装类
OC提供了NSValue、NSNumber来封装C语言基本类型(short、int、float等)。
在 Objective-C 中,**包装类(Wrapper Classes)**是用来把基本数据类型(如 int、float、char 等)“包装”为对象的类。因为 Objective-C 是面向对象的语言,有时候我们需要把基本类型当作对象使用,比如:
-
放入 NSArray、NSDictionary 这样的集合中(这些集合只能存放对象);
-
使用对象方法对数值进行操作;
-
与 Foundation 框架接口交互。
1.1 以下不是包装类:
1、NSInteger:大致等于long型整数
2、NSUInteger:大致等于unsigned long型整数
3、CGFLoat:在64位平台相当于double,在32位平台相当于float
以上的类型只是基本类型。为了更好的兼容不同的平台,当程序需要定义整形变量的时候,建议使用NSInteger,NSUInteger;当程序需要定义浮点型变量的时候,建议使用CGFLoat。
1.2 以下是包装类:
1、NSValue是NSNumber的父类,它代表一个更通用的包装类,可以包装int、short、long、float、char、指针、对象id等数据项。并将它们添加到NSArray、NSSet等集合中去。
2、NSNumber是更具体的包装类,用于包装c语言的各种数值类型。它有如下三类方法:
- [x] +numberWithXxx:直接将特定类型的值包装成NSNumber
- [x] -initWithXxx:该实例方法需要先创建一个NSNumber对象,再用一个基本类型的值来初始化NSNumber
- [x] -xxxValue:该实例方法返回该NSNumber对象包装的基本类型的值
如上方法的Xxx是Int,Char,Double,string等各种数据类型。
使用NSNumber的compare方法比较两个值,返回的对象可以转化为-1、0、1,分别代表小于、等于、大于。与bool值比较时,YES代表1,当另一个数大于1时返回1,小于1时返回-1。
基本类型变量和包装类对象之间的转换关系可以理解为:基本类型变量通过调用numberWithXxx:类方法来转换并返回包装类对象;包装类对象通过调用xxxValue来获取基本类型的值。
#import <Foundation/Foundation.h>int main(int argc, const char * argv[]) {@autoreleasepool {NSNumber *num = [NSNumber numberWithInt:66];NSNumber *de = [NSNumber numberWithDouble:7.7];NSLog(@"%d",[num intValue]);NSLog(@"%g",[de doubleValue]);NSNumber *ch = [[NSNumber alloc] initWithChar:'t'];NSLog(@"%@",ch);}return 0;
}
二、处理对象
2.1 处理对象和description方法
在 Objective-C 中,打印对象(NSLog(@"%@", obj)) 实际上是调用对象的 -description 方法。这个方法决定了你在控制台看到的输出内容。
一、NSLog(@"%@", obj) 做了什么?
当你写:
NSLog(@"%@", obj);
它相当于:
NSLog(@"%@", [obj description]);
也就是说:
NSLog 并不会直接打印对象地址;
-
它调用了 obj 的 -description 方法,获取一个 NSString* 类型的描述字符串来打印。
二、默认行为
如果你没有重写 -description,那么会输出类似:
<ClassName: 0x10060ae50>
这是 NSObject 默认的格式,表示“类名 + 内存地址”。
三、如何自定义打印内容?
你可以重写 -description 方法来自定义输出内容:
示例:
@interface Person : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@end@implementation Person
- (NSString *)description {return [NSString stringWithFormat:@"Person: name=%@, age=%d", self.name, self.age];
}
@end
使用:
Person *p = [[Person alloc] init];
p.name = @"Tom";
p.age = 20;
NSLog(@"%@", p);
输出:
Person: name=Tom, age=20
四、打印集合对象(NSArray、NSDictionary)
集合类如 NSArray、NSDictionary、NSSet,当你 NSLog 打印它们时,它们也会调用内部所有对象的 -description 方法。
NSArray *arr = @[p];
NSLog(@"%@", arr);
如果你没有给 p 写 -description,你就会看到一串地址;
如果写了,就会输出里面每个对象的自定义内容。
#import <Foundation/Foundation.h>@interface Person : NSObject
@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) int age;
@end@implementation Person//重写 description 方法
- (NSString *)description {return [NSString stringWithFormat:@"Person: name=%@, age=%d", self.name, self.age];
}@endint main(int argc, const char * argv[]) {@autoreleasepool {Person *p = [[Person alloc] init];p.name = @"Tom";p.age = 20;// 打印对象NSLog(@"%@", p); // 自动调用 [p description]}return 0;
}
不重写 VS 重写后
<Person: 0x600000e812e0>Person: name=Tom, age=20
2.2 == 和 isEqual方法
oc中测试两个变量事都相等的方式有两个,分别是:==方法和isEqual方法
2.2.1 ==方法
当用==方法时,若️①两个变量是基本类型的变量,️②两个变量都是数值型的变量(不一定要求数据类型严格相等),️③两个变量的值相等。则==判断返回真,否则返回假。
而对于指针类型的变量,则要两个指针指向同一个对象,则==返回真,否则返回假。
当使用==的两个类没有继承关系时,编译器会提示警告。
@“hello”和[NSString stringWithFormat:@“hello”]的区别:
当OC直接使用@”hello“,系统会使用常量池来管理这些字符串。常量池保证相同的字符串只会有一个,不会产生多个副本,因此创建的所有指向@“hello”的指针,指针变量保存的地址都是完全相同的。
而使用[NSStringstringWithFormat:@“hello”]创建的字符串对象是运行时创建出来的,它被保存在运行时的内存中(即堆内存),不会放入常量池中。因此它的地址和@“hello”的地址并不相同。
以下代码演示了==的用法:
#import <Foundation/Foundation.h>int main(int argc, const char * argv[]) {@autoreleasepool {int it = 65;int fl = 65.0f;char ch = 'A';NSString *str1 = @"hello";NSString *str2 = @"hello";NSString *str3 = @"byebye";NSLog(@"%d",(it==fl)); //结果为1NSLog(@"%d",(fl == ch)); //结果为1NSLog(@"%d",(str1 == str2)); //结果为1NSLog(@"%d",(str2 == str3)); //结果为0//常量池NSString *p1 = @"朱斌";NSString *p2 = @"朱斌";NSLog(@"p1地址:%p,p2地址:%p",p1,p2);NSLog(@"%d",(p1 == p2)); //结果为1NSString *p3 = [NSString stringWithFormat:@"朱斌"];NSString *p4 = [NSString stringWithFormat:@"朱斌"];NSLog(@"p3地址:%p",p3);NSLog(@"p4地址:%p",p4);NSLog(@"%d",(p1 == p3)); //结果为0NSLog(@"%d",(p4 == p3)); //结果为0NSString *p5 = [NSString stringWithFormat:@"zbchi"];NSString *p6 = [NSString stringWithFormat:@"zbchi"];NSLog(@"p5地址:%p",p5);NSLog(@"p6地址:%p",p6);NSLog(@"%d",(p5 == p6)); //结果为1}return 0;
}
2.2.2 isEqual方法
isEqual比较的是对象的内容。
isEqual默认实现是比较地址(跟 == 一样),但很多系统类(如NSString,NSArray,NSNumber)都 重写该方法 来比较内容。所以这个时候输出的是 内容相等
#import "FKPerson.h"@implementation FKPerson- (id) initWithName: (NSString*) name idStr: (NSString*) idStr {if(self = [super init]) {self.name = name;self.idStr = idStr;}return self;
}
- (BOOL) isEqual:(id) other {//如果两个对象指针相等,为同一个对象if(self == other) {return YES;}//当other不为nil且它为FKPerson的实例时if(other != nil && [other isMemberOfClass:FKPerson.class]) {FKPerson* target = (FKPerson*)other;//并且要判断当前对象的idStr和target对象的idStr相等才可以判断两个对象相等return [self.idStr isEqual: target.idStr];}return NO;
}@end
三、类别与拓展
在oc中,类别和拓展都是对类进行的“补充”机制。
3.1 类别
类别是oc中用于給已有方法添加方法的一种机制,不能添加成员变量
类别的定义:
命名规则:在接口文件部分的文件命名是“类名+类别名.h” 在实现部分的文件命名是“类名+类别名.m” 的形式。
类别的接口部分的声明和类的定义十分相似,但类别不继承父类,只需要在已有类的类名后面加一个括号,写入类别名,然后再在下面定义方法。
@interface ClassName (CategoryName)
- (void)newMethod;
@end@implementation ClassName (CategoryName)
- (void)newMethod {NSLog(@"Category method called");
}
@end