Java函数参数传递方式详解

转:http://zzproc.iteye.com/blog/1328591

在阅读本文之前,根据自己的经验和理解,大家可以先思考并选择一下Java函数的参数传递方式: 
A. 是按值传递的? 
B. 按引用传递的? 
C. 部分按值部分按引用? 
此处暂不宣布正确答案,我们通过一个简单的例子让大家自己找答案: 
1. 先定义一个类型Value 

Java代码   收藏代码
  1. public static class Value {  
  2.     private String value = "value";  
  3.     public String getValue() { return value; }  
  4.     public void setValue(String value) { this.value = value; }  
  5. }  

2. 写两个函数newValue和modifyValue:newValue会将入参指向一个新的对象,modifyValue会调用入参的setValue方法修改对象的value值。 
Java代码   收藏代码
  1. public static void newValue(Value value) {  
  2.     value = new Value();  
  3.     value.setValue("new value");  
  4.     System.out.println("In newValue, HashCode = " + value.hashCode() + ", value = " + value.getValue());  
  5. }  
  6.       
  7. public static void modifyValue(Value value) {  
  8.     value.setValue("new value");  
  9.     System.out.println("In modifyValue, HashCode = " + value.hashCode() + ", value = " + value.getValue());  
  10. }  

3. 简单的测试代码 
Java代码   收藏代码
  1. public static void main(String[] args) {  
  2.     Value value1 = new Value();  
  3.     System.out.println("Before modify, HashCode = " + value1.hashCode() + ", value = " + value1.getValue());  
  4.     // 将value1指向新的Value对象  
  5.     newValue(value1);  
  6.     System.out.println("After modify, HashCode = " + value1.hashCode() + ", value = " + value1.getValue() + "\n");  
  7.     Value value2 = new Value();  
  8.     System.out.println("Before modify, HashCode = " + value2.hashCode() + ", value = " + value2.getValue());  
  9.     // 使用object的set方法,修改对象的内部值  
  10.     modifyValue(value2);  
  11.     System.out.println("After modify, HashCode = " + value2.hashCode() + ", value = " + value2.getValue());  
  12. }  

4. 执行结果日志: 
Java代码   收藏代码
  1. Before modify, HashCode = 12677476, value = value  
  2. In newValue, HashCode = 33263331, value = new value  
  3. After modify, HashCode = 12677476, value = value  
  4.   
  5. Before modify, HashCode = 6413875, value = value  
  6. In modifyValue, HashCode = 6413875, value = new value  
  7. After modify, HashCode = 6413875, value = new value  


5. 结果分析: 
上述代码这是非常常见的一种编程模式:在外围定义|保存|获取一个值或对象,将这个对象作为参数传入一个方法,在方法中修改对象的属性、行为。但两个方法newValue和modifyValue的修改方式不一样,在方法调用之后,该对象在外围看来也有很大的差别!如何理解这种差异呢?先温故一下按值传递、按引用传递的概念: 
* 按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。 
* 按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数,参数的原始值(函数块之外的调用代码中)也随之改变。 
正确的答案:A——Java函数是按值传递参数的!  
分析一下日志: 
* 第一段日志输出,value1参数在newValue方法内部被改为指向新对象,并输出了新对象的hashCode和value值,但跳出newValue方法域之后,在main方法中的value1没有发生任何变化,这符合按值传递的定义和特点;如果是按引用传递,value1在调用newValue(Value value)方法之后,应该是发生变化的。 
* 第二段日志输出,value2在modifyValue方法内部进行了setValue操作,hashCode不变而value被修改,离开modifyValue方法域之后,在main方法中value2确实发生了变更。使用过C++的人容易将这种现象理解为:按引用传递函数参数!因为这跟C++中的按引用传递像极了!但这里恰恰是最容易陷入误区的地方! 
两段日志的不同现象背后所隐藏的是原理是:Java语言是按值传递参数,按引用传递对象的;Java中所操作的对象其实都是操作对象的引用,object本身保存在“堆”中,而对象的“引用“保存在寄存器或“栈”中。 
伪代码描述一下newValue方法和modifyValue方法的不同之处: 
Java代码   收藏代码
  1. newValue{  
  2.     Value_ref2 = value_ref1;    // 按值传入引用value_ref1,得到value_ref1的副本  
  3.     value_obj2 = new Value();   // value_obj2被创建、初始化在“堆“中  
  4.     value_ref2 -> value_obj2;    // value_ref2 指向value_obj2  
  5. value_ref2 ->value_obj2.setValue(“xxx”); // value_obj2 的value被修改  
  6. printValueObj2();           // 此处打印的是obj2的值  
  7. }  
  8. modifyValue{  
  9.     Value_ref2 = value_ref1;    // 按值传入引用value_ref1,得到value_ref1的副本  
  10. value_ref2 ->value_obj1.setValue(“xxx”); // value_obj1 的value被修改  
  11. printValueObj1();           // 此处打印的是obj1的值  
  12. }  

够清楚了吧!value1_ref1在作为参数传入函数的时候,首先被复制了一份副本value1_ref2供函数域使用,此时这两个ref都是指向同一个value_obj; newObject函数中的代码[ value = new Value(); ] 其实是将value1_ref1指向了一个新的对象value_obj2;在这之后的set操作都是对新对象的操作;modifyValue函数是通过set方法直接操作value_obj1,这是跟newValue函数的不同之处。 

如果还是不太明白,请先确定是否已经理解 “引用”、“对象”的概念,可以Google、百度相应的文章学习一下:) 

  • 4
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 10
    评论
Java 8 引入了函数编程的概念,其中最重要的两个特性是 Lambda 表达式和 Stream API。下面将对这两个特性进行详细介绍。 1. Lambda 表达式 Lambda 表达式是一种匿名函数,它可以作为参数传递给方法或存储在变量中。Lambda 表达式的语法如下: ``` (parameter1, parameter2, ..., parameterN) -> { statement1; statement2; ... } ``` 其中,参数列表可以为空,也可以包含多个参数;箭头符号 "->" 用于将参数列表和 Lambda 表达式的主体分开;体可以是一个语句块,也可以是一个表达式。 下面是一个 Lambda 表达式的例子: ```java (int x, int y) -> { return x + y; } ``` 这个 Lambda 表达式接受两个整数参数 x 和 y,返回它们的和。 Lambda 表达式可以用于函数式接口,即只包含一个抽象方法的接口。例如,下面是一个函数式接口的定义: ```java interface MyInterface { int myMethod(int x, int y); } ``` 可以使用 Lambda 表达式来实现这个接口: ```java MyInterface obj = (int x, int y) -> { return x + y; }; ``` Lambda 表达式还可以使用方法引用来简化代码。例如,可以使用静态方法引用来实现上面的例子: ```java MyInterface obj = Integer::sum; ``` 2. Stream API Stream API 是一种用于处理集合的 API,它提供了一种流式处理集合的方式。Stream API 可以用于对集合进行过滤、映射、排序等操作。 下面是一个使用 Stream API 进行过滤和映射的例子: ```java List<String> list = Arrays.asList("apple", "banana", "orange", "pear"); List<String> result = list.stream() .filter(s -> s.startsWith("a")) .map(String::toUpperCase) .collect(Collectors.toList()); ``` 这个例子首先创建了一个包含四个字符串的列表,然后使用 stream() 方法将其转换为一个流。接着使用 filter() 方法过滤出以字母 "a" 开头的字符串,再使用 map() 方法将这些字符串转换为大写形式。最后使用 collect() 方法将结果收集到一个列表中。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值