NSValue的valueWithBytes:objCType:方法

+ (NSValue *)valueWithBytes:(const void *)value objCType:(const char *)type;

NSValue的valueWithBytes:objCType:方法将参数value指针指向的内容存储到新创建的NSValue对象中,而指针指向的内容类型由type参数指定。比如下面的例子:

char *myCString = "123456789";
NSValue *theValue = [NSValue valueWithBytes:myCString objCType:@encode(char *)];
        
char buffer[10] = {0};
[theValue getValue:buffer];
NSLog(@"%s", buffer);

上面代码输出:12345678,而不是123456789。

原因是valueWithBytes:objCType方法保存的是myCString指向的内容(即字符串"123456789"),并且将这个内容当成一个char *指针,而一个指针只有8个字节(具体大小可能因机器而异),所以theValue只能保存字符串的前8个字节,因此,使用NSValue的getValue:方法取出的内容是12345678,而不是123456789。具体流程如下图所示:

 

 要解决这个问题,可以向valueWithBytes:objCType传递myCString变量本身的地址,如下面代码所示:

char *myCString = "123456789";
NSValue *theValue = [NSValue valueWithBytes:&myCString objCType:@encode(char *)];
        
char *buffer = NULL;
[theValue getValue:&buffer];
NSLog(@"%s", buffer);

这时输出的内容就是:123456789。

上面的代码中,在调用valueWithBytes:objCType之前,各个指针变量指向的值如下图所示:

在上面的图中,变量名上方的数字表示该指针变量存储的地址值,比如,myCString指向字符串123456789,这个字符串的首地址是11111111,buffer变量指向NULL。

调用vaueWithBytes:objCType之后,由于向valueWithBytes:objCType传递的是myCString变量本身的地址&myCString,那么theValue就保存这个地址所指向的内容,即myCString变量保存的地址值11111111,并把这个值当成char *看待。当将buffer变量本身的地址传递给NSValue的getValue方法时,getValue方法将theValue保存的值copy给&buffer指向的缓存区,即copy给buffer变量,因此,buffer变量保存的值就成了11111111,这个值就是字符串123456789的地址,buffer变量就指向了这个字符串。具体流程如下图所示:

 

还需要主意的是,上面代码传递给valueWithBytes:objCType方法的type参数不一定非要是char *,只要是指针类型就可以,因为指针的大小在同一台机器上是固定的。

参考资料

https://developer.apple.com/library/content/documentation/Cocoa/Conceptual/NumbersandValues/Articles/Values.html#//apple_ref/doc/uid/20000174-BAJJHDEG

https://developer.apple.com/reference/foundation/nsvalue/1551466-valuewithbytes?language=objc

 

posted @ 2017-01-12 18:41  chaoguo1234  阅读(1806)  评论(0编辑  收藏  举报