参考文章:
一,类加载器
1.1 什么是类加载器,作用是什么?
一个自己编写的 Java 代码源文件,起运行的过程主要分为三个阶段。
第一个阶段: SOURCE 级别,即源码阶段,已经写完代码,还未编译运行
第二个阶段: CLASS 级别, 即字节码 阶段,是写完代码,经过编译后的。
第三个阶段:RUNTIME 级别,是写好的源码经过编译成字节码, 由类加载器调入JDK运行时内存的阶段。
类加载器的作用就是----将字节码对象调用jdk运行内存,并生成相应的字节码对象
1.2 类加载器的种类
Java 中的类加载器有三种,不同的加载器分别加载不同的文件
- BootStrap:引导类加载器:加载都是最基础的文件
- ExtClassLoader:扩展类加载器:加载都是基础的文件
- AppClassLoader:应用类加载器:三方jar包和自己编写java文件
怎么获得类加载器?(重点)
ClassLoader 字节码对象.getClassLoader();
二,注解
2.1 什么是注解
注解就是符合一定格式的语法 @xxxx
注解作用:
注释:在阅读程序时清楚----给程序员看的
注解:给jvm看的,给机器看的
注解在目前而言最主流的应用:代替配置文件
关于配置文件与注解开发的优缺点:
注解优点:开发效率高 成本低
注解缺点:耦合性大 并且不利于后期维护
2.2 jdk5 提供的注解
@Override:告知编译器此方法是覆盖父类的
@Deprecated:标注过时
@SuppressWarnings:压制警告
发现的问题:
不同的注解只能在不同的位置使用(方法上、字段上、类上)
2.3 自定义注解
- 怎样去编写一个自定义的注解
下图是一个自定义的注解 MyAnno , 注解的声明符号是 @interface
注意:如果属性的名字是value,并且注解的属性值有一个 那么在使用注解时可以省略value
注解属性类型只能是以下几种
1.基本类型
2.String
3.枚举类型
4.注解类型
5.Class类型
6.以上类型的一维数组类型
- 怎样去使用注解
写一个类 MyAnnoTest ,使用刚刚定义过的注解 MyAnno
- 怎样去解析注解-----使用反射知识
下面这段代码是对自定义的注解,进行解析(即运行核心代码部分)
介入一个概念:元注解:代表修饰注解的注解,作用:限制定义的注解的特性
@Retention
SOURCE: 注解在源码级别可见
CLASS:注解在字节码文件级别可见
RUNTIME:注解在整个运行阶段都可见
@Target
代表注解修饰的范围:类上使用,方法上使用,字段上使用
FIELD:字段上可用此注解
METHOD:方法上可以用此注解
TYPE:类/接口上可以使用此注解
Retention 的属性取值可为 SOURCE, CLASS , RUNTIME 三个级别,这三个级别其实是注解的可见范围
三,动态代理
3.1 什么是代理(中介)
目标对象/被代理对象 ------ 房主:真正的租房的方法
代理对象 ------- 黑中介:有租房子的方法(调用房主的租房的方法)
执行代理对象方法的对象 ---- 租房的人
流程:我们要租房----->中介(租房的方法)------>房主(租房的方法)
抽象:调用对象----->代理对象------>目标对象
3.2 动态代理
动态代理:不用手动编写一个代理对象,不需要一一编写与目标对象相同的方法,这个过程,在运行时 的内存中动态生成代理对象。------字节码对象级别的代理对象
动态代理的API:
在jdk的API中存在一个Proxy中存在一个生成动态代理的的方法newProxyInstance
static Object
newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
返回值:Object就是代理对象
参数:loader:代表与目标对象相同的类加载器-------目标对象.getClass().getClassLoader()
interfaces:代表与目标对象实现的所有的接口字节码对象数组
h:具体的代理的操作,InvocationHandler接口
注意:JDK的Proxy方式实现的动态代理 目标对象必须有接口 没有接口不能实现jdk版动态代理
下面是一段实现动态代理的两种方式(代码):
测试动态代理代码一:
package com.github.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import org.junit.Test;
public class ProxyTest {
@Test
public void test1(){
//获得动态的代理对象----在运行时 在内存中动态的为Target创建一个虚拟的代理对象
//objProxy是代理对象 根据参数确定到底是谁的代理对象
TargetInterface objProxy = (TargetInterface) Proxy.newProxyInstance(
Target.class.getClassLoader(), //与目标对象相同的类加载器
new Class[]{TargetInterface.class},
new InvocationHandler() {
//invoke 代表的是执行代理对象的方法
@Override
//method:代表目标对象的方法字节码对象
//args:代表目标对象的响应的方法的参数
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("目标方法前的逻辑");
//执行目标对象的方法
Object invoke = method.invoke(new Target(), args);
System.out.println("目标方法后的逻辑");
return invoke;
}
}
);
objProxy.method1();
String method2 = objProxy.method2();
System.out.println(method2);
}
}
测试代理的代码二:
package com.github.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class ProxyTest2 {
public static void main(String[] args) {
final Target target = new Target();
//动态创建代理对象
TargetInterface proxy = (TargetInterface) Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
new InvocationHandler() {
@Override
//被执行几次?------- 看代理对象调用方法几次
//代理对象调用接口相应方法 都是调用invoke
/*
* proxy:是代理对象
* method:代表的是目标方法的字节码对象
* args:代表是调用目标方法时参数
*/
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//反射知识点
Object invoke = method.invoke(target, args);//目标对象的相应方法
//retrun返回的值给代理对象
return invoke;
}
}
);
proxy.method1();//调用invoke---Method:目标对象的method1方法 args:null 返回值null
String method2 = proxy.method2();//调用invoke---Method:目标对象的method2方法 args:null 返回值method2
int method3 = proxy.method3(100);调用invoke-----Method:目标对象的method3方法 args:Object[]{100} 返回值100
System.out.println(method2);
System.out.println(method3);
}
}
目标对象部分的代码:
package com.github.proxy;
public class Target implements TargetInterface{
@Override
public void method1() {
System.out.println("method1 running...");
}
@Override
public String method2() {
System.out.println("method2 running...");
return "method2";
}
@Override
public int method3(int x) {
return x;
}
}
目标对象实现的接口部分 --接口代码:
package com.github.proxy;
public interface TargetInterface {
public void method1();
public String method2();
public int method3(int x);
}