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/reference/foundation/nsvalue/1551466-valuewithbytes?language=objc
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· DeepSeek 开源周回顾「GitHub 热点速览」
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了