浅谈NSString*做为形参传值

  • 前言

我们知道,在C语言中,基本数据类型做为形参传递是无法直接更改其值的,因为是值传递,在函数中更改完毕后系统会收回为此函数分配的内存能及里面的局部变量。如果想要更改,可以通过指针,根据地址找到找到对应的值,然后可以更改。

  • 问题

那么问题来了,比如在OC中,拿NSString为特例,我声明的NSString明明也是NSString* ,并且我传递的也是NSString*指针,但是为什么不能更改?代码如下:

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
NSString *str = @"made";
NSLog(@"1----%p",str);
NSLog(@"2----%p",&str);
[self stringTestChange:str];
NSLog(@"after - %@",str);
}

- (void)stringTestChange:(NSString *)aString{
NSLog(@"3----%p",aString);
NSLog(@"4----%p",&aString);
aString = @"madeChina";
}

上述打印结果:


打印结果

可以看到值并没有改变,但是从上述打印,我们也可以看到,str与aString的值是相同的,但是你会发现它们的地址去并不相同。

也就是说,在函数中aString是重新分配了内存空间,地址不一样,但是它们指向的都是同一块存储空间,当再赋于新值给aString的时候,只不过是改变了它的指向而已,等函数结束后,原str的指向并未发生任何改变。当执行到打印after时,被改变的指向仅仅是aString,存储结构大致如下:

图解一

所以如果想通过函数改变原字符串,需要找到指向str的地址。所以更改如下:

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
NSString *str = @"made";
[self stringTestChange:&str];
NSLog(@"after - %@",str);
}
- (void)stringTestChange:(NSString **)aString{
*aString = @"madeChina";
}

这样再一打印,是没问题的。

  • 结论

为什么这样可以?因为我们首先通过地址找到了str,然后通过*str找到了指向made字符串的指针,然后重新改变了它的指向,最后就修改完成了。
图示如下:

图解一

如有问题,希望指出,一起探讨交流!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • __block和__weak修饰符的区别其实是挺明显的:1.__block不管是ARC还是MRC模式下都可以使用,...
    LZM轮回阅读 3,453评论 0 6
  • 307、setValue:forKey和setObject:forKey的区别是什么? 答:1, setObjec...
    AlanGe阅读 1,619评论 0 1
  • 1、 OC中与alloc相反的方法是:答案:(C) A、release B、retain C、dealloc ...
    失忆的程序员阅读 2,916评论 0 6
  • 多线程、特别是NSOperation 和 GCD 的内部原理。运行时机制的原理和运用场景。SDWebImage的原...
    LZM轮回阅读 2,049评论 0 12
  • 我想我们的关系如同那结扎血管的外科结,到老去的时候都不会脱落! 我不能保证是最好的那个人,但我愿意给你最好的我!
    f5c8833c9a44阅读 227评论 0 0