反射与动态代理

反射可以在程序运行期间调用对象信息(public protected default private 的字段、方法、接口类、父类等等)。

为什么会使用反射?比如我现在需要调用别人写的的一个类,但是,此时,别人的代码还没有完成,我是无法直接使用的(比如类里面的某个方法不存在,我直接调用了,编辑器就会报错,即无法通过编译),这个时候我可以通过反射的机制来使用类的所有信息。

动态代理是为某一个对象提供一个代理对象以控制对真正对象的访问。

动态代理的实现方式可以用JDK本身提供的(就是基于反射机制实现的);也可以使用CGLIB动态代理(基于ASM机制实现)。

JDK代理:

  • 由JDK本身提供,更加可靠
  • 代码实现简单
  • 可以平滑升级(不要停止服务,可以直接重启而不影响业务,用户也感觉不到)

CGLIB代理:

  • 可以处理没有实现接口的类(JDK代理处理的类需要实现接口)
  • 高性能
    基于反射的动态代理

    ```java public class MyDynamicProxy { public static void main (String[] args) { HelloImpl hello = new HelloImpl(); MyInvocationHandler handler = new MyInvocationHandler(hello); // 构造代码实例 /** *为什么我们这里可以将其转化为Hello类型的对象?原因就是在newProxyInstance这个方法的第二个参数上, 我们给这个代理对象提供了一组什么接口,那么我这个代理对象就会实现了这组接口,这个时候我们当然可以将这个 代理对象强制类型转化为这组接口中的任意一个,因为这里的接口是Hello类型,所以就可以将其转化为Hello类型了。 */ Hello proxyHello = (Hello) Proxy.newProxyInstance(HelloImpl.class.getClassLoader(), HelloImpl.class.getInterfaces(), handler); // 调用代理方法 proxyHello.sayHello();//代理对象调用方法时,会先被InvocationHandler的invoke()拦截 } } interface Hello { void sayHello(); } class HelloImpl implements Hello { @Override public void sayHello() { System.out.println(“Hello World”);//真正的方法 } } class MyInvocationHandler implements InvocationHandler { private Object target; public MyInvocationHandler(Object target) {//要被包裹(比如被保护的对象)的对象 this.target = target; }

    //拦截代理对象的代理方法 @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println(“Invoking sayHello”); Object result = method.invoke(target, args);//真正的方法被调用 return result; } }


##### 基于CGLIB的动态代理
```java
public class MyDynamicProxy_CGLIB {
    public static  void main (String[] args) {
        MyCglibProxy cglibProxy = new MyCglibProxy();
        
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(HelloImpl.class);
        enhancer.setCallback(cglibProxy);//MethodInterceptor是Callback的子接口
        HelloImpl  proxyHello = (HelloImpl)enhancer.create();
        // 调用代理方法
        proxyHello.sayHello();
    }
}
class HelloImpl  {
    
    public void sayHello() {
        System.out.println("Hello World");//真正的方法
    }
}
 public class MyCglibProxy implements MethodInterceptor{

	@Override
	public Object intercept(Object o, Method method, Object[] arg2, MethodProxy methodProxy) throws Throwable {
		// TODO Auto-generated method stub
		System.out.println("调用方法之  前  干点其他事儿......");
		Object o1 = methodProxy.invokeSuper(o, arg2);
		System.out.println("调用方法之  后  干点其他事儿......");
		return o1;
	}

}

参考资源:

强引用、软引用、弱引用、幻象引用有什么区别

在Java语言中,除了基本数据类型外,其他的都是指向各类对象的对象引用;Java中根据其生命周期的长短,将引用分为4类。

  • 强引用

特点:我们平常典型编码Object obj = new Object()中的obj就是强引用。通过关键字new创建的对象所关联的引用就是强引用。 当JVM内存空间不足,JVM宁愿抛出OutOfMemoryError运行时错误(OOM),使程序异常终止,也不会靠随意回收具有强引用的“存活”对象来解决内存不足的问题。对于一个普通的对象,如果没有其他的引用关系,只要超过了引用的作用域或者显式地将相应(强)引用赋值为 null,就是可以被垃圾收集的了,具体回收时机还是要看垃圾收集策略。

  • 软引用

特点:软引用通过SoftReference类实现。 软引用的生命周期比强引用短一些。只有当 JVM 认为内存不足时,才会去试图回收软引用指向的对象:即JVM 会确保在抛出 OutOfMemoryError 之前,清理软引用指向的对象。软引用可以和一个引用队列(ReferenceQueue)联合使用,如果软引用所引用的对象被垃圾回收器回收,Java虚拟机就会把这个软引用加入到与之关联的引用队列中。后续,我们可以调用ReferenceQueue的poll()方法来检查是否有它所关心的对象被回收。如果队列为空,将返回一个null,否则该方法返回队列中前面的一个Reference对象。

应用场景:软引用通常用来实现内存敏感的缓存。如果还有空闲内存,就可以暂时保留缓存,当内存不足时清理掉,这样就保证了使用缓存的同时,不会耗尽内存。

  • 弱引用

弱引用通过WeakReference类实现。 弱引用的生命周期比软引用短。在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。由于垃圾回收器是一个优先级很低的线程,因此不一定会很快回收弱引用的对象。弱引用可以和一个引用队列(ReferenceQueue)联合使用,如果弱引用所引用的对象被垃圾回收,Java虚拟机就会把这个弱引用加入到与之关联的引用队列中。

应用场景:弱应用同样可用于内存敏感的缓存。

  • 虚引用

特点:虚引用也叫幻象引用,通过PhantomReference类来实现。无法通过虚引用访问对象的任何属性或函数。幻象引用仅仅是提供了一种确保对象被 finalize 以后,做某些事情的机制。如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收器回收。虚引用必须和引用队列 (ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象的内存之前,把这个虚引用加入到与之关联的引用队列中。 ReferenceQueue queue = new ReferenceQueue (); PhantomReference pr = new PhantomReference (object, queue); 程序可以通过判断引用队列中是否已经加入了虚引用,来了解被引用的对象是否将要被垃圾回收。如果程序发现某个虚引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之前采取一些程序行动。

应用场景:可用来跟踪对象被垃圾回收器回收的活动,当一个虚引用关联的对象被垃圾收集器回收之前会收到一条系统通知。

参考资源:极客时间–Java核心技术36讲,公众号-Java大后端

谈谈final、finally、 finalize 有什么区别

final

  • 修饰类表示该类不可以被继承
  • 修饰字段或变量表示不可以被修改(注意修饰引用变量时,引用不可以修改,但是引用指向的类是可以被修改的)
  • 修饰方法时表示不可以被重写。
        final  int i = 9;
        i = 10; //会报错,变量 i 被final修饰,不可以改变
        
        
        final List<Integer> list1 = new ArrayList<>();
        List<Integer> list2 = new ArrayList<>();
        list1 = list2; //会报错,变量 list1 被final修饰,不可以改变
        list1.add(1); //正常执行,引用不可以修改,但是引用指向的类是可以被修改的

finally

finally是用于保证代码一定会被执行,常用于异常捕获,关闭资源(不过在Java7以后推荐使用try-with-resources的方式关闭资源,资源可以自动关闭,代码更优雅)。

 //常规读取文件的方式,需要处理很多异常
 public static  void main(String[] args){
        InputStream is = null;
        OutputStream os = null;
        try{
            is = new FileInputStream("F:\\inputFile.txt");
            os = new FileOutputStream("F:\\outputFile.txt");

            byte[] b = new byte[1024];
            int len = 0;
            while((len = is.read(b)) != -1 ){
                os.write(b,0,len);
            }
        }catch (FileNotFoundException e){
            e.printStackTrace();
        }catch (IOException e){
            e.printStackTrace();
        }
        finally {
            try{
                is.close();
            }catch (IOException e){
                e.printStackTrace();
            }
            try{
                os.close();
            }catch (IOException e){
                e.printStackTrace();
            }
        }

    }

//使用try-with-resources的方式关闭资源
try( InputStream is = new FileInputStream("F:\\inputFile.txt");
             OutputStream os = new FileOutputStream("F:\\outputFile.txt") ){

            byte[] b = new byte[1024];
            int len = 0;
            while((len = is.read(b)) != -1 ){
                os.write(b,0,len);
            }
        }catch (FileNotFoundException e){
            e.printStackTrace();
        }catch (IOException e){
            e.printStackTrace();
        }

注意:几个特例中finally不会执行 ```java //1. 程序被强制退出 try{ System.exit(-1); }finally { System.out.println(“aa”); }

//2.try中有死循环 try{ while(true){ System.out.println(“ss”); } }finally { System.out.println(“aa”); }

//3.线程被杀死 ```

finalize

finalizeObject对象自带的方法,用于对象被清除前,确保资源的回收。但是自己实现这个方法进行资源回收时,会降低性能。所以并不推荐使用它,并且在 JDK 9 开始被标记为 deprecatedJava平台目前在逐步使用 java.lang.ref.Cleaner来替换掉原有的 finalize 实现。Cleaner 的实现利用了幻象引用(PhantomReference),这是一种常见的所谓 post-mortem 清理机制。

参考资源:极客时间–Java核心技术36讲