OC参数传递的那些坑

OC参数传递的那些坑

有一些原本我们认为很基础的,而且很理所当然的,在实践之后才发现,原来跟想象的不同

回顾一下c语言的参数传递

C语言中的参数传递

1、传值方式
原理:形参和实参占不同内存单元,传递的实际上是实参变量或表达式的一个拷贝副本,将这个副本值传给形参,形参内存单元内容保存的正是这个副本值,相当于给形参进行初始化,形参的值发生变化也不会传回给实参,因此是单向传递。
例如:

 void increase(int x) {
    x++;
 }

当在主函数中调用上面这个函数时,会在ncrease函数内存栈中为形参x分配一个内存单元,然后把实参的值传到这个内存单元中,相当于给形参初初化了,然后形参x自增1,它改变的仅仅是形参内存单元中的内容,而实参内存单元中的内容并没有改变。当被调用函数执行完毕后,形参所分配的内存单元也被收回。

2、传地址方式
原理:和传值方式一样,当调用函数时也要为形参分配内存,被调函数执行完毕后也要收回内存。不同的是传递的是实参变量地址的拷贝值,而不是实参变量的值,在被调函数中对地址所指对象的操作会改变实参的值。但是形参的内容即存放的实参变量地址并不会改变。

OC中传递对象就是用这种方式呀,但是并没有被改变 我很疑惑

例如:

 void increase(int * x)
 {*x)++;
 }
 int main()
 {
 	int i, *x;
 	i = 10;
 	*x = 20;
 increase(&i);//如果定义的不是指针变量,那变要加上个取址号&,当然如果形参实参是数组的话,直接用数组名即可,因为数组名本身代表的就是数组的首地址
 increase(x);
 }

主函数调用被调函数后,主函数中的i和*x的值都会改变。

OC参数传递

NSString
先看看测试的代码

- (void)viewDidLoad {
  [super viewDidLoad];
 
  NSString *str = @"viewDidLoad";
  NSLog(@"viewDidLoad 改变前:%@",str);
  [self changNSString:str];
  NSLog(@"viewDidLoad 改变后:%@",str);
 
}
- (void)changNSString:(NSString *)str {
  NSLog(@"changNSString 改变前:%@",str);
  str = @"changNSString";
  NSLog(@"changNSString 改变后:%@",str);
}

大家觉得NSLog(@“viewDidLoad 改变后:%@”,str);会打印什么.

可以改变str的值么

经过测试,str 经过函数changNSString的修改后根本没有改变

** CommentDemo[8807:186874] viewDidLoad ****改变前****:viewDidLoad**
** CommentDemo[8807:186874] changNSString ****改变前****:viewDidLoad**
** CommentDemo[8807:186874] changNSString ****改变后****:changNSString**
** CommentDemo[8807:186874] viewDidLoad ****改变后****:viewDidLoad**

这完全不符合常理呀!这是传递的地址呀.

NSMutableString
之后我换可变的字符串对象测试了一下:

- (void)viewDidLoad {
  [super viewDidLoad];
 
  NSMutableString *mStr = [NSMutableString stringWithString:@"viewDidLoad"];
  NSLog(@"viewDidLoad 改变前:%@",mStr);
  [self changeMutableString:mStr];
  NSLog(@"viewDidLoad 改变后:%@",mStr);
}

- (void)changeMutableString:(NSMutableString *)mStr {
  NSLog(@"changNSMutableString 改变前:%@",mStr);
  mStr = [NSMutableString stringWithString:@"changeMutableString"];
  NSLog(@"changNSMutableString 改变后:%@",mStr);
}

测试结果


** CommentDemo[8856:191232] viewDidLoad ****改变前****:viewDidLoad**
** CommentDemo[8856:191232] changNSMutableString ****改变前****:viewDidLoad**
** CommentDemo[8856:191232] changNSMutableString ****改变后****:changeMutableString**
** CommentDemo[8856:191232] viewDidLoad ****改变后****:viewDidLoad**

NSMutableArray
后来我换了可变的数组

- (void)viewDidLoad {
  [super viewDidLoad];
 
  NSMutableArray *mArray = [NSMutableArray arrayWithObjects:@"1",@"2", nil];
  for (NSString *str in mArray) {
    NSLog(@"viewDidLoad 改变前 %@",str);
  }
 
  [self changeMutableArray:mArray];
 
  for (NSString *str in mArray) {
    NSLog(@"viewDidLoad 改变后 %@",str);
  }
 
 
}
- (void)changeMutableArray:(NSMutableArray *)array {
  [array addObject:@"3"];
  for (NSString *str in array) {
    NSLog(@"changeMutableArray %@",str);
  }
}

测试结果

** CommentDemo[8925:196443] viewDidLoad ****改变前**** 1**
** CommentDemo[8925:196443] viewDidLoad ****改变前**** 2**
** CommentDemo[8925:196443] changeMutableArray 1**
** CommentDemo[8925:196443] changeMutableArray 2**
** CommentDemo[8925:196443] changeMutableArray 3**
** CommentDemo[8925:196443] viewDidLoad ****改变后**** 1**
** CommentDemo[8925:196443] viewDidLoad ****改变后**** 2**
** CommentDemo[8925:196443] viewDidLoad ****改变后**** 3**

值改变了

经过测试,可以知道在OC中,看似对象是声明为指针类型(* Str),由于传递的时候我们没有用取地址(&).结果根本没有改变. 我开始怀疑传递的是不是地址了.是不是因为自始至终没有用到过取地址符.

然后自己改为了如下方式,就能够改变传进来的值了

能够改变传递的值
NSString正常改变

- (void)viewDidLoad {
  [super viewDidLoad];
  NSString *str = @"viewDidLoad";
  NSLog(@"viewDidLoad 改变前:%@",str);
  [self changNSString:&str];
  NSLog(@"viewDidLoad 改变后:%@",str);
}
- (void)changNSString:(NSString **)str {
  NSLog(@"changNSString 改变前:%@",*str);
  *str = @"changNSString";
  NSLog(@"changNSString 改变后:%@",*str);
}

测试结果


** CommentDemo[8993:203983] viewDidLoad ****改变前****:viewDidLoad**
** CommentDemo[8993:203983] changNSString ****改变前****:viewDidLoad**
** CommentDemo[8993:203983] changNSString ****改变后****:changNSString**
** CommentDemo[8993:203983] viewDidLoad ****改变后****:changNSString**

总结
看到这种状况我能说服自己的就是:
NSString *str = @“viewDidLoad”;这样的定义在OC中其实就是类似于c语言的普通变量而已,而不是指针变量.智商有点不够用.

原因或许很简单
看似和c语言地址传递一样,同样是*传递。但c语言拿到这个指针之后取了指针指向的内容并改变了内容。而oc中我们习惯直接str =✘。让这个指针指向了新的地方。并没有改变函数外面原来指针指向的内容。这点特别重要,指针传过来那个str本质是值传递,相当于copy了一份。所以在change函数里面的str并不是外面的str。哈哈。今天有点短路。

还是一点,c语言中改变了传进来指针原来指向的内容,而oc中只改变了新指针指向的地方。

谈谈自定义对象,给对象赋值又是怎样的.
经过测试,给对象的属性复制能够在另一个函数改变其属性
看看测试结果

** CommentDemo[3027:130157] name: xiaomao, age: 17**
** CommentDemo[3027:130157] name: newxiaomao, age: 89**
** CommentDemo[3027:130157] name: newxiaomao, age: 89**

测试代码

- (void)createCat {
    Cat *cat = [Cat new];
    cat.name = @"xiaomao";
    cat.age = @"17";
    NSLog(@"%@",cat);
    [self catObjectTest:cat];
    NSLog(@"%@",cat);
}
- (void)catObjectTest:(Cat *)cat {
    cat.name = @"newxiaomao";
    cat.age = @"89";
    NSLog(@"%@",cat);
}

分析
当我在调用对象的属性进行赋值的时候,其实是调用getter/setter方法,通过这样的赋值,属性值肯定改变呀.

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值