参考博客
http://blog.csdn.net/yhawaii/article/details/8222973
http://blog.csdn.net/sakulafly/article/details/13292457
http://blog.csdn.net/at9009/article/details/7863720
我们在编码中,很多时候需要将C里面原生的数据封装成对象,这样可以用NSDictionary或者NSArray来存取访问。尤其是一些做适配的情况下,这种封装是不可避免的。Objective-C提供了不少类可以帮助我们,比较常见的是NSNumber,NSValue和NSData。
NSNUmber
NSNumber主要是用来封装ANSI C内置的数据,比如char,float,int等等。
这个类提供了一批封装/解封的方法
+ (NSNumber *)numberWithChar:(char)value;
+ (NSNumber *)numberWithUnsignedChar:(unsigned char)value;
+ (NSNumber *)numberWithShort:(short)value;
+ (NSNumber *)numberWithUnsignedShort:(unsigned short)value;
+ (NSNumber *)numberWithInt:(int)value;
+ (NSNumber *)numberWithUnsignedInt:(unsigned int)value;
+ (NSNumber *)numberWithLong:(long)value;
+ (NSNumber *)numberWithUnsignedLong:(unsigned long)value;
- (char)charValue;
- (unsigned char)unsignedCharValue;
- (short)shortValue;
- (unsigned short)unsignedShortValue;
- (int)intValue;
- (unsigned int)unsignedIntValue;
- (long)longValue;
- (unsigned long)unsignedLongValue;
写起来一般就是:
NSNumber *number = [NSNumber numberWithInt: 1234];
int i = [number intValue];
@encode
从字典中取出的NSNumber无法通过isKindOfClass判断具体的数据类型,可以使用@encode来判别
NSDictionary *dic = [NSDictionary dictionaryWithObjectsAndKeys:[NSNumber numberWithBool:YES],@"key1",[NSNumber numberWithDouble:1.00f],@"key2",[NSNumber numberWithInt:1],@"key3",[NSNumber numberWithFloat:33.0f], @"key4", nil]; for(NSString *key in dic){ id value = [dic valueForKey:key]; if([value isKindOfClass:[NSNumber class]]){ const char * pObjCType = [((NSNumber*)value) objCType]; NSLog(@"%s",pObjCType); if (strcmp(pObjCType, @encode(int)) == 0) { NSLog(@"字典中key=%@的值是int类型,值为%d",key,[value intValue]); } if (strcmp(pObjCType, @encode(float)) == 0) { NSLog(@"字典中key=%@的值是float类型,值为%f",key,[value floatValue]); } if (strcmp(pObjCType, @encode(double)) == 0) { NSLog(@"字典中key=%@的值是double类型,值为%f",key,[value doubleValue]); } if (strcmp(pObjCType, @encode(BOOL)) == 0) { NSLog(@"字典中key=%@的值是bool类型,值为%i",key,[value boolValue]); } } }
NSValue
NSValue主要用来封装自定义的数据结构,可以是系统框架提供的CGRect/CGPoint/CGSize等数据结构,也可以是自己定义的struct。
这个类提供的封装/解封方法
+ (NSValue *)valueWithBytes:(const void *)value objCType:(const char *)type;
- (void)getValue:(void *)value;
例如:
typedef struct testTag
{
int a;
char *b;
}TEST_DATA_ST;
-- 定义数据结构
TEST_DATA_ST stValue = {0};
stValue.a = 58;
-- 生成数据结构,并赋初值
NSValue *value = [NSValue valueWithBytes:&stValue objCType:@encode(TEST_DATA_ST)];
-- 封装,注意type类型使用@encode来包装
TEST_DATA_ST testValue = {0};
[value getValue:&testValue];
-- 解封,获得数据
同时,针对很常见CGRect/CGPoint/CGSize等数据结构,提供了封装好的接口
+ (NSValue *)valueWithCGPoint:(CGPoint)point;
+ (NSValue *)valueWithCGSize:(CGSize)size;
+ (NSValue *)valueWithCGRect:(CGRect)rect;
- (CGPoint)CGPointValue;
- (CGSize)CGSizeValue;
- (CGRect)CGRectValue;
需要指出一下,NSNumber实际上是NSValue的一个特殊情况,所以在类关系中,NSNumber是NSValue的子类。
NSData
NSData主要是提供一块原始数据的封装,方便数据的封装与流动,比较常见的是NSString/NSImage数据的封装与传递。在应用中,最常用于访问存储在文件中或者网络资源中的数据。
这个类提供的封装/解封方法
+ (id)dataWithBytes:(const void *)bytes length:(NSUInteger)length;
- (NSUInteger)length;
- (const void *)bytes NS_RETURNS_INNER_POINTER;
从这几个方法可以看出,NSData根本不管传递的内容到底是什么,仅仅是传递一块内存 —— 仅需内存的起始地址和长度
例如:
char *str = "Hello, World!";
NSData *strData = [NSData dataWithBytes:str length:strlen(str)+1];
-- 封装
NSString *str = [NSString stringWithUTF8String:[strData bytes]];
[strData bytes]就可以获得传递的内容