Java基础知识(四)

36 篇文章 0 订阅

一、值传递 & 引用传递

  • 值传递 :方法接收的是实参值的拷贝,会创建副本,对形参的修改不会影响到实参。
  • 引用传递 :方法接收的直接是实参所引用的对象在堆中的地址,不会创建副本,对形参的修改将影响到实参。

二、为什么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 实现步骤

  1. 定义一个接口及其实现类;
  2. 创建一个代理类,同样实现该接口;
  3. 将目标对象注入到代理类中,然后在代理类对应方法中调用目标类的对应方法。我们就可以通过代理类来屏蔽对目标对象的访问,并且可以在目标方法执行前后做一些自己想做的事情。

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 动态代理实现步骤

  1. 定义一个接口及其实现类;
  2. 自定义 InvocationHandler 并重写invoke方法,在 invoke 方法中我们会调用原生方法(被代理类的方法),并自定义一些处理逻辑;
  3. 通过 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

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值