基础强化 单元测试、反射、注解

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可以作用于类、方法、成员变量上

解析注解:注解有啥用------一般用于替换配置文件

例子:反射案例的  ”框架类“

定义一个注解
在其中设置 类名方法名 属性(使用注解时才赋值

此时,注解修饰的是类,所以,通过类名


=>

案例:




                          
                         

 

 小结

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值