反射调用私有构造器创建对象 Class c=hungrysingleton.class; Constructor constructor=c.getDeclaredConstructor(); constructor.setAccessible(true);//入侵 hungrysingleton h1= (hungrysingleton) constructor.newInstance(); 1、获取类(Class)对象 通过forName() -> 示例:Class.forName(“PeopleImpl”) 通过getClass() -> 示例:new PeopleImpl().getClass() 直接获取.class -> 示例:PeopleImpl.class 2、类的常用方法 getName():获取类完整方法; getSuperclass():获取类的父类; newInstance():创建实例对象; getFields():获取当前类和父类的public修饰的所有属性; getDeclaredFields():获取当前类(不包含父类)的声明的所有属性; getMethod():获取当前类和父类的public修饰的所有方法; getDeclaredMethods():获取当前类(不包含父类)的声明的所有方法; 3、类方法调用 1 静态方法调用 // 核心代码(省略了抛出异常的声明) public static void main(String[] args) { Class myClass = Class.forName("example.PeopleImpl"); // 调用静态(static)方法 Method getSex = myClass.getMethod("getSex"); getSex.invoke(myClass); } 2 普通方法调用 普通非静态方法调用,需要先获取类示例,通过“newInstance()”方法获取,核心代码如下: Class myClass = Class.forName("example.PeopleImpl"); Object object = myClass.newInstance(); Method method = myClass.getMethod("sayHi",String.class); method.invoke(object,"老王"); getMethod 获取方法,可以声明需要传递的参数的类型。 3 调用私有方法 调用私有方法,必须使用“getDeclaredMethod(xx)”获取本类所有什么的方法,代码如下: Class myClass = Class.forName("example.PeopleImpl"); Object object = myClass.newInstance(); Method privSayHi = myClass.getDeclaredMethod("privSayHi"); privSayHi.setAccessible(true); // 修改访问限制 privSayHi.invoke(object); 除了“getDeclaredMethod(xx)”可以看出,调用私有方法的关键是设置 setAccessible(true) 属性,修改访问限制,这样设置之后就可以 进行调用了。 1、JDK Proxy 动态代理 JDK Proxy 是通过实现 InvocationHandler 接口来实现的,代码如下: interface Animal { void eat(); } class Dog implements Animal { @Override public void eat() { System.out.println("The dog is eating"); } } class Cat implements Animal { @Override public void eat() { System.out.println("The cat is eating"); } } // JDK 代理类 class AnimalProxy implements InvocationHandler { private Object target; // 代理对象 public Object getInstance(Object target) { this.target = target; // 取得代理对象 return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("调用前"); Object result = method.invoke(target, args); // 方法调用 System.out.println("调用后"); return result; } } public static void main(String[] args) { // JDK 动态代理调用 AnimalProxy proxy = new AnimalProxy(); Animal dogProxy = (Animal) proxy.getInstance(new Dog()); dogProxy.eat(); } 如上代码,我们实现了通过动态代理,在所有请求之前和之后打印了一个简单的信息。 注意: JDK Proxy 只能代理实现接口的类(即使是extends继承类也是不可以代理的)。 2、Cglib 动态代理 JDK 动态代理机制只能代理实现了接口的类,Cglib 是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实 现增强,但因为采用的是继承,所以不能对 final 修饰的类进行代理。 class Panda { public void eat() { System.out.println("The panda is eating"); } } class CglibProxy implements MethodInterceptor { private Object target; // 代理对象 public Object getInstance(Object target) { this.target = target; Enhancer enhancer = new Enhancer(); // 设置父类为实例类 enhancer.setSuperclass(this.target.getClass()); // 回调方法 enhancer.setCallback(this); // 创建代理对象 return enhancer.create(); } @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { System.out.println("调用前"); Object result = methodProxy.invokeSuper(o, objects); // 执行方法调用 System.out.println("调用后"); return result; } } public static void main(String[] args) { // CGLIB 动态代理调用 CglibProxy proxy = new CglibProxy(); Panda panda = (Panda)proxy.getInstance(new Panda()); panda.eat(); } JDK Proxy 的优势: 最小化依赖关系,减少依赖意味着简化开发和维护,JDK 本身的支持,更加可靠; 平滑进行 JDK 版本升级,而字节码类库通常需要进行更新以保证在新版上能够使用; Cglib 框架的优势: 可调用普通类,不需要实现接口; 高性能; 反射获取属性: Class c=o.getClass(); Field f=c.getDeclaredField(fieldname); f.setAccessible(true); Object dd=f.get(o); 用反射完成对象属性复制: /** * 复制一个类对象属性到另一个类对象中 * @param objA 需要复制的对象 * @param objB 复制到的目标对象类型 * @return 返回复制后的目标对象 */ private static void parseObj(Object objA,Object objB) throws Exception{ if (objA == null){ return; } //获取objA的类信息 Class classA = objA.getClass(); Class classB = objB.getClass(); try { //获取objA的所有字段 Field[] fieldsA = classA.getDeclaredFields(); //获取objB的所有字段 Field[] fieldsB = classB.getDeclaredFields(); if (fieldsA == null || fieldsA.length <= 0 || fieldsB == null || fieldsB.length <= 0){ return; } //生成查询map Map<String,Field> fieldMap = new HashMap<>(); for (Field field:fieldsA){ fieldMap.put(field.getName(),field); } //开始复制字段信息 for (Field fieldB : fieldsB){ //查找是否在objB的字段中存在该字段 Field fielaA = fieldMap.get(fieldB.getName()); if (fielaA != null){ fieldB.setAccessible(true); fieldB.set(objB,getFieldValue(objA,fielaA.getName())); } } } catch (IllegalStateException e) { throw new IllegalStateException("instace fail: " ,e); } } public void test(){ try{ //生成Employee对象 Employee employee = new Employee("Frank",6666.66); //生成一个Manager对象 Manager manager = new Manager(); //复制对象 parseObj(employee,manager); System.out.println(manager.getName()); System.out.println(manager.getSalary()); }catch (Exception e){ e.printStackTrace(); } }