+ (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/reference/foundation/nsvalue/1551466-valuewithbytes?language=objc