/*** 类的加载概述:* 当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载、连接、初始化三步来实现对这个类进行初始化* 加载* 就是指将class文件读入内存,并为之创建一个class对象,任何类被使用时系统都会建立一个class对象* 连接* 验证:是否有正确的内部结构,并和其他类协调一致* 准备: 负责为类的静态成员分配内存,并设置默认初始化值* 解析: 将类的二进制数据中的符号引用替换为直接引用* 初始化* 就是初始化步骤* * 加载时机* 创建类的实例* 访问类的静态变量,或者为静态变量赋值* 调用类的静态方法* 使用反射方法来强制创建某个类或接口对应的java.lang.class对象* 初始化某个类的子类* 直接使用java.exe命令来运行某个主类*/
/*** 类加载器概述:负责将.class文件加载到内存中,并为之生成对应的Class对象* 反射概述:java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法* 对于任意一个对象,都能够调用它的任意一个方法和属性* 这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制* 想要解剖一个类,必须先要获取到该类的字节码文件对象* 而解剖使用的就是Class类中的方法,所以先要获取到每一个字节码文件对应的Class类型的对象* * 三种方法:* Object类的getClass()方法,判断两个对象是否是一个字节码文件* 静态属性class,锁对象* Class类中静态方法forName(),读取配置文件*/
/*** 通过反射获取带参的构造方法* * Class类的newInstance()方法是使用该类的无参的构造函数创建对象,如果一个类没无参的构造函数* 就不能这样创建,可以调用Class类的getConstructor* (String.class,int.class)方法获取一个指定的构造函数然后在调用Construction类的newInstance* ("张三",20)方法创建对象* @param args*/public static void main(String[] args) throws Exception{Class clazz = Class.forName("day27.Person"); //加载class对象//Person p = (Person) clazz.newInstance(); //通过无参构造创建对象//System.out.println(p); Constructor c = clazz.getConstructor(String.class,int.class); //获取有参构造Person p = (Person) c.newInstance("张三",20); //通过有参构造创建对象 System.out.println(p);}
/**通过反射获取成员变量并使用* * Class.getFiled(String)方法可以获取类中的指定字段(可见的)* 如果是私有的可以用getDeclaedFiled("name")方法获取,通过set(obj,"李四")方法可以设置指定对象上该字段的值* 如果是私有的需要调用setAccessible(true)设置访问权限,用获取的指定的字段调用get(obj)可以获取指定对象中该字段的值*/public static void main(String[] args) throws Exception{Class clazz = Class.forName("day27.Person"); //加载class对象Constructor c = clazz.getConstructor(String.class,int.class); //获取有参构造Person p = (Person) c.newInstance("张三",20); //通过有参构造创建对象//Field f = clazz.getField("name"); //获取姓名字段//f.set(p, "李四"); //修改姓名值 Field f = clazz.getDeclaredField("name"); //暴力反射获取字段f.setAccessible(true); //去除私有权限f.set(p, "李四");System.out.println(p);}
/*** 通过反射获取方法并使用* * Class.getMethod(String,Class...)和getDeclaredMethod(String,Class..)方法可以获取* 类中的指定方法,调用invoke(Objct,Object...)可以调用该方法,Class.getMethod("eat") invoke(obj)* Class.getMethod("eat",int.class) invoke(obj,10)*/public static void main(String[] args) throws Exception{Class clazz = Class.forName("day27.Person"); //加载class对象Constructor c = clazz.getConstructor(String.class,int.class); //获取有参构造Person p = (Person) c.newInstance("张三",20); //通过有参构造创建对象 Method m = clazz.getMethod("eat"); //获取Person类中的eat方法 m.invoke(p);Method m2 = clazz.getMethod("eat", int.class); //获取有参的eat方法m2.invoke(p, 2);}
/*** 通过反射越过泛型检查* * ArrayList<Integer>的一个对象,在这个集合中添加一个字符串数据* 泛型只在编译期有效,在运行期会被擦除掉*/public static void main(String[] args) throws Exception{ArrayList<Integer> list = new ArrayList<>();list.add(1);list.add(22);Class clazz = Class.forName("java.util.ArrayList"); //获取字节码对象Method m = clazz.getMethod("add", Object.class); //获取add方法m.invoke(list, "zhangsna");System.out.println(list);}
public class Tool {/*** 此方法可将obj对象中名为propertyName的属性值设置为value*/public void setProperty(Object obj,String propertyName,Object value) throws Exception{Class clazz = obj.getClass(); //获取字节码对象Field f = clazz.getDeclaredField(propertyName); //暴力反射获取字段f.setAccessible(true); //去除权限 f.set(obj, value);}} ------------------------------------------------------------/*** public void setProperty(Object obj,String propertyName,Object value){},* 此方法可将obj对象中名为propertyName的属性值设置为value*/public static void main(String[] args) throws Exception {Student s = new Student("张三", 23); System.out.println(s); //Student [name=张三, age=23] Tool t = new Tool();t.setProperty(s, "name", "李四");System.out.println(s); //Student [name=李四, age=23]}
/*** 已知一个DemoClass类* 写一个Properties格式的配置文件,配置类的完整名称* 读取这个Properties配置文件,获取类的完整名称并加载这个类,用反射的方式运行run方法* @throws Exception */public static void main(String[] args) throws Exception {BufferedReader br = new BufferedReader(new FileReader("config.Properties"));Class clazz = Class.forName(br.readLine());DemoClass dc = (DemoClass) clazz.newInstance();dc.run();}
动态代理
public interface User {public void add();public void delete(); } ---------------------------------------------- public class Userimp implements User {@Overridepublic void add() {System.out.println("新增方法");}@Overridepublic void delete() {System.out.println("删除方法");}} ----------------------------------------------------------public class MyInvocatioHandler implements InvocationHandler {private Object target;public MyInvocatioHandler(Object target){this.target = target;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args)throws Throwable {System.out.println("权限校验");method.invoke(target, args); //执行被代理target对象的方法System.out.println("日志校验");return null;}}--------------------------------------------/*** @param args*/public static void main(String[] args) {Userimp ui = new Userimp();ui.add();ui.delete();System.out.println("------------------");//现在想在add和delete方法中加入权限校验和日志记录,不修改Userimp里的方法MyInvocatioHandler m = new MyInvocatioHandler(ui);User u = (User)Proxy.newProxyInstance(ui.getClass().getClassLoader(), ui.getClass().getInterfaces(), m);u.add();u.delete();}
public class Demo_Template {/*** 模板设计模式* 定义一个算法的骨架,而将具体的算法延迟到子类中来实现* 优点:再定义算法骨架的同时,可以很灵活的实现具体的算法,满足用户灵活多变的需求* 缺点:如果算法骨架有修改的话,则需要修改抽象类*/public static void main(String[] args) {Demo d = new Demo();System.out.println(d.getTime());}}abstract class GetTime{public final long getTime(){ //final 不让子类重写long start = System.currentTimeMillis();//需要测试的代码 code();long end = System.currentTimeMillis();return end-start;}public abstract void code(); //定义为abstract方法,目的让子类重写 }class Demo extends GetTime{@Overridepublic void code() {//这样测试的代码可以自己任意书写int i=0;while(i<10000){System.out.println("#");i++;}}}
枚举
public class Demo_Enum {/*** 枚举* 指将变量的值一一列出来,变量的值只限于列举出来的值的范围内(如:一周只有7天,一年只有12个月)*/public static void main(String[] args) {//demo();//demo2();//有抽象方法Week3 mon = Week3.MON;mon.show();}/*** 无参*/public static void demo(){Week mon = Week.MON;Week tue = Week.TUE;Week wed = Week.WED;}/**、* 有参*/public static void demo2(){Week2 mon = Week2.MON;System.out.println(mon.getName());}} ------------------------------------------------------------------ public class Week {public static final Week MON = new Week();public static final Week TUE = new Week();public static final Week WED = new Week();private Week(){} //私有构造,不让其他类创造本类对象 } -------------------------------------------------------------------------- public class Week2 {public static final Week2 MON = new Week2("星期一");public static final Week2 TUE = new Week2("星期二");public static final Week2 WED = new Week2("星期三");private String name;private Week2(String name){this.name = name;}public String getName() {return name;}} -------------------------------------------------------------------------- public abstract class Week3 {public static final Week3 MON = new Week3("星期一"){public void show(){System.out.println("星期一");}};public static final Week3 TUE = new Week3("星期二"){public void show(){System.out.println("星期二");}};public static final Week3 WED = new Week3("星期三"){public void show(){System.out.println("星期三");}};private String name;private Week3(String name){this.name = name;}public String getName() {return name;} public abstract void show();}
Enum
/*** Enum*/public static void main(String[] args) {//demo1();//demo2();Week3 mon = Week3.MON;mon.show(); //星期一 }public static void demo1(){Week mon = Week.MON;System.out.println(mon);//MON }public static void demo2(){Week2 mon = Week2.MON;System.out.println(mon.getName());//星期一 } --------------------------------------------------------------------- public enum Week {MON,TUE,WED; } --------------------------------------------------------------------- public enum Week2 {MON("星期一"),TUE("星期二"),WED("星期三");private String name;private Week2(String name){this.name = name;}public String getName() {return name;}}--------------------------------------------------------------------- public enum Week3 {MON("星期一"){public void show(){System.out.println("星期一");}},TUE("星期二"){public void show(){System.out.println("星期二");}},WED("星期三"){public void show(){System.out.println("星期三");}};private String name;private Week3(String name){this.name = name;}public String getName() {return name;}public abstract void show();}
/*** 枚举的一些方法*/public static void main(String[] args) {//demo();Week2 mon = Week2.valueOf(Week2.class,"MON"); // 星期一 通过字节码对象获取枚举项 Week2[] arr = Week2.values();for(Week2 week2: arr){System.out.println(week2);}}public static void demo(){Week2 mon = Week2.MON;Week2 tue = Week2.TUE;System.out.println(mon.ordinal()); //0 在枚举中的编号System.out.println(tue.ordinal()); //1 在枚举中的编号 System.out.println(mon.compareTo(tue)); //比较编号 System.out.println(mon.name()); // MON 获取实列名称 System.out.println(mon.toString());}