Java--类加载器 、反射、 动态代理

类加载器
反射
动态代理
JDK1.5以及1.7的新特性

反射(类的加载概述和加载时机)

A:类的加载概述
	当程序要使用某个类时,如果该类还未被加载到内存中,
	则系统会通过加载,连接,初始化三步来实现对这个类进行初始化。
	加载 
		就是指将class文件读入内存,并为之创建一个Class对象。
		任何类被使用时系统都会建立一个Class对象。
	连接
		验证 :	是否有正确的内部结构,并和其他类协调一致
		准备 :	负责为类的静态成员分配内存,并设置默认初始化值
		解析:	把类中的符号引用转换为直接引用
	初始化		就是我们以前讲过的初始化步骤

B:类的加载时机
	创建类的实例 new Student()  
	访问类的静态变量,或者为静态变量赋值 Math.PI   Math.class
	调用类的静态方法  Math.abs()
	使用反射方式来强制创建某个类或接口对应的java.lang.Class对象
	初始化某个类的子类
	直接使用java.exe命令来运行某个主类  

反射(类加载器的概述和分类)

A:类加载器的概述
	负责将.class文件加载到内在中,并为之生成对应的Class对象。
B:类加载器的分类
	Bootstrap ClassLoader 根类加载器
	Extension ClassLoader 扩展类加载器
	Sysetm ClassLoader 系统类加载器
C:类加载器的作用
	Bootstrap ClassLoader 根类加载器
		也被称为引导类加载器,负责Java核心类的加载
		比如System,String等。在JDK中JRE的lib目录下rt.jar文件中
	Extension ClassLoader 扩展类加载器
		负责JRE的扩展目录中jar包的加载。
		在JDK中JRE的lib目录下ext目录
	Sysetm ClassLoader 系统类加载器  
		负责在JVM启动时加载来自java命令的class文件,以及classpath环境变量所指定的jar包和类路径

反射(反射概述以及获取class文件对象的三种方式)

A:反射概述
	JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;
	对于任意一个对象,都能够调用它的任意一个方法和属性;
	这种动态获取类的信息以及动态调用对象的方法的功能称为java语言的反射机制。
	要想解剖一个类,必须先要获取到该类的字节码文件对象。
	而解剖使用的就是Class类中的方法,所以先要获取到每一个字节码文件对应的Class类型的对象
B:获取class文件对象的三种方式
	a:Object类的getClass()方法
	b:静态属性class
	c:Class类中静态方法forName()
C:案例演示:	获取class文件对象的三种方式
public class MyTest {
    public static void main(String[] args) throws ClassNotFoundException {
        //方式一
        Object obj = new Object();
        Class aClass = obj.getClass();
        Class aClass1 = obj.getClass();

        Object obj2 = new Object();
        Class aClass2 = obj2.getClass();
        System.out.println(aClass==aClass1);

        System.out.println(aClass1==aClass2);

        //方式二
        Class objectClass = Object.class;
        System.out.println(aClass2==objectClass);
        Class stringClass = String.class;

        //方式三
        Class name = Class.forName("Demo4.ServerThread");
        Class aClass3 = Class.forName("java.lang.Object");

    }
}

		反射: 就是在运行状态中的一种动态调用方法或者属性的一种机制.
		- 就是获取字节码文件对象,然后剖析该类中存在哪些构造方法,哪些成员变量,哪些成员方法
- 
- 类的成员:
- 成员变量		Field
- 构造方法		Constructor
- 成员方法		Method
- 
- 如何获取一个类对应的字节码文件对象:
- 
- a: 第一种通过Object类中的getClass方法
- b: 通过静态属性(class属性)
- c: 通过Class类中的一个静态方法:
- public static Class forName(String className): 
- className: 这个表示的是一个类对应的全类名(就是需要加上包名)

反射(通过反射获取无参构造方法并使用)

A:获取所有构造方法
	public Constructor<?>[] getConstructors() 获取所有的构造方法不包含私有的
	public Constructor<?>[] getDeclaredConstructors() 获取所有的构造方法 包括私有的
B:获取单个构造方法
	public Constructor<T> getConstructor(Class<?>... parameterTypes) 获取单个的构造方法 不包含私有的
	public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes) 获取单个的构造方法包含私有的
C:案例演示:	通过反射获取无参构造方法并使用
public class MyTest {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class studentClass = Class.forName("Demo5.Student");
        Constructor constructor = studentClass.getConstructor();

        Object obj = constructor.newInstance();
        System.out.println(obj);
    }
}

反射(通过反射获取带参构造方法并使用)

A:案例演示:	通过反射获取带参构造方法并使用
public class MyTest {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class studentClass = Class.forName("Demo5.Student");
        Constructor constructor1 = studentClass.getConstructor(String.class);
        Student obj2 = (Student) constructor1.newInstance("李四");
        System.out.println(obj2);

        Constructor constructor2 = studentClass.getConstructor(String.class, int.class, char.class);
        Student student =(Student) constructor2.newInstance("王五", 20, "女");
        System.out.println(student);

    }
}

反射(通过反射获取私有构造方法并使用)

A:案例演示:	通过反射获取私有构造方法并使用
	// 获取字节码文件对象
	Class clazz = Class.forName("com.click369.Student") ;
	Constructor con = clazz.getDeclaredConstructor(String.class , int.class) ;
	// 值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。
	con.setAccessible(true) ; 取消语法检查不然会报错 因为私有的外界不能直接访问
	Object obj = con.newInstance("张三" , 23) ;
	System.out.println(obj);
public class MyTest {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class studentClass = Class.forName("Demo5.Student");
  Constructor declaredConstructor = studentClass.getDeclaredConstructor(String.class, int.class);
        declaredConstructor.setAccessible(true);
        Object student1 = declaredConstructor.newInstance("哈哈", 18);
        }
    }

反射(通过反射获取成员变量并使用)

A:获取所有成员变量
	public Field[] getFields() 获取所有的成员变量包含从父类继承过来的
	public Field[] getDeclaredFields() 获取所有的成员变量 包含私有的 也包含从父类继承过来的成员变量
B:获取单个成员变量
	public Field getField(String name)
	public Field getDeclaredField(String name)
C:案例演示:	通过反射获取成员变量并使用
public class MyTest {
    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, InstantiationException {
        Class teacherClass = Teacher.class;
        Field field = teacherClass.getField("name");
        Object obj = teacherClass.newInstance();
        field.set(obj,"一一");

        Object o = field.get(obj);
        System.out.println(o);

        Field sexFiled = teacherClass.getDeclaredField("sex");
        sexFiled.setAccessible(true);
        sexFiled.set(obj,"女");

        Object o1 = sexFiled.get(obj);
        System.out.println(o1);
    }
}

反射(通过反射获取无参无返回值成员方法并使用)

A:获取所有成员方法
	public Method[] getMethods() //获取所有的公共的成员方法不包含私有的 包含从父类继承过来的过来的公共方法
	public Method[] getDeclaredMethods()//获取自己的所有成员方法 包含私有的
B:获取单个成员方法
	//参数1: 方法名称  参数2:方法行参的class 对象
	public Method getMethod(String name,Class<?>... parameterTypes) //获取单个的方法 不包含私有的
	public Method getDeclaredMethod(String name,Class<?>... parameterTypes) 获取单个方法包括私有的
C:案例演示:	通过反射获取无参无返回值成员方法并使用
public class MyTest {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException {
        Class studentClass = Class.forName("Demo6.Student");
        Method showMethod = studentClass.getMethod("show");
        Object obj = studentClass.newInstance();
        showMethod.invoke(obj);
        
    }
}

反射(通过反射获取带参带返回值成员方法并使用)

A:案例演示:	通过反射获取带参带返回值成员方法并使用
public class MyTest4 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException {
        Class studentClass = Class.forName("Demo6.Student");
        Method show3 = studentClass.getMethod("show3", String.class, int.class, double.class);
        Integer invoke = (Integer) show3.invoke(obj, "王五", 25, 3.5);
        System.out.println(invoke);

       Method test = studentClass.getDeclaredMethod("test", String.class, int.class, double.class);
        test.setAccessible(true);
        Integer result =(Integer) test.invoke(obj, "二二", 18, 5.5);
        System.out.println(result);
    }
}

反射(通过反射运行配置文件内容)

A:案例演示:	通过反射运行配置文件内容
public class MyTest {
    public static void main(String[] args) throws IOException, ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Properties properties = new Properties();
        properties.load(new FileReader("src/peizhi.txt"));

        Class dogClass = Class.forName(properties.getProperty("className"));
        Object obj = dogClass.getDeclaredConstructor().newInstance();
        Method methodName = dogClass.getMethod(properties.getProperty("methodName"));
        methodName.invoke(obj);
    }
}

public class Dog {
    public void eat(){
        System.out.println("狗吃骨头");
    }
}

public class Cat {
    public void eat(){
        System.out.println("猫吃鱼");
    }
}
peizhi.txt
className=Demo7.Cat
methodName=eat

反射(通过反射越过泛型检查)

A:案例演示:	我给你ArrayList<Integer>的一个对象,我想在这个集合中添加一个字符串数据,如何实现呢?
public class MyTest {
    public static void main(String[] args) throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        ArrayList<Integer> list = new ArrayList<>();
        list.add(200);
        list.add(200);
        list.add(200);
        //list.add("abc");
        Class aClass = list.getClass();
        Method add = aClass.getMethod("add", Object.class);
        add.invoke(list,"abc");
        System.out.println(list);
    }
}

反射(通过反射写一个通用的设置某个对象的某个属性为指定的值)

A:案例演示
	public void setProperty(Object obj, String propertyName, Object value){},
	此方法可将obj对象中名为propertyName的属性的值设置为value。
public class MyTest {
    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
        Student student = new Student();
        MyUtils.setProperty(student,"name","张三");
        MyUtils.setProperty(student,"age",18);
        System.out.println(student.getName());
        System.out.println(student.getAge());
    }
}

public class Student {
    private String name;
    private int age;

    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }
}

public class MyUtils {
    public static void setProperty(Object obj, String propertyName, Object value) throws NoSuchFieldException, IllegalAccessException {
        Class aClass = obj.getClass();
        Field declaredField = aClass.getDeclaredField(propertyName);
        declaredField.setAccessible(true);
        declaredField.set(obj,value);
    }
}

反射(通过用户的增删改查引出中介)

A:案例演示:	用户的增删改查
public class MyTest {
    public static void main(String[] args) {
        UserDaoImpl userDao = new UserDaoImpl();
        userDao.insert();
        userDao.delete();
        userDao.update();
        userDao.query();
    }
}

public interface UserDao {
    void insert();
    void delete();
    void update();
    void query();
}

public class UserDaoImpl implements UserDao {
    @Override
    public void insert() {
        System.out.println("全权限校验");
        System.out.println("添加一个用户");
        System.out.println("记录日志");
    }

    @Override
    public void delete() {
        System.out.println("删除用户");
    }

    @Override
    public void update() {
        System.out.println("修改用户");
    }

    @Override
    public void query() {
        System.out.println("查询用户");
    }
}

反射(动态代理的概述和实现)

A:动态代理概述
	代理:本来应该自己做的事情,却请了别人来做,被请的人就是代理对象。
	举例:春季回家买票让人代买
	动态代理:在程序运行过程中产生的这个对象
	而程序运行过程中产生对象其实就是我们刚才反射讲解的内容,所以,动态代理其实就是通过反射来生成一个代理
	
	在Java中java.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口,
	通过使用这个类和接口就可以生成动态代理对象。JDK提供的代理只能针对接口做代理。
	我们有更强大的代理cglib,Proxy类中的方法创建动态代理类对象
		public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
	最终会调用InvocationHandler的方法
		InvocationHandler Object invoke(Object proxy,Method method,Object[] args)
B:案例演示:	动态代理的实现
public class MyTest {
    public static void main(String[] args) {
        UserDao userDao = new UserDaoImpl();

        UserDao proxy = ProxyUtils.getProxy(userDao);
        proxy.insert();
        System.out.println("----------------");
        proxy.delete();
        System.out.println("----------------");
        proxy.update();
        System.out.println("----------------");
        proxy.query();
    }
}

public interface UserDao {
    void insert();
    void delete();
    void update();
    void query();
}

public class ProxyUtils {
    public static UserDao getProxy(UserDao userDao){
        UserDao obj = (UserDao) Proxy.newProxyInstance(userDao.getClass().getClassLoader(), userDao.getClass().getInterfaces(), new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
              //全部
                System.out.println("权限校验");
                Object result = method.invoke(userDao);
                System.out.println("记录日志");

               //单独
                Object result=null;
                if (method.getName().equals("delete")){
                    System.out.println("权限校验");
                    result = method.invoke(userDao);
                    System.out.println("记录日志");
                }
                return result;


            }
        });
        return obj;
    }
}

public class UserDaoImpl implements UserDao {
    @Override
    public void insert() {
        System.out.println("添加一个用户");
    }

    @Override
    public void delete() {
        System.out.println("删除用户");
    }

    @Override
    public void update() {

        System.out.println("修改用户");
    }

    @Override
    public void query() {

        System.out.println("查询用户");
    }
}
我们可以通过Proxy类中的静态方法获取一个代理对象:
- 
- public static Object newProxyInstance(ClassLoader loader,  Class<?>[] interfaces,  InvocationHandler h)

loader: 			类加载器,负责加载代理对象的字节码,这个加载器,跟目标对象用的是同一个类加载器
interfaces:			接口对应的一个Class数组
InvocationHandler:	这个其实就是要代理对象所做的事情的一个类的封装,需要增强的代码

注意:JDK给我们提供的动态代理,只能对接口进行代理.


动态代理:
         *  特点:字节码随用随创建,随用随加载
         *  作用:不修改源码的基础上对方法增强
         *  分类:
         *      基于接口的动态代理
         *      基于子类的动态代理
         *  基于接口的动态代理:
         *      涉及的类:Proxy
         *      提供者:JDK官方
         *  如何创建代理对象:
         *      使用Proxy类中的newProxyInstance方法
         *  创建代理对象的要求:
         *      被代理类最少实现一个接口,如果没有则不能使用
         *  newProxyInstance方法的参数:
         *      ClassLoader:类加载器
         *          它是用于加载代理对象字节码的。和被代理对象使用相同的类加载器。固定写法。
         *      Class[]:字节码数组
         *          它是用于让代理对象和被代理对象有相同方法。固定写法。
         *      InvocationHandler:用于提供增强的代码
         *          它是让我们写如何代理。我们一般都是些一个该接口的实现类,通常情况下都是匿名内部类,但不是必须的。
         *          此接口的实现类都是谁用谁写。

new InvocationHandler() {
                    /**
                     * 作用:执行被代理对象的任何接口方法都会经过该方法
                     * 方法参数的含义
                     * @param proxy   代理对象的引用
                     * @param method  当前执行的方法
                     * @param args    当前执行方法所需的参数
                     * @return        和被代理对象方法有相同的返回值
                     */
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                    }
                }

JDK5新特性(JDK1.5的新特性回归以及自己实现枚举类)

A: JDK1.5的新特性: 自动拆装箱 , 泛型 , 增强for  , 可变参数 , 枚举
B:枚举概述:	就是一个类只能存在几个固定的对象,那么这个就是枚举.我们就可以使用这些对象可以表示一些固定的值.
			举例:一周只有7天,一年只有12个月等。
C:案例演示:	自己实现枚举类
public class MyTest {
    public static void main(String[] args) {
        Direction front = Direction.front;
        Direction behind = Direction.behind;
        Direction left = Direction.left;
        Direction right = Direction.right;
        System.out.println(front.name);
        System.out.println(behind.name);
        System.out.println(left.name);
        System.out.println(right.name);

        front.show("前");
        behind.show("后");
        left.show("左");
        right.show("右");
    }
}

public abstract class Direction {
    public static final Direction front= new Direction("前") {
        @Override
        public void show(String name) {
            System.out.println(name);
        }
    };
    public static final Direction behind= new Direction("后") {
        @Override
        public void show(String name) {
            System.out.println(name);
        }
    };
    public static final Direction left= new Direction("左") {
        @Override
        public void show(String name) {
            System.out.println(name);
        }
    };
    public static final Direction right= new Direction("右") {
        @Override
        public void show(String name) {
            System.out.println(name);
        }
    };
    String name;

    public Direction(String name) {
        this.name = name;
    }

    public abstract void show(String name);
}

JDK5新特性(通过enum实现枚举类)

A:案例演示:	通过enum实现枚举类
public class MyTest {
    public static void main(String[] args) {
        Direction front = Direction.FRONT;
        String name = front.name;
        System.out.println(name);
        Direction behind = Direction.BEHIND;
        String name1 = behind.name;
        System.out.println(name1);
        Direction left = Direction.LEFT;
        String name2 = left.name;
        System.out.println(name2);
        Direction right = Direction.RIGHT;
        String name3 = right.name;
        System.out.println(name3);
    }
}

public enum Direction {
    FRONT("前"),BEHIND("后"),LEFT("左"),RIGHT("右"),
    ;
    String name;

    Direction(String name) {
        this.name=name;
    }
}

JDK5新特性(枚举的注意事项)

A:案例演示
	定义枚举类要用关键字enum
	所有枚举类都是Enum的子类,但是不要显式的写出来
	枚举类的第一行上必须是枚举项,最后一个枚举项后的分号是可以省略的,但是如果枚举类有其他的东西,这个分号就不能省略。建议不要省略
	枚举类可以有构造器,但必须是private的,它默认的也是private的。枚举项的用法比较特殊:枚举(“”);
	枚举类也可以有抽象方法,但是枚举项必须重写该方法
	枚举在switch语句中的使用
public class MyTest {
    public static void main(String[] args) {
        Direction front = Direction.FRONT;
        switch (front){
            case FRONT:
                System.out.println("前");
                break;
            case BEHIND:
                System.out.println("后");
                break;
            case LEFT:
                System.out.println("左");
                break;
            case RIGHT:
                System.out.println("右");
                break;
        }
    }
}

public enum Direction {
    FRONT,BEHIND,LEFT,RIGHT,
    ;
}

JDK5新特性(枚举类的常见方法)

A:枚举类的常见方法
	int ordinal()  返回枚举项的序号
	int compareTo(E o) 比较两个枚举项的 返回的是两个枚举项序号的 差值
	String name() 获取枚举项的名称
	String toString()获取枚举项的名称
	<T> T valueOf(Class<T> type,String name) 用来获取指定的枚举项  参数1:枚举类对应的字节码对象 参数2 枚举项的名称
	values()  获取所有的枚举项
	此方法虽然在JDK文档中查找不到,但每个枚举类都具有该方法,它遍历枚举类的所有枚举值非常方便
B:案例演示:	枚举类的常见方法
public class MyTest {
    public static void main(String[] args) {
        System.out.println(Direction.FRONT.ordinal());
        Direction left = Direction.LEFT;
        int index = left.ordinal();
        System.out.println(index);

        String s = left.toString();
        String name = left.name();
        System.out.println(s);
        System.out.println(name);

        Direction[] values = Direction.values();
        for (Direction value : values) {
            System.out.println(value.name());
        }
    }
}

public enum Direction {
    FRONT,BEHIND,LEFT,RIGHT,
    ;
}
public static void main(String[] args) {
	
	// 测试
	Direction front = Direction.FRONT ;
	Direction behind = Direction.BEHIND;
	Direction left = Direction.LEFT ;
	Direction right = Direction.RIGHT ;
	
	System.out.println(front.ordinal());
	System.out.println(behind.ordinal());
	System.out.println(left.ordinal());
	System.out.println(right.ordinal());
	
	System.out.println("----------------------------------");

	System.out.println(front.compareTo(right));
	
	System.out.println("----------------------------------");
	
	System.out.println(front.name());
	
	System.out.println("----------------------------------");
	
	System.out.println(front.toString());
	System.out.println(front);
	
	System.out.println("----------------------------------");
	
	// <T> T valueOf(Class<T> type,String name):	用来获取指定的枚举项
	// type: 表示的是对应的枚举的字节码文件对象
	// name: 就是枚举项的名称
	Direction direction = Direction.valueOf(Direction.class, "RIGHT") ;
	System.out.println(direction);
	
	System.out.println("----------------------------------");
	
	Direction[] directions = Direction.values() ;
	
	for(Direction d : directions){
		System.out.println(d);
	}

}

JDK7新特性(JDK7的六个新特性回顾和讲解)

A:二进制字面量
	JDK7开始,终于可以用二进制来表示整数(byte,short,int和long)。
	使用二进制字面量的好处是,可以使代码更容易被理解。语法非常简单,只要在二进制数值前面加 0b或者0B
	int x = 0b110110
B:数字字面量可以出现下划线
	为了增强对数值的阅读性,如我们经常把数据用逗号分隔一样。JDK7提供了_对数据分隔。
	举例:
		int x = 100_1000;
	注意事项:
		不能出现在进制标识和数值之间
		不能出现在数值开头和结尾
		不能出现在小数点旁边
C:switch 语句可以用字符串
D:泛型简化

public class MyTest {
    public static void main(String[] args) {
        System.out.println(0b100);
        System.out.println(0100);
        System.out.println(100);
        System.out.println(0x100);

        int num=1_000_000_000;

        ArrayList<String> strings = new ArrayList<>();
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值