一、值传递 & 引用传递
- 值传递 :方法接收的是实参值的拷贝,会创建副本,对形参的修改不会影响到实参。
- 引用传递 :方法接收的直接是实参所引用的对象在堆中的地址,不会创建副本,对形参的修改将影响到实参。
二、为什么Java只有值传递?
我们通过 3 个案例来分析原因:
案例 1 : 传递基本类型的实参:
public static void main(String[] args) {
int num1 = 10;
int num2 = 20;
swap(num1, num2);
System.out.println("num1 = " + num1);
System.out.println("num2 = " + num2);
}
public static void swap(int a, int b) {
int temp = a;
a = b;
b = temp;
System.out.println("a = " + a);
System.out.println("b = " + b);
}
//输出结果
a = 20
b = 10
num1 = 10
num2 = 20
//结论
一个方法不能修改一个基本数据类型的参数。
案例 2 : 传递引用类型的实参:
public static void main(String[] args) {
int[] arr = { 1, 2, 3, 4, 5 };
System.out.println(arr[0]);
change(arr);
System.out.println(arr[0]);
}
public static void change(int[] array) {
// 将数组的第一个元素变为0
array[0] = 0;
}
//输出结果
1
0
//结论
通过执行代码发现实参中的元素值改变了,难道这是用的引用传递?
其实不是的,这里传的还是值,只不过这个值是实参的地址罢了。
案例 3 : 传递引用类型的实参 (二):
public class Person {
private String name;
// 省略构造函数、Getter&Setter方法
}
public static void main(String[] args) {
Person xiaoZhang = new Person("小张");
Person xiaoLi = new Person("小李");
swap(xiaoZhang, xiaoLi);
System.out.println("xiaoZhang:" + xiaoZhang.getName());
System.out.println("xiaoLi:" + xiaoLi.getName());
}
public static void swap(Person person1, Person person2) {
Person temp = person1;
person1 = person2;
person2 = temp;
System.out.println("person1:" + person1.getName());
System.out.println("person2:" + person2.getName());
}
//打印结果
person1:小李
person2:小张
xiaoZhang:小张
xiaoLi:小李
//结论
swap方法中对两个引用类型进行互换,并没有影响到实参,
因为 person1 和 person2 互换的只是两个地址罢了。
三、Java代理模式详解
代理设计模式就是:使用代理对象来代替对真实对象的访问,以达到在不修改原真实对象的情况下,提供额外的功能操作,扩展目标对象的功能,比如说在原真实对象中某个方法执行前后,增加一些自定义的操作。
代理模式分为:静态代理、动态代理。
3.1 静态代理
静态代理中,对原实际对象中每个方法的功能扩展都是需要手动完成的,非常的不灵活 (比如:接口一单新增家方法,目标对象和代理对象都需要进行修改),且麻烦 (需要对每个目标类,都单独写一个代理类)。因此静态代理的实际应用非常少。
以上是从实现和应用的角度来说的,而从JVM的层面来讲:静态代理在编译时就将 接口、实现类、代理类 这些都变成了一个个实际的 class 文件。
3.1.1 实现步骤
- 定义一个接口及其实现类;
- 创建一个代理类,同样实现该接口;
- 将目标对象注入到代理类中,然后在代理类对应方法中调用目标类的对应方法。我们就可以通过代理类来屏蔽对目标对象的访问,并且可以在目标方法执行前后做一些自己想做的事情。
3.1.2 实际应用
// 定义接口
public interface SmsService {
String send(String message);
}
// 用于发送短信的实现类
public class SmsServiceImpl implements SmsService {
public String send(String message) {
System.out.println("send message:" + message);
return message;
}
}
// 代理类
public class SmsProxy implements SmsService {
private final SmsService smsService;
public SmsProxy(SmsService smsService) {
this.smsService = smsService;
}
@Override
public String send(String message) {
//调用方法之前,我们可以添加自己的操作
System.out.println("before method send()");
smsService.send(message);
//调用方法之后,我们同样可以添加自己的操作
System.out.println("after method send()");
return null;
}
}
// 实际调用执行
public class Main {
public static void main(String[] args) {
SmsService smsService = new SmsServiceImpl();
SmsProxy smsProxy = new SmsProxy(smsService);
smsProxy.send("java");
}
}
// 打印输出结果
before method send()
send message:java
after method send()
3.2 动态代理
与静态代理相比,动态代理更加灵活。我们不需要针对每个目标类都单独创建一个代理类,并且也不需要必须实现接口,我们可以直接代理实现类( CGLIB 动态代理机制)。
从 JVM 的角度来说,动态代理是在运行时动态的生成 class 类字节码,并加载到 JVM 中的。
动态代理在日常开发中使用的比较少,但是在一些框架中则是必用的技术。
Java中动态代理的两种实现方式:JDK动态代理、CGLIB动态代理。
3.2.1 JDK 动态代理实现步骤
- 定义一个接口及其实现类;
- 自定义
InvocationHandler
并重写invoke
方法,在invoke
方法中我们会调用原生方法(被代理类的方法),并自定义一些处理逻辑; - 通过
Proxy.newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
方法创建代理对象;
// 定义发送短信的接口
public interface SmsService {
String send(String message);
}
//定义实现类
public class SmsServiceImpl implements SmsService {
public String send(String message) {
System.out.println("send message:" + message);
return message;
}
}
//定义一个JDK动态代理类
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class DebugInvocationHandler implements InvocationHandler {
/**
* 代理类中的真实对象
*/
private final Object target;
public DebugInvocationHandler(Object target) {
this.target = target;
}
// invoke() 方法: 当我们的动态代理对象调用被代理对象的原生方法的时候,最终实际上
// 调用到的是 invoke() 方法,然后 invoke() 方法代替我们去调用了被代理对象的原生方法。
public Object invoke(Object proxy, Method method, Object[] args) throws
InvocationTargetException, IllegalAccessException {
//调用方法之前,我们可以添加自己的操作
System.out.println("before method " + method.getName());
Object result = method.invoke(target, args);
//调用方法之后,我们同样可以添加自己的操作
System.out.println("after method " + method.getName());
return result;
}
}
//获取代理对象的工厂类
public class JdkProxyFactory {
public static Object getProxy(Object target) {
//通过Proxy.newProxyInstance()方法获取某个类的代理对象
return Proxy.newProxyInstance(
target.getClass().getClassLoader(), // 目标类的类加载
target.getClass().getInterfaces(), // 代理需要实现的接口,可指定多个
new DebugInvocationHandler(target) // 代理对象对应的自定义 InvocationHandler
);
}
}
//实际使用
SmsService smsService = (SmsService) JdkProxyFactory.getProxy(new SmsServiceImpl());
smsService.send("java");
//代码执行结果
before method send
send message:java
after method send