Junit
单元测试
Junit使用白盒测试
步骤
Junit_@Before&@After
反射
如IDEA,当创建String 对象str时,就会将String类加载进内存,生成Class类对象,通过返回Method方法对象中的方法名,当
输出str. 时的自动提示,其中的方法名就是反射机制返回的方法名
类文件Person.java ------ javac编译,得到字节码文件Person.class(成员变量、成员方法、构造方法等)---在硬盘中存储
通过类加载器(ClassLoader对象)将字节码文件加载进内存,加载进内存后,得到Class类对象(描述字节码文件共同的属性)
Class类对象中,成员变量、成员方法、构造方法都分别封装成了Field、Method、Constructor对象(都通过数组来描述)
创建对象(通过Class类对象创建类对象)
方式一,使用类的class属性:
Class<java.util.Date> clz1 = java.util.Date.class; 使用较多
方式二,通过Class类中的静态方法forName(String className),传入类的全限定名(必须添加完整包名)
Class<?> clz2 = Class.forName(“java.util.Date”);
方式三,通过对象的getClass方法来实现,其中,getClass()是Object类中的方法,所有的对象都可以调用该方法
java.util.Date str = new java.util.Date();
Class<?> clz3 = str.getClass();
第一种方式和第二种方式都是直接根据类来获取字节码对象,相比之下,第一种方式更加安全,因为在编译时期就可以检查要访问的Class对象是否存在,同时不用调用方法,性能也更好,因此用第一种方式比较多。但是如果只能获得一个字符串,如”java.lang.String”,就只能用第二种方式,第二种方式可能会抛出ClassNotFoundException异常。
注意:同一个类在JVM中只存在一份字节码对象,也就说上述,claz1 == clz2 == clz3;
基本数据类型不能表示为对象,也就不能使用getClass的方式,基本类型没有类名的概念,也不能使用Class.forName的方式
所有的数据类型都有class属性.因此可以用class属性来表示:
Class clz = 数据类型.class;
获取Class对象
第一个阶段,只有字节码文件,没有进内存,需要通过类加载器加载进内存,生成字节码文件对象--Class.forName("全类名")
第二个阶段,Class对象已经有了,字节码文件已经加载进内存,只需要获取Class对象----类名.Class(类名的属性获取)
第三个阶段,已经有类对象了,通过方法获取----对象.getClass()
Class对象的功能
获取功能--------有参,传递的是参数类型.class : (String.class,int.class)
获取成员变量、构造方法、成员方法,需要注意:
调用私有构造器时,用setAccessible(true)
有参,传递的是参数类型.class : (String.class,int.class)
获取成员变量的方法
.getFields()、getField(String name)、getDeclaredFields()、getDeclaredField(String name)
getField()和getFields()方法,只能获取public的成员变量,就算加了setAccessible(true)也不能暴力反射
获取私有的成员变量,需要使用1、getDeclaredFields(), 和2、getDeclaredField("d")+setAccessible(true)
Class对象获取构造器的方法 .getConstructors()、getConstructor(Class<>... parameterTypers)
、getDeclaredConstructor(Class<>... parameterTypers)、getDeclaredConstructors()
获取的构造器,用newInstance方法创建对象
有参,传递的是参数类型.class : (String.class,int.class)
Class对象获取成员方法的方法:getMethods()、getConstructor(String name,Class<>... parameterTypers)、
getDeclaredConstructors()、getDeclaredConstructor(CString name,lass<>... parameterTypers)
反射案例:
写一个"框架",不能改变该类的任何代码的前提下,可以帮我们创建任意类的对象,并且执行其中任意方法
通过配置文件,修改其中类名,执行不同的类
新建配置文件
新建一个properties结尾的配置文件(存入需要创建的对象的全类名和需要执行的方法定义)
1、加载配置文件
properties是Map子类,双列集合
1、通过类名.class 获取Class对象,然后通过 . getClassLoader获取该字节码文件对应的类加载器
2、通过类加载器的 .getResourceAsStream(“配置文件名”)获取对应的字节输入流(类加载器去找配置文件)
3、将获取的流,传递给Properties对象的 .load 方法,将配置文件写入到内存
自定义的类都是系统类加载器加载
类名.
2、获取配置文件的内容(类全名、方法)
获取配置文件中的数据----Properties对象的 getProperty(“全类名 / 方法名”)
3、使用反射技术,加载类文件进内存,调用方法
1、通过获取配置文件中的全类名,Class.forName(全类名),将该类加载进内存,返回Class对象
2、通过返回的Class对象 .newInstance()创建对象,.getMethod(配置文件中的方法名)获取方法对象
3、 .invoke(创建的对象)执行方法
运行结果==》
修改配置文件中的类名,和方法
====》结果
注解
|
@Override 监测方法是否继承(接口)父类
@Deprecated 注解的方法已经过时
@SuppressWarnings 注解的方法、类、成员变量 压制警告 通常 @SuppressWarnings(“all”)
自定义注解
将自定义注解的文件 javac javap 反编译,得到本质 为一个接口---默认继承Annotation接口
属性其实是抽象方法
定义属性,使用时赋值
元注解 -----修饰注解的注解
@MyAnno3可以作用于类、方法、成员变量上
解析注解:注解有啥用------一般用于替换配置文件
例子:反射案例的 ”框架类“
定义一个注解
在其中设置 类名、方法名 属性(使用注解时才赋值)
此时,注解修饰的是类,所以,通过类名
=>
案例:
小结