引子
開陽ssl適用于網(wǎng)站、小程序/APP、API接口等需要進行數(shù)據(jù)傳輸應(yīng)用場景,ssl證書未來市場廣闊!成為創(chuàng)新互聯(lián)建站的ssl證書銷售渠道,可以享受市場價格4-6折優(yōu)惠!如果有意向歡迎電話聯(lián)系或者加微信:18982081108(備注:SSL證書合作)期待與您的合作!
繼多版本模擬器的支持工作告一段落之后,如何利用這些技術(shù)產(chǎn)生更大的價值,成為了接下來需要思考的問題。當然,接下來的課題就涉及到了今天的圖像對比技術(shù)。說來有點內(nèi)疚,雖然也算是科班出身,只可惜大學還沒有真正理解圖像處理的價值,現(xiàn)在又要為自己的過去買單,看來出來混,遲早是要換的。
大環(huán)境
先談一下圖像對比在我廠使用的大環(huán)境,調(diào)研了幾類產(chǎn)品,雖然不能說很全,但是也可以略見一斑。
面對海量的圖片數(shù)據(jù),使用最多的就是使用全局特征及局部特征進行去重、分類,這個主要應(yīng)用于圖片相關(guān)的部門。
還有一種需求可以歸納為測試需要,什么性能測試、競品測試及UI類測試,一切圍繞著相似度,來獲取我們需要的信息。
這次,我要做的就是第二類測試需要,主要基于下面幾個使用場景:
第一,在移動web自動化方面,對于UI的驗證還是使用selenium+webdriver去獲取WebElement,不過這種方式只能驗證這個元素是否存在,并不能驗證元素的樣式是否滿足我們的預(yù)期,同時,對于selector的維護成本還是比較大的,尤其是面對一群對于可測試性毫不care的fe。
第二,說起來遇到不靠譜的fe,對于一些開發(fā)能力比較強的測試開發(fā)來說,可以直接通過codeReview的方式確定功能的影響范圍。但是對于很多同學來說,這個要求還是比較難的。因此,也考慮到這一點,可以通過將線上線下環(huán)境對比的方式,來獲取到UI的不同,從而為測試范圍的裁剪提供依據(jù)。
第三,在代碼合并階段,經(jīng)常出現(xiàn)某些同學把svn代碼合錯或者漏合的現(xiàn)象,但是由于平時版本迭代較快,很多同學也只負責自己的項目,對用例更新不及時,對最近上線的項目不了解,就可能導(dǎo)致回歸時的疏漏,造成事故。基于這點考慮,只要使用圖像對比技術(shù),將線上與線下的UI進行對比,就可以在一定程度上規(guī)避一些較明顯問題。
大環(huán)境應(yīng)該就是這樣,接下來就該思考一下如何實現(xiàn)了。
思考及調(diào)研
初步的想法是先著手去做線上線下測試,一來收益會比較明顯,二來可以作為后續(xù)工作的基礎(chǔ)。大體的思路就是如何去獲取頁面截圖,然后如何去對比,最后如何把對比的結(jié)果展示出來。
如何截圖,在當前情況下,并沒有認為這個是多大的難點,既然之前就已經(jīng)使用了selenium的截圖功能,這個就應(yīng)該可以實現(xiàn),因此就著重去考慮對比的問題了。
對于圖像對比,第一件做的事情,是先去了解下有沒有比較相似的產(chǎn)品。在這里也感謝下老大的支持,在調(diào)研的過程中,老大給我推薦了幾個接觸過圖像對比技術(shù)的同學,在跟他們的交流過程中,也漸漸有了思路。
第一個接觸的,是一個實習生MM,正在跟一個高工在做Android底圖的性能測試,提供了3中思路:RGB對比、灰度直方圖、SIFT特征提取。RGB對比,簡單點說就是通過對每個像素點進行R、G、B三個通道的值進行對比,從而得到整張圖的相似度,這種方式較后兩種來說會比較精確?;叶戎狈綀D和SIFT特征提取對于整體上的匹配效果較好,但在對比粒度上會相對差一些。
第二個是網(wǎng)搜的同學,提供了一個叫圖以類聚的平臺,提供對海量圖片的去重分類服務(wù),也是使用特征提取的方式。
第三個是移動云的同學,之前是通過圖像對比技術(shù)解決Android客戶端自動化基于不同分辨率坐標點的匹配,最后因為某些原因被擱置了。
第四個是在內(nèi)網(wǎng)上搜到的一個工具,是基于selenium進行截圖的工具。
實現(xiàn)方案
在經(jīng)過充分思考之后,開始著手與接下來的開發(fā)工作,實現(xiàn)思路整理如下:
這塊需要說明的是,基準圖片與測試截圖的環(huán)境,需要盡可能保持一致,這樣才可以避免由于環(huán)境差異導(dǎo)致的問題,比如IP定位。在做這個的時候,第一個想法其實是線上跟線下環(huán)境直接比,最后發(fā)現(xiàn)某些頁面還是會有一定區(qū)別,因此就采用了這種同步一套線上環(huán)境最為基準的方式。
在獲取基準圖片和測試截圖的過程中,需要保證頁面已經(jīng)加載完畢。在功能自動化中,為了便于項目可測試,我們在頁面中添加了monitor的標記,當這個標記出現(xiàn)時,我們則認為頁面已經(jīng)加載完畢。其實這個對于大部分頁面來說只能說明我想要測試的元素已經(jīng)加載完了,并且已經(jīng)將事件綁定完畢,但是有一小部分頁面,比如違章查詢,已經(jīng)不去遵守這個原則了。并且,頁面的加載完畢,并不能代碼所有的資源都已經(jīng)完全呈現(xiàn)出來,這就導(dǎo)致需要一種機制來解決這個問題。因此,在截圖這個流程中,就使用了我們的圖像對比技術(shù),sleep 2秒,然后截圖,隨后再跟上一張圖進行對比,如果相似度滿足一定要求,則認為頁面已經(jīng)渲染完畢。
頁面截圖完畢后,接下來就將這兩張圖片進行對比,并記錄下來兩張圖不相似的地方,并生成對比結(jié)果圖片,方便后續(xù)對測試結(jié)果的查看。
實施——截圖
首先是獲取基準圖片和測試圖片,實現(xiàn)比較簡單,直接驗證頁面的monitor元素是否已經(jīng)出現(xiàn),時間邏輯為
while(執(zhí)行耗時 < 預(yù)期最大耗時 && 沒有找到monitor){
if (monitor元素 != null)
找到monitor;
else
等一段時間;
}
等待2秒;
截圖;
雖然在找到monitor元素后等待了2秒,大部分頁面都可以完全呈現(xiàn),但是還是有些頁面無法加載完成,最長的加載時間會達到10秒以上。如果再增大等待時間,勢必會對其他用例的執(zhí)行時間產(chǎn)生影響,并且也不能保證在低網(wǎng)速的情況下所有頁面都完全加載完畢。因此為了避免頁面不完全加載的情況,在此使用定時截圖,定時對比的方式,來保證頁面完全加載,實現(xiàn)邏輯修改如下:
上一次截圖 = null
while(執(zhí)行耗時 < 預(yù)期最大耗時 && 頁面沒有加載完畢){
當前截圖 = 截圖();
if (對比相似度(當前截圖 , 上一次截圖) > 一定相似度)
頁面加載完畢;
else{
上一次截圖 = 當前截圖;
等待2秒;
}
}
這樣一改,就能夠保證,如果這個頁面在2秒鐘之內(nèi)沒有變化的話,就認為頁面已經(jīng)完全加載完畢了。不過也會有一個問題,假如頁面消耗了2.01秒加載完畢,那么我們要在第三次截圖的時候才能判斷這個頁面已經(jīng)加載完畢了,也就是說從加載完畢到程序反饋有4秒鐘的時間浪費,這樣整體執(zhí)行下來,整個用例的執(zhí)行時間會有所提升,如果以一個case每次對比多3秒來計算,生成基準圖和當前圖共需要浪費6秒的時間,如果是執(zhí)行100條用例,那么將會是10分鐘的浪費。從時間上來看,其實并不是很長,但是最后還是想到了一種優(yōu)化策略:
定義截圖數(shù)組 ;
while(執(zhí)行耗時 < 預(yù)期最大耗時 && 頁面沒有加載完畢 && 截圖數(shù)組.length > 3){
當前截圖 = 截圖();
if (對比相似度(當前截圖 , 截圖數(shù)組[length-3]) > 一定相似度)
頁面加載完畢;
else{
截圖數(shù)組.add(當前截圖);
等待0.5秒;
}
}
如此,既能夠保證兩張對比圖的時間間隔,同時也可以在0.5ms內(nèi)完成響應(yīng)。
實施——圖像對比
初步的圖像對比工作,已經(jīng)在實現(xiàn)截圖的過程中完成了。邏輯如下:
if ( 當前圖片.width != 基準圖片.width || 當前圖片.height != 基準圖片.height){
圖片不一致,返回;
}
相似像素數(shù) = 0;
for(遍歷 當前圖片.width){
for( 遍歷 當前圖片.height){
if ( 當前圖片元素RGB數(shù)組[x][y] - 基準圖片元素RGB數(shù)組[x][y] < 色差閾值 ){
相似像素數(shù)++;
}
}
}
相似度 = 相似像素數(shù) / 總像素數(shù);
if( 相似度 > 0.9 )
相似;
else
不相似;
已經(jīng)可以對兩張圖片的相似度進行對比,但是在調(diào)試中發(fā)現(xiàn),由于像素點較多,如果只有很小的一部分有所更改,這種方式便很難發(fā)現(xiàn),對比的精確度有待提高。因此又將圖片進行了水平和垂直的切分,將圖片切成 水平切分數(shù)*垂直切分數(shù)個圖塊,然后對每個圖塊進行相似度對比,從而提高了圖片的相似度。
隨后又發(fā)現(xiàn),在截圖過程中也會存在頁面對部分樣式進行了細微調(diào)整,比如對某個元素的向左偏了1px,對于用戶來說,是看不出來這種差別的,而我們的對比結(jié)果卻會因為這種原因而變得不準確。圍繞著以用戶視覺為基準的原則,又對當前的算法進行了優(yōu)化,對每個像素進行了偏移量支持,并以圖塊為單位進行整體偏移驗證。
再后來,面對實際的用戶需求,對于某些頁面,可能會有一些動態(tài)文字,隨著時間的不同有所不同,比如時間類的文字。對于用戶來說,這個是不在頁面差異的范圍內(nèi)的,但是我們的截圖會由于獲取時間不同而存在或多或少的差異。于是,有添加了對于執(zhí)行區(qū)域不進行驗證的功能。
實施——結(jié)果圖生成
結(jié)果圖的目的主要還是為了更快的找到頁面的差異,例如下面這張結(jié)果圖,對于頁面的不同一眼就能看出來。(右上角的不同是個人手機截圖的問題)
分享及優(yōu)化
功能都實現(xiàn)完畢,接下來就帶給大組的同事們一次分享。在最后的Q&A階段,有一個問題引起了后續(xù)的思考。有一位同學提到截圖的性能問題。如果截圖的底層是經(jīng)由adb實現(xiàn),由于android sd卡I/O瓶頸,則很難在2秒的時間完成截圖、保存、傳輸?shù)絇C端這個過程。于是就讀了下selenium的截圖實現(xiàn),實現(xiàn)流程大致如下:
AndroidDriver
從這里乍一看貌似是返回了一個圖像信息的字符串
AndroidWebDriver
ViewAdapter
最優(yōu)經(jīng)由反射機制調(diào)用WebView的capturePicture方法,獲取瀏覽器返回的截圖數(shù)據(jù),經(jīng)由response返回。
在閱讀源碼之前,也對當前截圖的耗時進行了驗證,平均截圖時間在1秒左右,也驗證了這種B/S形式傳輸?shù)男室捎赼db。既然截圖會存在一定的耗時,那么,對于我們現(xiàn)在的截圖功能來說,實際獲得的截圖則會比獲得完整截圖時的時間早1秒左右,同時我想到能不能去并行截圖呢?
嘗試了一下,發(fā)現(xiàn)截圖的時間反倒慢了,看了下Android webview的實現(xiàn),由于synchronized(obj)的原因,只能同時進行一個頁面的截圖。最后采取了比較折中的方式,每0.5秒進行一次截圖任務(wù)的派送,經(jīng)由截圖隊列將任務(wù)發(fā)送至截圖線程,從而降低了由于截圖耗時導(dǎo)致的無效等待時間。以下是優(yōu)化后的部分代碼。
CaptureThread,進行截圖工作
@Override public void run() { System.out.println("截圖線程"+ this.id + "已啟動"); while(true){ if(mission== null){ continue; } //獲取隊列數(shù)據(jù) String currentSessionId = String.copyValueOf(CaptureMissionManager.getInstance(this.managerId).sessionId.toCharArray()); try { SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String beginTime = df.format(new Date()); System.out.println("截圖開始時間為:"+beginTime); File tmpfile = ((TakesScreenshot)mission.getDriver()).getScreenshotAs(OutputType.FILE); // 關(guān)鍵代碼,執(zhí)行屏幕截圖,默認會把截圖保存到temp目錄 FileUtils.copyFile(tmpfile, new File(CompareImage.captureDir + File.separator +mission.getCaptureName() + ".jpg")); //同一session時,會將截圖信息保存到圖片列表 if(currentSessionId.equals(CaptureMissionManager.getInstance(this.managerId).sessionId)){ CaptureMissionManager.getInstance(this.managerId).p_w_picpathList.add(mission.getCaptureName()); //重新排序,避免由于截圖完成時間不同導(dǎo)致的判斷失誤 Collections.sort(CaptureMissionManager.getInstance(this.managerId).p_w_picpathList); System.out.println(CaptureMissionManager.getInstance(this.managerId).p_w_picpathList); } this.mission = null; this.isUsed = false; } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } } }
CaptureMissionManager 負責截圖線程池管理及任務(wù)發(fā)送
public class CaptureMissionManager extends Thread{ private static HashMap<String,CaptureMissionManager> managers = null; public BlockingQueue queue = new BlockingQueue(30); private static final int MAX_THREAD_COUNT = 1; //最大線程數(shù) public ArrayList<String> p_w_picpathList = new ArrayList<String>(); public String sessionId = ""; /** * 圖片截取線程池 */ public ArrayList<CaptureThread> threadPool = new ArrayList<CaptureThread>(); public CaptureMissionManager(String id){ this.updateSessionId(); //創(chuàng)建線程池資源 for(int i=0; i<MAX_THREAD_COUNT ;i++){ CaptureThread thread = new CaptureThread(i+1,id); threadPool.add(thread); thread.start(); } System.out.println("啟動截圖管理線程"); this.start(); } /** * 獲取可用線程 * @return */ private CaptureThread getThread(){ for(int i = 0 ; i < threadPool.size() ; i ++){ CaptureThread thread = threadPool.get(i); if(!thread.isUsed()){ return thread; } } return null; } public static CaptureMissionManager getInstance(String key){ CaptureMissionManager manager = CaptureMissionManager.managers.get(key); if( manager == null){ manager = new CaptureMissionManager(key); CaptureMissionManager.managers.put(key, manager); } return manager; } /** * 添加截圖任務(wù) * @param captureName */ public void addCaptureMission(WebDriver driver,String captureName){ try { CaptureMission mission = new CaptureMission(driver, captureName); queue.put(mission); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } /** * 清空任務(wù)及任務(wù)記錄 */ public void clearAllMissionAndRecord(){ this.updateSessionId(); this.queue.clear(); this.p_w_picpathList.clear(); } public void updateSessionId(){ Calendar c = Calendar.getInstance(); this.sessionId = c.getTimeInMillis() + ""; } @Override public void run() { while(true){ CaptureThread thread = this.getThread(); if(thread != null && this.queue.size() > 0 ){ try { CaptureMission mission = (CaptureMission)this.queue.get(); thread.setMission(mission); thread.setUsed(true); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } } } }
當前文章:圖像對比在UI自動化中的應(yīng)用
分享路徑:http://aaarwkj.com/article30/jegiso.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供定制開發(fā)、微信公眾號、網(wǎng)站維護、自適應(yīng)網(wǎng)站、網(wǎng)站設(shè)計、用戶體驗
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)