亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

文章標題

2019-11-10 20:02:52
字體:
來源:轉載
供稿:網友

先介紹下這篇博文的由來,之前已經對JUnit的使用經行了深入的介紹和演示(參考JUnit學習(一),JUnit學習(二)),其中的部分功能是通過分析JUnit源代碼找到的。得益于這個過程有幸完整的拜讀了JUnit的源碼十分贊嘆作者代碼的精美,一直計劃著把源碼的分析也寫出來。突發奇想決定從設計模式入手賞析JUnit的流程和模式的應用,希望由此能寫出一篇耐讀好看的文章。于是又花了些時日重讀《設計模式》以期能夠順暢的把兩者結合在一起,由于個人水平有限難免出現錯誤、疏漏,還請各位高手多多指出、討論。

測試用例的一生——運行流程圖

這里寫圖片描述

圖一 TestCase運行時序圖

首先,介紹下JUnit的測試用例運行會經過哪些過程,這里說起來有些抽象會讓人比較迷惑,在看了后面章節的內容之后就比較清晰了:

Client(JUnitCore、Eclipse):這里是TestCase開始的地方,如果是Main函數來啟動測試用例的話一般會調用JUnitCore的方法,如果是在Eclipse中使用JUnit插件則實際上調用的是org.eclipse.jdt.internal.junit4.runner.JUnit4TestClassReference這個類。 Request(org.junit.runner.Request):Client會根據測試類或方法而創建一個Request實體,Request包含了要被執行的測試類、測試方法等信息。 RunnerBuilder:在創建后Client調用Request.getRunner()方法獲取用于執行測試的Runner,該過程是由RunnerBuilder這個工廠類完成的。 RunNotifier:在執行Runner.run方法時Client還會傳遞一個RunNotifier對象,是事件的監聽器的管理員。Runner在開始執行、成功、失敗和執行結束時會調用RunNotifier中相應的方法從而發送事件給注冊了的監聽器,JUnit運行的最終結果就是這些監聽器收集展現的。 Runner:從名字可以猜出這里應該是測試用例運行的地方了,在Client調用了Runner.run()方法之后,Runner會先構造Statement對象將所有要執行的邏輯委托給它,接著執行Statement.evaluate()方法。在這期間Runner會觸發RunNotifier(如測試開始、測試結束等等)。 Statement:測試用例的運行時描述,關注于測試用例如何運行和調用測試代碼。比如在執行測試用例前是否有@Before注釋的方法需要調用等信息都會被Runner構造為Statement。

JUnit的類設計和模式應用

這里寫圖片描述

圖二 JUnit的類結構

首先先介紹下JUnit中的模型類(Model),在JUnit模型類可以劃分為三個范圍:

描述模型:是對要執行的測試用例的描述(比如要執行哪個類的哪個方法,是否有指定Runner,使用了哪些注解等),這一層類似于流程文件之于流程引擎——不是用來執行的,而是描述要有哪些環節、細節。個人認為這一模型包括測試類本身(即你自己編寫的測試用例)和Request。其中測試類本身等同于描述文件,Request則記錄了是要運行的Suit、測試類或者是某個具體的方法、過濾器、排序的Comparator等信息(JUnit是支持對測試方法排序和過濾的)。運行時模型:是JUnit中可執行的模型,包括FrameworkMember(org.junit.runners.model.FrameworkMember)及其子類、TestClass(org.junit.runners.model.TestClass)、Statement。FrameworkMember的子類包括FrameworkMethod和FrameworkField分別描述了測試類的方法和變量信息,比如是否為靜態、作用域、包含哪些注解等JUnit運行時需要用到的信息;TestClass的作用有些類似FrameworkMember,是針對測試的Class的描述。Statement在上面已經介紹過是對測試執行流程和細節的描述。

結果模型:JUnit中用于描述用例的類,包括Description(org.junit.runner.Description)、Result(org.junit.runner.Result)、Failure(org.junit.runner.notification.Failure)。Description是對測試用例的描述(測試名稱、所在Class的名字、是否是suit等等)只為RunNotifier提供服務。Result是運行結果的描述,用例執行完成后RunNotifier的 fireTestRunFinished(final Result result)方法會被觸發,傳入的Result實例描述了運行耗時、忽略的用例次數、是否成功等信息。Failure則是用例失敗后Runner傳遞給RunNotifier的對象用于描述錯誤信息,特別包含了錯誤的StackTrace。 言歸正傳,下面討論設計模式和JUnit的源碼:

工廠方法模式、職責鏈:用例啟動,Client在創建Request后會調用RunnerBuilder(工廠方法的抽象類)來創建Runner,默認的實現是AllDefaultPosibilitiesBuilder,根據不同的測試類定義(@RunWith的信息)返回Runner。AllDefaultPosibilitiesBuilder使用職責鏈模式來創建Runner,部分代碼如下。代碼A是AllDefaultPosibilitiesBuilder的主要構造邏輯構造了一個【IgnoreBuilder->AnnotatedBuilder->SuitMethodBuilder->JUnit3Builder->JUnit4Builder】的職責鏈,構造Runner的過程中有且只有一個handler會響應請求。代碼B是Junit4Builder類實現會返回一個BlockJUnit4ClassRunner對象,這個是JUnit4的默認Runner。public Runner runnerForClass(Class<?> testClass) throws Throwable { List<RunnerBuilder> builders = Arrays.asList( ignoredBuilder(), annotatedBuilder(), suiteMethodBuilder(), junit3Builder(), junit4Builder()); for (RunnerBuilder each : builders) { Runner runner = each.safeRunnerForClass(testClass); if (runner != null) { return runner; } } return null; }

代碼A 職責鏈實現

public class JUnit4Builder extends RunnerBuilder { @Override public Runner runnerForClass(Class<?> testClass) throws Throwable { return new BlockJUnit4ClassRunner(testClass); }}

代碼B JUnit4Builder實現

組合模式:將具備樹形結構的數據抽象出公共的接口,在遍歷的過程中應用同樣的處理方式。這個模式在Runner中的應用不是很明顯,扣進來略有牽強。Runner是分層次的,父層包括@BeforeClass、@AfterClass、@ClassRule注解修飾的方法或變量,它們在測試執行前或執行后執行一次。兒子層是@Before、@After、@Rule修飾的方法或變量它們在每個測試方法執行前后執行。當編寫的用例使用Suit來運行時則是三層結構,上面的父子結構中間插入了一層childrenRunners,也就是一個Suilt中每個測試類都會生成一個Runner,調用順序變成了Runner.run()>childRunner.run()<即遍歷childrenRunners>->testMethod()。ParentRunner中將變化的部分封裝為runChild()方法交給子類實現,達到了遍歷過程使用同樣處理方式的目的——ParentRunner.this.runChild(each,notifier)。public void run(final RunNotifier notifier) { EachTestNotifier testNotifier = new EachTestNotifier(notifier, getDescription()); try { Statement statement = classBlock(notifier); statement.evaluate(); } catch (AssumptionViolatedException e) { testNotifier.fireTestIgnored(); } catch (StoppedByUserException e) { throw e; } catch (Throwable e) { testNotifier.addFailure(e); } } PRivate void runChildren(final RunNotifier notifier) { for (final T each : getFilteredChildren()) { fScheduler.schedule(new Runnable() { public void run() { ParentRunner.this.runChild(each, notifier); } }); } fScheduler.finished(); }

代碼C ParentRunner的組合模式應用

模板方法模式:模板方法的目的是抽取公共部分封裝變化,在父類中會包含公共流程的代碼,將變化的部分封裝為抽象方法由子類實現(就像模板一樣框架式定好的,你去填寫你需要的內容就行了)。JUnit的默認Runner——BlockJUnit4ClassRunner繼承自ParentRunner,ParentRunner類定義了Statement的構造和執行流程,而如何執行兒子層的runChild方法時交給子類實現的,在BlockJUnit4ClassRunner中就是去構造和運行TestMethod,而另一個子類Suit中則是執行子層次的runner.run。

這里寫圖片描述 圖三 RunNotifier的公共方法

private abstract class SafeNotifier { private final List<RunListener> fCurrentListeners; SafeNotifier() { this(fListeners); } SafeNotifier(List<RunListener> currentListeners) { fCurrentListeners = currentListeners; } void run() { synchronized (fListeners) { List<RunListener> safeListeners = new ArrayList<RunListener>(); List<Failure> failures = new ArrayList<Failure>(); for (Iterator<RunListener> all = fCurrentListeners.iterator(); all .hasNext(); ) { try { RunListener listener = all.next(); notifyListener(listener); safeListeners.add(listener); } catch (Exception e) { failures.add(new Failure(Description.TEST_MECHANISM, e)); } } fireTestFailures(safeListeners, failures); } } abstract protected void notifyListener(RunListener each) throws Exception; }public void fireTestRunStarted(final Description description) { new SafeNotifier() { @Override protected void notifyListener(RunListener each) throws Exception { each.testRunStarted(description); } ; }.run(); }

代碼D RunNotifier代碼截取

裝飾模式:保持對象原有的接口不改變而透明的增加對象的行為,看起來像是在原有對象外面包裝了一層(或多層)行為——雖然對象還是原來的類型但是行為逐漸豐富起來。 之前一直在強調Statement描述了測試類的執行細節,到底是如何描述的呢?代碼E展示了Statement的構筑過程,首先是調用childrenInvoker方法構建了Statement的基本行為——執行所有的子測試runChildren(notifier)(非Suit情況下就是TestMethod了,如果是Suit的話則是childrenRunners)。接著是裝飾模式的應用,代碼F是withBeforeClasses()的實現——很簡單,檢查是否使用了@BeforeClasses注解修飾如果存在構造RunBefores對象——RunBefore繼承自Statement。代碼H中的evaluate()方法可以發現新生成的Statement在執行runChildren(fNext.evaluate())之前遍歷所有使用@BeforeClasses注解修飾的方法并執行。產生的效果即使用@BeforeClasses修飾的方法會在所有用例運行前執行且只執行一次。后面的withAfterClasses、withClassRules方法原理一樣都使用了裝飾模式,不再贅述。protected Statement classBlock(final RunNotifier notifier) { Statement statement = childrenInvoker(notifier); statement = withBeforeClasses(statement); statement = withAfterClasses(statement); statement = withClassRules(statement); return statement; } protected Statement childrenInvoker(final RunNotifier notifier) { return new Statement() { @Override public void evaluate() { runChildren(notifier); } }; }

代碼E Statement的構造

protected Statement withBeforeClasses(Statement statement) { List<FrameworkMethod> befores = fTestClass .getAnnotatedMethods(BeforeClass.class); return befores.isEmpty() ? statement : new RunBefores(statement, befores, null); }

代碼F withBeforeClasses實現

public class RunBefores extends Statement { private final Statement fNext; private final Object fTarget; private final List<FrameworkMethod> fBefores; public RunBefores(Statement next, List<FrameworkMethod> befores, Object target) { fNext = next; fBefores = befores; fTarget = target; } @Override public void evaluate() throws Throwable { for (FrameworkMethod before : fBefores) { before.invokeExplosively(fTarget); } fNext.evaluate(); }}

代碼H RunBefores實現

策略模式:針對相同的行為在不同場景下算法不同的情況,抽象出接口類,在子類中實現不同的算法并提供算法執行必須Context信息。JUnit中提供了Timeout、ExpectedException、ExternalResource等一系列的TestRule用于豐富測試用例的行為,這些TestRule的都是通過修飾Statement實現的。修飾Statement的代碼在withRules()方法中實現,使用了策略模式。代碼I描述了JUnit是如何處理@Rule標簽的,withRules方法獲取到測試類中所有的@Rule修飾的變量,分別調用withMethodRules和withTestRules方法,前者是為了兼容JUnit3版本的Rule這里忽略,后者withTestRules的邏輯很簡單首先查看是否使用了@Rule,如存在就交給RunRules類處理。代碼J是RunRules的實現,在構造函數中處理了修飾Statement的邏輯(applyAll方法)——抽象接口是TestRule,根據不同的場景(即使用@Rule修飾的不同的TestRule的實現)選擇不同的策略(TestRule的具體實現),而Context信息就是入參(result:Statement, description:Description)。private Statement withRules(FrameworkMethod method, Object target, Statement statement) { List<TestRule> testRules = getTestRules(target); Statement result = statement; result = withMethodRules(method, testRules, target, result); result = withTestRules(method, testRules, result); return result; }private Statement withTestRules(FrameworkMethod method, List<TestRule> testRules, Statement statement) { return testRules.isEmpty() ? statement : new RunRules(statement, testRules, describeChild(method)); }

代碼I rule的執行

public class RunRules extends Statement { private final Statement statement; public RunRules(Statement base, Iterable<TestRule> rules, Description description) { statement = applyAll(base, rules, description); } @Override public void evaluate() throws Throwable { statement.evaluate(); } private static Statement applyAll(Statement result, Iterable<TestRule> rules, Description description) { for (TestRule each : rules) { result = each.apply(result, description); } return result; }}

代碼J RunRules實現

這里簡單舉一個Timeout的代碼。Timeout首先實現了TestRule方法并在apply中定義了自己的算法——構造一個FailOnTimeout對象并返回。不再詳細描述FailOnTimeout的實現,主要原理就是起一個線程來跑測試代碼并等待線程指定的時間,如果超時就會拋出異常。public class Timeout implements TestRule { private final int fMillis; /** * @param millis the millisecond timeout */ public Timeout(int millis) { fMillis = millis; } public Statement apply(Statement base, Description description) { return new FailOnTimeout(base, fMillis); }}

代碼K Timeout實現

外觀模式:封裝統一的對外接口,隱藏內部各個小模塊之間的調用細節,使得用戶既可以簡單的使用facade來達到目的,必要時又可以自行操作內部對象。這里的舉例可能不是非常明顯。圖四是TestClass的公共方法,代碼L給出了 TestClass(org.junit.runners.model.TestClass)的一小部分代碼,TestClass主要作用是封裝對Class的操作提供了JUnit運行時需要用到的功能并隱藏其中操作的細節。在TestClass內部將功能委托給了三個對象Class(java.lang.Class)、FrameworkMethod、FrameworkField來實現,本身充當了外觀(facade)對外提供了getName、getOnlyConstructor、getAnnotations等等接口,在必要的時又可以通過這個外觀獲取到Class、FrameworkMethod等對象。

這里寫圖片描述

圖四 TestClass的公共方法

public class TestClass { private final Class<?> fClass; private Map<Class<?>, List<FrameworkMethod>> fMethodsForAnnotations = new HashMap<Class<?>, List<FrameworkMethod>>(); private Map<Class<?>, List<FrameworkField>> fFieldsForAnnotations = new HashMap<Class<?>, List<FrameworkField>>(); /** * Creates a {@code TestClass} wrapping {@code klass}. Each time this * constructor executes, the class is scanned for annotations, which can be * an expensive process (we hope in future JDK's it will not be.) Therefore, * try to share instances of {@code TestClass} where possible. */ public TestClass(Class<?> klass) { fClass = klass; if (klass != null && klass.getConstructors().length > 1) { throw new IllegalArgumentException( "Test class can only have one constructor"); } for (Class<?> eachClass : getSuperClasses(fClass)) { for (Method eachMethod : MethodSorter.getDeclaredMethods(eachClass)) { addToAnnotationLists(new FrameworkMethod(eachMethod), fMethodsForAnnotations); } for (Field eachField : eachClass.getDeclaredFields()) { addToAnnotationLists(new FrameworkField(eachField), fFieldsForAnnotations); } } } ……}

代碼L TestClass截取

寫在最后

讀JUnit代碼時確實非常贊嘆其合理的封裝和靈活的設計,自己雖然也寫了幾年代碼但是在JUnit的源碼中收獲很多。由于對源碼的鉆研深度以及設計模式的領會不夠深入,文中有很多牽強和錯誤的地方歡迎大家討論指正。最喜歡的是JUnit對裝飾模式和職責鏈的應用,在看到AllDefaultPossiblitiesBuilder中對職責鏈的應用還覺得設計比較合理,等到看到Statement的創建和組裝就感慨設計的精湛了,無論是基本的調用測試方法的邏輯還是@Before、@After等以及實現自TestRule的邏輯一并融入到Statement的構造中,又不會牽扯出太多的耦合??傊疅o論是設計模式還是設計思想,歸根結底就是抽取公共部分,封裝變化,做到靈活、解耦。最后說明這篇文章根據的源代碼是JUnit4.11的,maven坐標如下,在JUnit的其它版本中源碼差別比較大沒有研究過。<dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.11</version> </dependency>

轉自:junit源碼與設計模式欣賞


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
不卡av在线网站| 亚洲高清av在线| 日韩av在线网页| 日本一区二区在线播放| 日韩禁在线播放| 精品久久久久久久久久久久久久| 91免费人成网站在线观看18| 国产一区二区三区在线看| 高清一区二区三区四区五区| 国产精品海角社区在线观看| 亚洲韩国欧洲国产日产av| 欧美在线一区二区视频| 中文字幕成人精品久久不卡| 久久国产精品久久久久久| 欧美日韩国产色视频| 色偷偷噜噜噜亚洲男人的天堂| 人妖精品videosex性欧美| 欧美日韩亚洲高清| 日本久久亚洲电影| 国产91在线高潮白浆在线观看| 日韩69视频在线观看| 亚洲人午夜精品| 高潮白浆女日韩av免费看| 国产精品一区二区久久久久| 亚洲精品福利在线观看| 亚洲高清久久网| 欧美性猛交xxxx久久久| 亚洲国产精品网站| 精品综合久久久久久97| 中文字幕av一区中文字幕天堂| 成人久久一区二区| 国产精品视频免费在线观看| 色综合亚洲精品激情狠狠| 欧美性猛交xxxx乱大交极品| 国产成人精品一区| 国产精品一香蕉国产线看观看| 亚洲成人亚洲激情| 亚洲男人天堂视频| 国产精品视频地址| 成人精品福利视频| 亚洲男人天堂2024| 国产精品爽爽ⅴa在线观看| 精品久久久久久久久中文字幕| 欧美最猛性xxxxx免费| 欧美精品电影免费在线观看| 国模精品一区二区三区色天香| 久久久午夜视频| 国产成人中文字幕| 亚洲综合在线做性| 91av视频导航| 日韩中文字幕在线精品| 国产成人av网址| 欧美综合第一页| 国产欧美精品日韩精品| 亚洲韩国日本中文字幕| 国产不卡视频在线| 日韩精品中文字幕久久臀| 日本一区二区在线免费播放| 2021国产精品视频| 亚洲高清一区二| 日本电影亚洲天堂| 97在线视频免费看| 亚洲国产美女久久久久| 在线播放国产一区中文字幕剧情欧美| 国产综合久久久久| 51精品国产黑色丝袜高跟鞋| 69久久夜色精品国产69| 欧美性xxxx| 精品国产一区二区三区久久| 久久视频精品在线| 国产精品一区二区三区成人| 欧美国产视频日韩| 欧美激情在线一区| 欧美成人免费全部观看天天性色| 日韩av免费一区| 欧美成人午夜激情视频| 亚洲美女免费精品视频在线观看| 日韩中文有码在线视频| 欧美极度另类性三渗透| 欧美日韩免费区域视频在线观看| 亚洲精品二三区| 国产亚洲欧洲黄色| 国产美女久久精品| 亚洲欧洲在线播放| 欧美一级电影免费在线观看| 国产精品一区二区久久| 91亚洲一区精品| 91久久精品国产91性色| 亚洲2020天天堂在线观看| 日韩欧美视频一区二区三区| 夜夜嗨av一区二区三区四区| 中文字幕亚洲无线码在线一区| 成人黄色片网站| 国产亚洲视频中文字幕视频| 亚洲国产成人久久| 在线成人免费网站| 欧美国产日韩精品| 欧美成人高清视频| 黑人精品xxx一区| 国产91色在线播放| 日韩精品极品在线观看| 久久91亚洲精品中文字幕| 在线播放精品一区二区三区| 日韩在线精品一区| 国内精品久久久久影院优| 伦伦影院午夜日韩欧美限制| 欧美野外wwwxxx| 精品国产网站地址| 色yeye香蕉凹凸一区二区av| 97涩涩爰在线观看亚洲| 91久久久久久久一区二区| 欧美激情欧美激情在线五月| 日韩在线一区二区三区免费视频| 91在线|亚洲| 日韩av黄色在线观看| 欧美xxxx做受欧美| 国产丝袜一区视频在线观看| 国产一区二区在线免费| 中文字幕亚洲一区| 欧美大片免费看| 最好看的2019的中文字幕视频| 精品国产乱码久久久久久天美| 91chinesevideo永久地址| 国产成人精品日本亚洲专区61| 色中色综合影院手机版在线观看| 精品中文字幕久久久久久| 中文字幕精品www乱入免费视频| 91精品视频在线免费观看| 日本sm极度另类视频| 久久精品视频中文字幕| 日韩影视在线观看| 欧美一区二三区| 久久久久亚洲精品成人网小说| 久久成年人免费电影| 九九热这里只有精品6| 亚洲性日韩精品一区二区| 亚洲性夜色噜噜噜7777| 亚洲最新视频在线| 久久久成人精品视频| 欧美视频在线视频| 亚洲视频欧美视频| 国模吧一区二区| 啊v视频在线一区二区三区| 日韩av在线网址| 亚洲欧美日本伦理| 欧美老少做受xxxx高潮| 中文字幕亚洲国产| 国产精品久久久久久久久粉嫩av| 欧美视频专区一二在线观看| 久久色在线播放| 国产一区二区三区久久精品| 国产精品免费久久久| 亚洲视频一区二区| 九色91av视频| 国产一区二区三区三区在线观看| 亚洲美女免费精品视频在线观看| 色午夜这里只有精品| 欧美日韩一区二区免费视频| 一区三区二区视频| 欧美国产日韩中文字幕在线| 久久好看免费视频| 欧美激情免费在线| 亚洲最大的av网站| 51久久精品夜色国产麻豆|