一、Junit單元測試
1、測試分類:
1)黑盒測試:不需要寫代碼,給輸入值,看程序是否能夠輸出期望的值。
2)白盒測試:需要寫代碼的。關注程序具體的執行流程。
2、Junit使用:白盒測試
操作步驟:
1)定義一個測試類(測試用例)
測試類名:被測試的類名Test CalculatorTest
包名:xxx.xxx.xx.test cn.itcast.test
2)定義測試方法:可以獨立運行
方法名:test測試的方法名 testAdd()
返回值:void
參數列表:空參
3)給方法加@Test
4)導入junit依賴環境
5)判定結果:
紅色:失敗。綠色:成功。一般我們會使用斷言操作來處理結果。Assert.assertEquals(期望的結果,運算的結果)。
6)@Before代表修飾的方法會在測試方法之前被自動執行。
@After代表 修飾的方法會在測試方法執行之後自動被執行。
二、反射:框架設計的靈魂
類的方法進行總結,寫個類出來一堆方法可以進行選擇。
第一個階段在硬盤,第二個階段在內存
-
框架:半成品軟件。可以在框架的基礎上進行軟件開發,簡化編碼
-
反射:將類的各個組成部分封裝爲其他對象,這就是反射機制
- 好處:
- 可以在程序運行過程中,操作這些對象。
- 可以解耦,提高程序的可擴展性。
- 好處:
-
獲取Class對象的方式:(三個階段)
- Class.forName(“全類名”):將字節碼文件加載進內存,返回Class對象
- 多用於配置文件,將類名定義在配置文件中。讀取文件,加載類
- 類名.class:通過類名的屬性class獲取
- 多用於參數的傳遞
- 對象.getClass():getClass()方法在Object類中定義着。
- 多用於對象的獲取字節碼的方式
- 結論:
同一個字節碼文件(*.class)在一次程序運行過程中,只會被加載一次,不論通過哪一種方式獲取的Class對象都是同一個。
- Class.forName(“全類名”):將字節碼文件加載進內存,返回Class對象
-
Class對象功能:
- 獲取功能:
-
獲取成員變量們
-
Field[] getFields() :獲取所有public修飾的成員變量
-
Field getField(String name) 獲取指定名稱的 public修飾的成員變量
-
Field[] getDeclaredFields() 獲取所有的成員變量,不考慮修飾符
-
Field getDeclaredField(String name)
-
-
獲取構造方法們
-
Constructor<?>[] getConstructors()
-
Constructor getConstructor(類<?>… parameterTypes)
-
Constructor getDeclaredConstructor(類<?>… parameterTypes)
-
Constructor<?>[] getDeclaredConstructors()
-
-
獲取成員方法們:
-
Method[] getMethods()
-
Method getMethod(String name, 類<?>… parameterTypes)
-
Method[] getDeclaredMethods()
-
Method getDeclaredMethod(String name, 類<?>… parameterTypes)
-
-
獲取全類名
- String getName()
-
- 獲取功能:
-
Field:成員變量
- 操作:
-
設置值
- void set(Object obj, Object value)
-
獲取值
- get(Object obj)
-
忽略訪問權限修飾符的安全檢查
- setAccessible(true):暴力反射
-
- 操作:
-
Constructor:構造方法
- 創建對象:
-
T newInstance(Object… initargs)
-
如果使用空參數構造方法創建對象,操作可以簡化:Class對象的newInstance方法
-
- 創建對象:
-
Method:方法對象
-
執行方法:
- Object invoke(Object obj, Object… args)
-
獲取方法名稱:
- String getName:獲取方法名
-
-
案例:
- 需求:寫一個"框架",不能改變該類的任何代碼的前提下,可以幫我們創建任意類的對象,並且執行其中任意方法
- 實現:
- 配置文件
- 反射
- 步驟:
- 將需要創建的對象的全類名和需要執行的方法定義在配置文件中
- 在程序中加載讀取配置文件
- 使用反射技術來加載類文件進內存
- 創建對象
- 執行方法
- 實現:
- 需求:寫一個"框架",不能改變該類的任何代碼的前提下,可以幫我們創建任意類的對象,並且執行其中任意方法
/**
* 框架類
*/
public class ReflectTest {
public static void main(String[] args) throws Exception {
//可以創建任意類的對象,可以執行任意方法
/*
前提:不能改變該類的任何代碼。可以創建任意類的對象,可以執行任意方法
*/
/* Person p = new Person();
p.eat();*/
/*
Student stu = new Student();
stu.sleep();*/
//1.加載配置文件
//1.1創建Properties對象
Properties pro = new Properties();
//1.2加載配置文件,轉換爲一個集合
//1.2.1獲取class目錄下的配置文件
ClassLoader classLoader = ReflectTest.class.getClassLoader();
InputStream is = classLoader.getResourceAsStream("pro.properties");
pro.load(is);
//2.獲取配置文件中定義的數據
String className = pro.getProperty("className");
String methodName = pro.getProperty("methodName");
//3.加載該類進內存
Class cls = Class.forName(className);
//4.創建對象
Object obj = cls.newInstance();
//5.獲取方法對象
Method method = cls.getMethod(methodName);
//6.執行方法
method.invoke(obj);
}
}
三、註解:(可以代替配置文件)
-
概念:說明程序的。給計算機看的
-
註釋:用文字描述程序的。給程序員看的
-
定義:註解(Annotation),也叫元數據。一種代碼級別的說明。它是JDK1.5及以後版本引入的一個特性,與類、接口、枚舉是在同一個層次。它可以聲明在包、類、字段、方法、局部變量、方法參數等的前面,用來對這些元素進行說明,註釋。
-
概念描述:
- JDK1.5之後的新特性
- 說明程序的
- 使用註解:@註解名稱
-
作用分類:
①編寫文檔:通過代碼裏標識的註解生成文檔【生成文檔doc文檔】 javadoc.文件名
②代碼分析:通過代碼裏標識的註解對代碼進行分析【使用反射】
③編譯檢查:通過代碼裏標識的註解讓編譯器能夠實現基本的編譯檢查【Override】 -
JDK中預定義的一些註解
- @Override :檢測被該註解標註的方法是否是繼承自父類(接口)的
- @Deprecated:該註解標註的內容,表示已過時
- @SuppressWarnings:壓制警告
- 一般傳遞參數all @SuppressWarnings(“all”)
-
自定義註解
-
格式:
元註解
public @interface 註解名稱{
屬性列表;
} -
本質:註解本質上就是一個接口,該接口默認繼承Annotation接口
- public interface MyAnno extends java.lang.annotation.Annotation {}
-
屬性:接口中的抽象方法
- 要求:
-
屬性的返回值類型有下列取值
- 基本數據類型
- String
- 枚舉(類)
- 註解
- 以上類型的數組
-
定義了屬性,在使用時需要給屬性賦值
- 如果定義屬性時,使用default關鍵字給屬性默認初始化值,則使用註解時,可以不進行屬性的賦值。
- 如果只有一個屬性需要賦值,並且屬性的名稱是value,則value可以省略,直接定義值即可。
- 數組賦值時,值使用{}包裹。如果數組中只有一個值,則{}可以省略
-
- 要求:
-
元註解:用於描述註解的註解
- @Target:描述註解能夠作用的位置
- ElementType取值:
- TYPE:可以作用於類上
- METHOD:可以作用於方法上
- FIELD:可以作用於成員變量上
- ElementType取值:
- @Retention:描述註解被保留的階段
- @Retention(RetentionPolicy.RUNTIME):當前被描述的註解,會保留到class字節碼文件中,並被JVM讀取到
- @Documented:描述註解是否被抽取到api文檔中
- @Inherited:描述註解是否被子類繼承
- @Target:描述註解能夠作用的位置
-
-
在程序使用(解析)註解:獲取註解中定義的屬性值
- 獲取註解定義的位置的對象 (Class,Method,Field)
- 獲取指定的註解
-
getAnnotation(Class)
//其實就是在內存中生成了一個該註解接口的子類實現對象public class ProImpl implements Pro{ public String className(){ return "cn.itcast.annotation.Demo1"; } public String methodName(){ return "show"; } }
-
- 調用註解中的抽象方法獲取配置的屬性值
案例:簡單的測試框架
/**
* 小明定義的計算器類
*/
public class Calculator {
//加法
@Check
public void add(){
String str = null;
str.toString();
System.out.println("1 + 0 =" + (1 + 0));
}
//減法
@Check
public void sub(){
System.out.println("1 - 0 =" + (1 - 0));
}
//乘法
@Check
public void mul(){
System.out.println("1 * 0 =" + (1 * 0));
}
//除法
@Check
public void div(){
System.out.println("1 / 0 =" + (1 / 0));
}
public void show(){
System.out.println("永無bug...");
}
}
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Check {
}
/**
* 簡單的測試框架
*
* 當主方法執行後,會自動自行被檢測的所有方法(加了Check註解的方法),判斷方法是否有異常,記錄到文件中
*/
public class TestCheck {
public static void main(String[] args) throws IOException {
//1.創建計算器對象
Calculator c = new Calculator();
//2.獲取字節碼文件對象
Class cls = c.getClass();
//3.獲取所有方法
Method[] methods = cls.getMethods();
int number = 0;//出現異常的次數
BufferedWriter bw = new BufferedWriter(new FileWriter("bug.txt"));
for (Method method : methods) {
//4.判斷方法上是否有Check註解
if(method.isAnnotationPresent(Check.class)){
//5.有,執行
try {
method.invoke(c);
} catch (Exception e) {
//6.捕獲異常
//記錄到文件中
number ++;
bw.write(method.getName()+ " 方法出異常了");
bw.newLine();
bw.write("異常的名稱:" + e.getCause().getClass().getSimpleName());
bw.newLine();
bw.write("異常的原因:"+e.getCause().getMessage());
bw.newLine();
bw.write("--------------------------");
bw.newLine();
}
}
}
bw.write("本次測試一共出現 "+number+" 次異常");
bw.flush();
bw.close();
}
}