摘要:不同目標的自動化測試有不同的測試工具,但是任何工具都無不例外的需要編程的過程,實現源代碼,也可以稱之為測試腳本。
寫在最前面:目前自動化測試并不屬于新鮮的事物,或者說自動化測試的各種方法論已經層出不窮,但是,能夠在項目中持之以恒的實踐自動化測試的團隊,卻依舊不是非常多。有的團隊知道怎么做,做的還不夠好;有的團隊還正在探索和摸索怎么做,甚至還有一些多方面的技術上和非技術上的舊系統需要重構……
?
本文將會從使用和實踐兩個視角,嘗試對基于Web UI自動化測試做細致的分析和解讀,給各位去思考和實踐做一點引路,以便各團隊能找到更好的方式。
1. 使用測試工具《論語》有云:工欲善其事,必先利其器。在開始具體的自動化測試之前,我們需要做好更多的準備,包括以下幾個方面:
?
認識自動化測試
?
?
準備自動化測試工具
?
?
使用有效的方式
?
?
針對具體的測試對象
?
接下來的第一部分內容,我們將會從上述的幾個方面進行探討。
1.1 自動化測試理論介紹?
自動化測試的5W
正如開篇所提到的,自動化測試不再是一個陌生的話題,而是一個具體的存在。作為測試實踐活動的一部分,我們首先分析一下自動化測試的方方面面。
?
WHAT, 什么是自動化測試
G.J.Myers在其經典的著作《軟件測試藝術》(The Art of Software Testing)一書中,給出了測試的定義:
?
“程序測試是為了發現錯誤而執行的過程?!?/p>
這個概念產生于30年前,對軟件測試的認識還非常有局限性,當然也是因為受瀑布開發模型的影響,認為軟件測試是編程之后的一個階段。只有等待代碼開發出來以后,通過執行程序,像用戶那樣操作軟件去發現問題。這里向大家推薦一個測試交流圈q裙:1007119548。
?
自動化測試:以人為驅動的測試行為轉化為機器執行的一種過程
自動化測試,就是把手工進行的測試過程,轉變成機器自動執行的測試過程。該過程,依舊是為了發現錯誤而執行。因此自動化測試的關鍵在于“自動化”三個字。自動化測試的內容,也就相應的轉變成如何“自動化”去實現原本手工進行的測試的過程。
?
所有的“自動化”,依靠的無疑都是程序。
通過程序,可以把手工測試,轉變成自動化測試。
?
?
WHEN, 在什么時候開展自動化測試
自動化測試的開展,依賴于“程序”。那么程序,其實就是由“源代碼”構建而來的。那么原則上,只要能做出自動化測試所需要的“程序”的時候,變可以進行自動化測試。但往往,并不是所有的“時候”都是好的“時機”。從這個W開始,我們將會加入對于成本的顧慮,也正是因為“成本”的存在,才使得下面的討論,變得有意義。
所有的開銷,都是有成本的。構建成“程序”的源代碼,也是由工程師寫出來的。那么需要考慮這個過程中的成本。基于這個考慮,在能夠比較穩定的構建“程序”的時候,不需要花費太多開銷在“源代碼”的時候,就是開展自動化測試的好時機。這個開銷包括編寫和修改源代碼,而源代碼指的是構建出用來做自動化測試的程序的源代碼。
?
?
WHERE, 在什么地方進行自動化測試
自動化測試的執行,依靠的是機器。那么自動化測試必將在“機器”上進行。一般來說,這個機器包括桌面電腦和服務器。通過將寫好的源代碼部署在機器上,構建出用來做自動化測試的"程序",并且運行該程序,實現自動化測試。
?
?
WHICH, 對什么目標進行自動化測試
自動化測試的目標,是被測試的軟件。拋開人工智能的成分,手工測試必將在“人工智能”足夠普及和足夠“智能”之前,替代一大部分不需要“人類智能”的手工測試;以及自動化測試會做一些手工測試無法實施的,或者手工測試無法覆蓋的測試。
不需要“人類智能”的普通手工測試
界面的普通操作
通過固定輸入和固定操作而進行的流程化測試
重復的普通測試
?
手工測試無法實施或者覆蓋的
大量的數據的輸入
大量的步驟的操作
源代碼基本的測試
系統模塊間接口的調用測試
……
在這里向大家推薦一個學習交流群:672899761
?
?
?
HOW, 如何開展自動化測試
準備測試用例
找到合適的自動化測試工具
用準確的編程形成測試腳本
在測試腳本中對目標進行“檢查”,即做斷言
記錄測試日志,生成測試結果
?
?
和所有的其他測試一樣,自動化測試的流程也是由“用例”執行和“缺陷”驗證組成。差別是需要找到合適的“工具”來替代“人手”。不同目標的自動化測試有不同的測試工具,但是任何工具都無不例外的需要“編程”的過程,實現“源代碼”,也可以稱之為測試腳本。于是開展自動化測試的方式基本上如下:
?
?
?
自動化測試的典型金字塔原理
談到自動化測試,就不得不提的一個人和概念就是:Martin Fowler和他的金字塔原理。首先請看金字塔原理的圖示如下:
?
?
?
該圖說明了三個問題:
自動化測試包括三個方面:UI前端界面,Service服務契約和Unit底層單元
越是底層的測試,運行速度越快,時間開銷越少,金錢開銷越少
越是頂層的測試,運行速度越慢,時間開銷越多,金錢開銷越多
這是理想中的金字塔原理。
在實際的項目中,尤其是結合國內的項目實踐,其實還隱藏了另一個問題:越是頂層的測試,效果越明顯。有句話說“貴的東西,除了貴,其他都是好的!”能夠很清晰的闡述這個觀點。
金字塔原理在國內的適應性也有一定的問題
自動化測試的起步不是特別早
甚至軟件測試很長一段時間都在進行基于業務的手工測試,測試人員的代碼能力相對較弱
開發人員在代碼中不太習慣寫單元測試
近些年基于服務契約的API測試也在興起
相對來說,在基于UI前端界面的自動化測試反倒是開展和實施的不是特別多。盡管基于界面的測試帶來的效果還是能夠立竿見影的。對于產品的質量提升,還是比較容易有保證。
?
自動化測試的適用范圍
自動化測試可以涉及和試用的范圍主要在以下方面:
基于Web UI的瀏覽器應用的界面測試
基于WebService或者WebAPI的服務契約測試
基于WCF、.net remoting、Spring等框架的服務的集成測試
基于APP UI的移動應用界面測試
基于Java、C#等編程文件進行的單元測試
本文集中討論第一條:基于Web UI的瀏覽器應用的界面測試。界面的改動對于測試來說,具有較大的成本風險。主要考慮以下方面:
任務測試明確,不會頻繁變動
每日構建后的測試驗證
比較頻繁的回歸測試
軟件系統界面穩定,變動少
需要在多平臺上運行的相同測試案例、組合遍歷型的測試、大量的重復任務
軟件維護周期長
項目進度壓力不太大
被測軟件系統開發比較規范,能夠保證系統的可測試性
具備大量的自動化測試平臺
測試人員具備較強的編程能力
?
?
自動化測試的流程
自動化測試和普通的手工測試遵循的測試流程,與項目的具體實踐相關。一般來說,也是需要從測試計劃開始涉及自動化測試的。
測試計劃:劃定自動化測試的范圍包含哪些需求,涉及到哪些測試過程
測試策略:確定自動化測試的工具、編程方案、代碼管理、測試重點
測試設計:使用測試設計方法對被測試的需求進行設計,得出測試的測試點、用例思維導圖等
測試實施:根據測試設計進行用例編寫,并且將測試用例用編程的方式實現測試腳本
測試執行:執行測試用例,運行測試腳本,生成測試結果
?
1.2 自動化測試工具基于Web UI的自動化測試工具主要有兩大類:付費的商業版工具和免費使用的開源版工具。典型的有兩種:
UFT,QTP被惠普收購以后的新名稱。
通過程序的錄制,可以實現測試的編輯
錄制的測試腳本是 VBScript 語法
成熟版的商業付費工具
工具比較龐大,對具體的項目定制測試有難度
?
SELENIUM,本次選擇的開源工具
本身不是測試工具,只是模擬瀏覽器操作的工具
背后有 Google 維護源代碼
支持全部主流的瀏覽器
支持主流的編程語言,包括:Java、Python、C#、PHP、Ruby、JavaScript等
工具很小,可以實現對測試項目的定制測試方案
基于標準的 WebDriver 語法規范
?
1.2.1 Selenium 基本介紹
Selenium`是開源的自動化測試工具,它主要是用于Web 應用程序的自動化測試,不只局限于此,同時支持所有基于web 的管理任務自動化。
?
Selenium官網的介紹
?
Selenium is a suite of tools to automate web browsers across many platforms.
runs in many browsers and operating systems
can be controlled by many programming languages and testing frameworks.
Selenium 官網:seleniumhq.org/
Selenium Github 主頁:github.com/SeleniumHQ/…
Selenium 是用于測試 Web 應用程序用戶界面 (UI) 的常用框架。它是一款用于運行端到端功能測試的超強工具。您可以使用多個編程語言編寫測試,并且 Selenium 能夠在一個或多個瀏覽器中執行這些測試。
Selenium 經歷了三個版本:Selenium 1,Selenium 2 和 Selenium 3。Selenium 也不是簡單一個工具,而是由幾個工具組成,每個工具都有其特點和應用場景。
Selenium 誕生于 2004 年,當在 ThoughtWorks 工作的 Jason Huggins 在測試一個內部應用時。作為一個聰明的家伙,他意識到相對于每次改動都需要手工進行測試,他的時間應該用得更有價值。他開發了一個可以驅動頁面進行交互的 Javascript 庫,能讓多瀏覽器自動返回測試結果。那個庫最終變成了 Selenium 的核心,它是 Selenium RC(遠程控制)和 Selenium IDE 所有功能的基礎。Selenium RC 是開拓性的,因為沒有其他產品能讓你使用自己喜歡的語言來控制瀏覽器。這就是 Selenium 1。
然而,由于它使用了基于 Javascript 的自動化引擎,而瀏覽器對 Javascript 又有很多安全限制,有些事情就難以實現。更糟糕的是,網站應用正變得越來越強大,它們使用了新瀏覽器提供的各種特性,都使得這些限制讓人痛苦不堪。
在 2006 年,一名 Google 的工程師, Simon Stewart 開始基于這個項目進行開發,這個項目被命名為 WebDriver。此時,Google 早已是 Selenium 的重度用戶,但是測試工程師們不得不繞過它的限制進行工具。Simon 需要一款能通過瀏覽器和操作系統的本地方法直接和瀏覽器進行通話的測試工具,來解決Javascript 環境沙箱的問題。WebDriver 項目的目標就是要解決 Selenium 的痛點。
到了 2008 年,Selenium 和 WebDriver 兩個項目合并。Selenium 有著豐富的社區和商業支持,但 WebDriver 顯然代表著未來的趨勢。兩者的合并為所有用戶提供了一組通用功能,并且借鑒了一些測試自動化領域最閃光的思想。這就是 Selenium 2。
2016 年,Selenium 3 誕生。移除了不再使用的 Selenium 1 中的 Selenium RC,并且官方重寫了所有的瀏覽器驅動。
?
?
Selenium 工具集
?
Selenium IDE
Selenium IDE (集成開發環境) 是一個創建測試腳本的原型工具。它是一個 Firefox 插件,實現簡單的瀏覽器操作的錄制與回放功能,提供創建自動化測試的建議接口。Selenium IDE 有一個記錄功能,能記錄用戶的操作,并且能選擇多種語言把它們導出到一個可重用的腳本中用于后續執行。這里向大家推薦一個測試交流圈q裙:1007119548。
?
?
Selenium RC
Selenium RC 是selenium 家族的核心工具,Selenium RC 支持多種不同的語言編寫自動化測試腳本,通過selenium RC 的服務器作為代理服務器去訪問應用從而達到測試的目的。
selenium RC 使用分Client Libraries 和Selenium Server。
Client Libraries 庫主要主要用于編寫測試腳本,用來控制selenium Server 的庫。
Selenium Server 負責控制瀏覽器行為,總的來說,Selenium Server 主要包括3 個部分:Launcher、Http Proxy、Core。
?
?
Selenium Grid
Selenium Grid 使得 Selenium RC 解決方案能提升針對大型的測試套件或者哪些需要運行在多環境的測試套件的處理能力。Selenium Grid 能讓你并行的運行你的測試,也就是說,不同的測試可以同時跑在不同的遠程機器上。這樣做有兩個有事,首先,如果你有一個大型的測試套件,或者一個跑的很慢的測試套件,你可以使用 Selenium Grid 將你的測試套件劃分成幾份同時在幾個不同的機器上運行,這樣能顯著的提升它的性能。同時,如果你必須在多環境中運行你的測試套件,你可以獲得多個遠程機器的支持,它們將同時運行你的測試套件。在每種情況下,Selenium Grid 都能通過并行處理顯著地縮短你的測試套件的處理時間。
?
?
Selenium WebDriver
WebDriver 是 Selenium 2 主推的工具,事實上WebDriver是Selenium RC的替代品,因為Selenium需要保留向下兼容性的原因,在 Selenium 2 中, Selenium RC才沒有被徹底的拋棄,如果使用Selenium開發一個新的自動化測試項目,那么我們強烈推薦使用Selenium2 的 WebDriver進行編碼。另外, 在Selenium 3 中,Selenium RC 被移除了。
?
?
?
Python 語言的選擇,便捷
測試人員的編程能力普遍不是很強,而Python作為一種腳本語言,不僅功能強大,而且語法優美,支持多種自動化測試工具,而且學習上手比較容易。
Python的社區發展比較好,有著非常多的文檔和支持庫,另外Python也可以在Web開發、數據處理、科學計算等縱多領域有著非常好的應用前景。
對于有一定編程基礎的人員,使用Python作為自動化測試的語言可以非常順暢的轉換,幾乎沒有學習成本。同時Python是標準的面向對象的編程語言,對于C#、Java等面向對象的語言有著非常好的示例作用,通過Python的示例可以非常輕松的觸類旁通,使用其他語言進行Selenium2.0的WebDriver的使用。
讀音:/"pa?θ?n/
Python的創始人為Guido Van Rossum。1989年圣誕節期間,在阿姆斯特丹,Guido為了打發圣誕節的無趣,決心開發一個新的腳本解釋程序,做為ABC 語言的一種繼承。之所以選中Python(大蟒蛇的意思)作為程序的名字,是因為他是一個叫Monty Python的喜劇團體的愛好者。
Python 語言除了在自動化測試領域有出色的表現外,在系統編程,網絡編程,web 開發,GUI開發,科學計算,游戲開發等多個領域應用非常廣泛,而且具有非常良好的社區支持。也就是說學習和掌握python 編程,其實是為你打開了一道更廣闊的大門。
?
?
使用的工具集
IDE: Jetbrains PyCharm
語言: Python
工具: Selenium WebDriver
源代碼管理: SVN/Git
?
1.2.2 JetBrains PyCharm 使用
?
JetBrains PyCharm 的介紹
PyCharm 是 JetBrains 公司針對Python推出的IDE(Integrated Development Environment,集成開發環境)。是目前最好的Python IDE之一。目前包含了兩個版本:
社區版,Community Edition
專業版,Professional Edition
付費
比社區版主要多了Web開發框架
?
我們推薦使用免費的社區版本,進行Python腳本的編寫和自動化測試執行。
PyCharm可以在官網下載,www.jetbrains.com
?
PyCharm 安裝后,如果也安裝過 Python 環境,可以直接進行操作。否則請在 1.2.3 中安裝好 Python,再使用 PyCharm。
?
安裝按照默認的步驟安裝
?
使用方式
?
Create New Project:
創建新的項目,選擇項目創建的位置,選擇Python的解釋器
?image
?
設置location,項目的路徑和名稱
名稱必須以英文字母開頭
名稱不可以有空格
位置不可以在 C:Pytho34中,應該放到普通的目錄中
?
設置interpreter
一個電腦可以裝多個 Python
這里選擇一個你需要的 Pythpn
?
新建Python文件
image
?
?image
?
?在創建的文件中編寫第一個Python語句
?
?
?
?
print("hello Python!")
image
?
右鍵該文件,選擇Run hello,運行該語句,在下面的運行框中會顯示運行結果
C:Python35python.exe D:/Git/WeekendSelenium/untitled/hello.py
hello python!
Process finished with exit code 0
如圖
image
?
Open
打開已經存在的項目,比如別人發給你的項目,或者已經創建過的項目
image
?
image
?
?
安裝后進行設置如下:
?
設置行號的顯示
在PyCharm 里,顯示行號有兩種辦法:
?
臨時設置(不推薦)。右鍵單擊行號處,選擇 Show Line Numbers。
image
?
image.gif
?
?
但是這種方法,只對一個文件有效,并且,重啟PyCharm 后消失。
?
?
永久設置。File --> Settings -->Editor -->Appearance , 之后勾選Show Line Numbers。
?
?
?
image
?
?
設置字體
選擇 Settings | Editor | Colors & Fonts | Fonts
Save AS 主題
選擇 Source Code Pro(建議選擇,等寬字體)
?
?
SVN / Git 在工具中的集成
源代碼管理工具(VCS, version control system)
?
如果TortoiseSVN版本低于 1.8,需要先升級安裝1.8以上的版本
選擇SVN(git)作為代碼的源代碼管理工具。集成在PyCharm中的步驟如下
代碼已經存在在SVN repo中:把代碼放到SVN在本地簽出(check out)的文件夾目錄中,例如 D:SVNXXProjectTrunck
?
image
?
2. 代碼沒有創建:在本地的SVN項目文件夾中新建項目,用PyCharm打開,提交。
image
?
image
?
image
?
3. 用PyCharm打開 剛剛部署的代碼 4. 選擇PyCharm的 `VCS`|`Enable VCS integration`,選擇 Subversion(svn) 或者 Git
image
?
5. 右鍵項目文件的根目錄,選擇 Subversion | add to VCS
?
?
6. 右鍵項目文件的根目錄,或者選 VCS | Commit Directory...
image
?
?
每天打開代碼后,右鍵項目文件的根目錄,首先 Subversion | update project
?
?
如果有沖突,先本地手工保存你做的修改(備份你的文件到其他地方,SVN目錄之外的地方,然后Revert)
?
1.2.3 Selenium 的環境搭建
?
在 Windows 搭建和部署 Selenium 工具
主要包括兩個步驟:
?
安裝 Python 語言
Python的官方網站:www.python.org
Python 目前并行了兩套版本,2.x 和 3.x。如果你之前沒有 Python 的使用經驗,建議使用 Python 3.x 版本。兩套版本互相不兼容,并且 Python 從 3.5(含)開始,不再支持 Windows XP 系統,請注意。
?
選擇安裝目錄
3.4或者3.4以下的版本,都是 C:python34
3.5以上的目錄,默認裝個人文件夾,建議用類似上面的目錄,比如C:python35
?
?
?
?
勾選添加環境變量
勾選Add Python.exe to PATH
?
?
安裝過程中不要關閉彈出來的命令行窗口
?
關于 Python 的安裝,也可以選擇一些第三方的Python 安裝包,典型的有 Anaconda3,這樣的包有豐富的第三方庫,在使用 Python 的過程中會更加方便。這里向大家推薦一個測試交流圈q裙:1007119548。
Anaconda 的官網:www.continuum.io/anaconda-ov…
?
?
?
安裝 Selenium 工具包
由于 安裝好的 Python 默認有 pip Python 包管理工具,可以通過 pip 非常方便的安裝 Selenium。
?
啟動命令行工具:Win+R | 輸入 cmd | 回車
?
?
輸入命令:
pip install selenium
image.gif
?
該命令的執行需要有互聯網聯網環境。此外該命令有以下幾種選項可以使用
?
安裝指定的版本,例如安裝指定的 Selenium 3.4.3
pip install selenium==3.4.3
?
?
安裝最新版的 Selenium
pip install -U selenium
# -U 也可以用 --upgrade
pip install --upgrade selenium
?
?
?
?
卸載安裝當前的 Selenium
pip uninstall selenium
image.gif
?
?
當然,如果您的機器處于非接入互聯網的環境,您可以事先下載 Selenium 的 Python 安裝包,再進行手動安裝。
?
?
官方下載地址:pypi.python.org/pypi/seleni…
?
?
上述地址會下載最新版的 Selenium,目前最先版的是 3.4.3,您也可以根據以下路徑下載指定的 3.4.3
?
?
Selenium 3.4.3 下載地址:pypi.python.org/pypi/seleni…
?
?
下載后,解壓該壓縮包
?
?
然后用命令行進入該壓縮包的根目錄,輸入命令進行安裝
?
python setup.py install
?
配置 瀏覽器 和 驅動
?
Selenium 2 可以默認支持Firefox 46.0或者更低版本,對于其他瀏覽器需要額外安裝驅動。
?
?
Selenium 3 對于所有的瀏覽器都需要安裝驅動,本文以 Chrome 和 Firefox、IE為例設置瀏覽器和驅動。
?
ChromeDriver下載地址:chromedriver.storage.googleapis.com/index.html
?
?
ChromeDriver 與 Chrome 對應關系表:
ChromeDriver版本 | 支持的Chrome版本 |
---|---|
v2.31 | v58-60 |
v2.30 | v58-60 |
v2.29 | v56-58 |
v2.28 | v55-57 |
v2.27 | v54-56 |
v2.26 | v53-55 |
v2.25 | v53-55 |
v2.24 | v52-54 |
v2.23 | v51-53 |
v2.22 | v49-52 |
v2.21 | v46-50 |
v2.20 | v43-48 |
?
?
GeckoDriver下載地址:github.com/mozilla/gec…
?
?
GeckoDriver 與 Firefox 的對應關系表:
GeckoDriver版本 | 支持的Firefox版本 |
---|---|
v0.18.0 | v56 |
v0.17.0 | v55 |
v0.16.0 | v54,需要Selenium 3.4或者以上 |
v0.15.0 | v53,需要Selenium 3.3或者以上 |
?
?
IEDriverServer下載地址:selenium-release.storage.googleapis.com/index.html
?
?
IEDriverServer 的版本需要與 Selenium 保持嚴格一致。
?
?
?
瀏覽器驅動的配置
首先,將下載好的對應版本的瀏覽器安裝。
其次,在 Python 的根目錄中,放入瀏覽器驅動。
最好再重啟電腦,一般情況下不重啟也可以的。
?
?
?
1.3 Selenium 的最簡腳本通過上一節的環境安裝成功以后,我們可以進行第一個對Selenium 的使用,就是最簡腳本編寫。腳本如下:
# 聲明一個司機,司機是個Chrome類的對象
driver = webdriver.Chrome()
# 讓司機加載一個網頁
driver.get("http://demo.ranzhi.org")
# 給司機3秒鐘去打開
sleep(3)
# 開始登錄
# 1. 讓司機找用戶名的輸入框
we_account = driver.find_element_by_css_selector("#account")
we_account.clear()
we_account.send_keys("demo")
# 2. 讓司機找密碼的輸入框
we_password = driver.find_element_by_css_selector("#password")
we_password.clear()
we_password.send_keys("demo")
# 3. 讓司機找 登錄按鈕 并 單擊
driver.find_element_by_css_selector("#submit").click()
sleep(3)
?
實際上一段20行的代碼,也不能算太少了。但是這段代碼的使用,確實體現了 Selenium 的最簡單的使用。我們在下面內容進行闡述。
?
關于面向對象編程
通過前面的介紹,我們知道 Selenium 支持多種語言,并且推薦使用面向對象的方式進行編程。接下來我們將著重介紹如何使用面向對象的方式進行編程。
我們利用 Python 進行面向對象編程,需要首先了解一個概念:類
?
類
類是任何面向對象編程的語言的基本組成,描述了使用的基本方法。我們可能在目前,還不是特別明白類的含義,但是我們可以通過類的使用,來進一步了解。
?
?
類的使用
類,通過實例化進行使用。比如有一個類: Driver,該類有一個方法: head(road)
那么關于這個類的使用,只需要兩個步驟:
實例化該類:d = Driver()
調用類的方法:d.head("中山路")
?
?
了解上述例子和使用以后,我們來看具體的 Selenium 的使用。
?
?這里向大家推薦一個測試交流圈q裙:1007119548。
?
?
具體的對象的使用
在面向對象的理念看來,任何的編碼,都是由對象而來的,這里也不例外。和之前介紹 WebDriver 時候的描述對應,我們需要用到兩種主要的類,并將其實例化。
WebDriver 類:主要靠直接實例化該類為對象,然后用其對象直接調用該類的方法和屬性
WebElement 類:主要通過 WebDriver 類實例化的對象,通過對頁面元素的查找,得到 WebElement 類的對象,然后調用該類的方法和屬性。
上述代碼中,使用了一個 WebDriver 類 的對象,即第2行,聲明了該類的對象,并賦值給變量 driver,接著變量 driver 作為 WebDriver 類的對象,使用了多個 WebDriver 類的方法。
?
注意:Chrome 是 WebDriver 的子類,是 WebDriver 類的一種
get(url): 第5行,打開網址
find_element_by_css_selector(selector): 第12、17、22行都使用了該方法,同時通過對該方法的調用,分別各產生了一個 WebElement類的對象,we_account,we_password和最后一個匿名的對象,并通過產生的三個對象,調用 WebElement 類的方法
clear():清理頁面元素中的文字
send_keys(text):給頁面元素中,輸入新的文字
click():鼠標左鍵點擊頁面元素
?
正是通過這樣的面向對象的方式,產生 Web司機(WebDriver類的對象),并且通過 Web司機不懈的努力,尋找到各種 Web元素(WebElement類的對象)進行操作,這樣便實現了 Selenium WebDriver 作為一款出色的瀏覽器測試工具,進行瀏覽器UI界面的自動化測試的代碼編寫和用例執行。
?
1.4 Selenium WebDriver API 的使用通過上述最簡腳本的使用,我們可以來進一步了解 Selenium 的使用。事實上,上一節用的,便是 Selenium 的 WebDriver API。API(Application Programming Interface,應用程序編程接口,即通過編程語言,操作 WebDriver 的方法集合)
Selenium WebDriver API 官方參考:seleniumhq.github.io/selenium/do…
具體API文檔地址:seleniumhq.github.io/selenium/do…
API 使用: 用現成的類(大部分情況)的方法進行編程
WebDriver
WebElement
?
API 文檔
編程使用說明
介紹了每個方法的使用
方法的作用
方法的參數
方法的返回值
?
?
1.4.1 控制瀏覽器
瀏覽器的控制也是自動化測試的一個基本組成部分,我們可以將瀏覽器最大化,設置瀏覽器的高度和寬度以及對瀏覽器進行導航操作等。
# 瀏覽器打開網址
driver.get("https://www.baidu.com")
# 瀏覽器最大化
driver.maximize_window()
# 設置瀏覽器的高度為800像素,寬度為480像素
driver.set_window_size(480, 800)
# 瀏覽器后退
driver.back()
# 瀏覽器前進
driver.forward()
# 瀏覽器關閉
driver.close()
# 瀏覽器退出
driver.quit()
1.4.2 元素定位操作
WebDriver提供了一系列的定位符以便使用元素定位方法。常見的定位符有以下幾種:
id
name
class name
tag
link text
partial link text
xpath
css selector
那么我們以下的操作將會基于上述的定位符進行定位操作。
對于元素的定位,WebDriver API可以通過定位簡單的元素和一組元素來操作。在這里,我們需要告訴Selenium如何去找元素,以至于他可以充分的模擬用戶行為,或者通過查看元素的屬性和狀態,以便我們執行一系列的檢查。
在Selenium2中,WebDriver提供了多種多樣的find_element_by方法在一個網頁里面查找元素。這些方法通過提供過濾標準來定位元素。當然WebDriver也提供了同樣多種多樣的find_elements_by的方式去定位多個元素。
盡管上述的方式,可以進行元素定位,實際上我們也是更多的用組合的方式進行元素定位。
方法Method | 描述Description | 參數Argument | 示例Example |
---|---|---|---|
id | 該方法通過ID的屬性值去定位查找單個元素 | id: 需要被查找的元素的ID | find_element_by_id("search") |
name | 該方法通過name的屬性值去定位查找單個元素 | name: 需要被查找的元素的名稱 | find_element_by_name("q") |
class name | 該方法通過class的名稱值去定位查找單個元素 | class_name: 需要被查找的元素的類名 | find_element_by_class_name("input-text") |
tag_name | 該方法通過tag的名稱值去定位查找單個元素 | tag: 需要被查找的元素的標簽名稱 | find_element_by_tag_name("input") |
link_text | 該方法通過鏈接文字去定位查找單個元素 | link_text: 需要被查找的元素的鏈接文字 | find_element_by_link_text("Log In") |
partial_link_text | 該方法通過部分鏈接文字去定位查找單個元素 | link_text: 需要被查找的元素的部分鏈接文字 | find_element_by_partial_link_text("Long") |
xpath | 該方法通過XPath的值去定位查找單個元素 | xpath: 需要被查找的元素的xpath | find_element_by_xpath("http://*[@id="xx"]/a") |
css_selector | 該方法通過CSS選擇器去定位查找單個元素 | css_selector: 需要被查找的元素的ID | find_element_by_css_selector("#search") |
接下來的列表將會詳細展示find_elements_by的方法集合。這些方法依據匹配的具體標準返回一系列的元素。
方法Method | 描述Description | 參數Argument | 示例Example |
---|---|---|---|
id | 該方法通過ID的屬性值去定位查找多個元素 | id: 需要被查找的元素的ID | find_elements_by_id("search") |
name | 該方法通過name的屬性值去定位查找多個元素 | name: 需要被查找的元素的名稱 | find_elements_by_name("q") |
class_name | 該方法通過class的名稱值去定位查找多個元素 | class_name: 需要被查找的元素的類名 | find_elements_by_class_name("input-text") |
tag_name | 該方法通過tag的名稱值去定位查找多個元素 | tag: 需要被查找的元素的標簽名稱 | find_elements_by_tag_name("input") |
link_text | 該方法通過鏈接文字去定位查找多個元素 | link_text: 需要被查找的元素的鏈接文字 | find_elements_by_link_text("Log In") |
partial_link_text | 該方法通過部分鏈接文字去定位查找多個元素 | link_text: 需要被查找的元素的部分鏈接文字 | find_elements_by_partial_link_text("Long") |
xpath | 該方法通過XPath的值去定位查找多個元素 | xpath: 需要被查找的元素的xpath | find_elements_by_xpath("http://div[contains(@class,"list")]") |
css_selector | 該方法通過CSS選擇器去定位查找多個元素 | css_selector: 需要被查找的元素的ID | find_element_by_css_selector(".input_class") |
依據ID查找
請查看如下HTML的代碼,以便實現通過ID的屬性值去定義一個查找文本框的查找:
"search" type="text" name="q" value=""
class="input-text" maxlength="128" autocomplete="off"/>
根據上述代碼,這里我們使用find_element_by_id()的方法去查找搜索框并且檢查它的最大長度maxlength屬性。我們通過傳遞ID的屬性值作為參數去查找,參考如下的代碼示例:
def test_search_text_field_max_length(self):
# get the search textbox
search_field = self.driver.find_element_by_id("search")
# check maxlength attribute is set to 128
self.assertEqual("128", search_field.get_attribute("maxlength"))
如果使用find_elements_by_id()方法,將會返回所有的具有相同ID屬性值的一系列元素。
依據名稱name查找
這里還是根據上述ID查找的HTML代碼,使用find_element_by_name的方法進行查找。參考如下的代碼示例:
# get the search textbox
self.search_field = self.driver.find_element_by_name("q")
image.gif
?
同樣,如果使用find_elements_by_name()方法,將會返回所有的具有相同name屬性值的一系列元素。
依據class name查找
除了上述的ID和name的方式查找,我們還可以使用class name的方式進行查找和定位。
?
事實上,通過ID,name或者類名class name查找元素是最提倡推薦的和最快的方式。當然Selenium2 WebDriver也提供了一些其他的方式,在上述三類方式條件不足,查找無效的時候,可以通過這些其他方式來查找。這些方式將會在后續的內容中講述。
請查看如下的HTML代碼,通過改代碼進行練習和理解.
image.gif
?
根據上述代碼,使用find_element_by_class_name()方法去定位元素。
def test_search_button_enabled(self):
# get Search button
search_button = self.driver.find_element_by_class_name("button")
# check Search button is enabled
self.assertTrue(search_button.is_enabled())
同樣的如果使用find_elements_by_class_name()方法去定位元素,將會返回所有的具有相同name屬性值的一系列元素。
依據標簽名tag name查找
利用標簽的方法類似于利用類名等方法進行查找。我們可以輕松的查找出一系列的具有相同標簽名的元素。例如我們可以通過查找表中的 下面有一個HTML的示例,這里在無序列表中使用了標簽。 這里面我們使用find_elements_by_tag_name()的方式去獲取全部的圖片,在此之前,我們將會使用find_element_by_class_name()去獲取到指定的 具體代碼如下: 依據鏈接文字link查找 鏈接文字查找通常比較簡單。使用find_element_by_link_text請查看以下示例 測試代碼如下: 依據部分鏈接文字partial text查找 這里依舊使用上述的列子進行代碼編寫: 依據XPath進行查找 XPath是一種在XML文檔中搜索和定位節點node的一種查詢語言。所有的主流Web瀏覽器都支持XPath。Selenium2可以用強大的XPath在頁面中查找元素。 常用的XPath的方法有starts-with(),contains()和ends-with()等 若想要了解更多關于XPath的內容,請查看www.w3schools.com/XPath/ 如下有一段HTML代碼,其中里面的沒有使用ID,name或者類屬性,所以我們無法使用之前的方法。亞這里我們可以通過的alt屬性,定位到指定的tag。 具體代碼如下: 當然,如果使用find_elements_by_xpath()的方法,將會返回所有匹配了XPath查詢的元素。 依據CSS選擇器進行查找 CSS是一種設計師用來描繪HTML文檔的視覺的層疊樣式表。一般來說CSS用來定位多種多樣的風格,同時可以用來是同樣的標簽使用同樣的風格等。類似于XPath,Selenium2也可以使用CSS選擇器來定位元素。 請查看如下的HTML文檔。 "block-subtitle">Recently added item(s)
"close skip-link-close" href="#" title="Close">×
"empty">You have no items in your shopping cart.
我們來創建一個測試,驗證這些消息是否正確。 特殊 iframe 操作 iframe 元素會創建包含另外一個文檔的內聯框架(即行內框架)。 iframe: 紫禁城 在一個中,包含了另一個 示例
這里是段落,標記一個段落,屬于外層
需要定位上面示例中的 :這里是div中的段落,需要被定位 如下是selenium WebDiriver的代碼 特殊 Select 操作 是選擇列表 Select 是個selenium的類selenium.webdriver.support.select.Select Select 類的路徑: C:Python35Libsite-packagesseleniumwebdriversupportselect.py 示例,選擇 Audi 組合操作 自動化經驗的積累,需要100%按照手工的步驟進行操作。 比如步驟如下: 自動產生了一個 點擊 代碼示例 1.4.3 鼠標事件操作 Web測試中,有關鼠標的操作,不只是單擊,有時候還要做右擊、雙擊、拖動等操作。這些操作包含在ActionChains類中。 常用的鼠標方法: context_click() # 右擊 double_click() # 雙擊 drag_and_drop() # 拖拽 move_to_element() # 鼠標停在一個元素上 click_and_hold() # 按下鼠標左鍵在一個元素上 例子: 1.4.4 鍵盤事件操作 鍵盤操作經常處理的如下: 代碼如下 1.4.5 截圖操作 截圖的方法:save_screenshot(file) 在上一節,我們對 Selenium WebDriver 的使用,僅僅停留在讓網頁自動的進行操作的階段,并沒有對任何一個步驟進行“檢查”。當然,這樣沒有“檢查”的操作,實際上是沒有測試意義的。那么第一項,我們需要解決的便是“檢查”的問題。 所謂“檢查”,實際上就是斷言。對需要檢查的步驟操作,通過對預先設置的期望值,和執行結果的實際值之間的對比,得到測試的結果。在這里,我們并不需要多帶帶的寫 if 語句進行各種判定,而是可以使用編程語言中對應的單元測試框架,即可解決好此類問題。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。 轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/6925.html 摘要:跳躍表的空間復雜度為。不過,二叉查找樹是有可能出現一種極端的情況的,就是如果插入的數據剛好一直有序,那么所有節點會偏向某一邊。例如這種接結構會導致二叉查找樹的查找效率變為這會使二叉查找樹大打折扣。假如我們要用某種數據結構來維護一組有序的int型數據的集合,并且希望這個數據結構在插入、刪除、查找等操作上能夠盡可能著快速,那么,你會用什么樣的數據結構呢?
數組
一種很簡單的方法應該就是采用數組了... 摘要:同時,在元類中,我們還需要加上一個判斷,只有在這個類創建時才需要控制其類的生成,其他的就不需要了。完整代碼后臺回復元類獲取原創不易,如果文章對你有用的話,點贊留言轉發是對我的最大支持日常學代碼不止,還有美和樂趣
我之前在深入理解python中的類和對象中說過,python中的類也是一個對象,可以說是類對象,可以由type來創建類對象的。有了這個知識我們先看看下面這個函數:
showIm... 摘要:所以,雅虎的開發人員就試圖開發一個通用的無單點問題的分布式協調框架,以便讓開發人員將精力集中在處理業務邏輯上。在立項初期,考慮到之前內部很多項目都是使用動物的名字來命名的例如著名的項目雅虎的工程師希望給這個項目也取一個動物的名字。
前言
提到ZooKeeper,相信大家都不會陌生。Dubbo,Kafka,Hadoop等等項目里都能看到它的影子。但是你真的了解 ZooKeeper 嗎?如... 摘要:如果你只是簡單羅列出這幾個鉤子函數的名稱,不具體深入闡述的話,你這樣的回答很難令面試官滿意。那么接下來,閏土大叔將手摸手教你如何深入淺出地說出令面試官滿意的有亮點的回答。
當面試官問:談談你對vue的生命周期的理解,聽到這句話你是不是心里暗自竊喜:這也太容易了吧,不就是beforeCreate、created、beforeMount、mounted、beforeUpdate、updat... 摘要:是一個瀏覽器自動化測試框架可以模擬用戶的所有操作很久以前就想把李敖大師的所有微博爬取下來一直沒空前天看見群里有人推薦和就學了做了個你們想爬別人的只要把李敖大師的地址換成你要的就行了我還沒學不過我猜他就是去掉功能的瀏覽器這樣可以讓爬蟲更快如果
selenium 是一個瀏覽器自動化測試框架.可以模擬用戶的所有操作.
很久以前就想把李敖大師的所有微博爬取下來.一直沒空,前天看見群里有人推薦s... 閱讀 1990·2021-09-22 16:05 閱讀 9252·2021-09-22 15:03 閱讀 2879·2019-08-30 15:53 閱讀 1697·2019-08-29 11:15 閱讀 902·2019-08-26 13:52 閱讀 2347·2019-08-26 11:32 閱讀 1797·2019-08-26 10:38 閱讀 2561·2019-08-23 17:19來獲取行數。
"promos">
。
def test_count_of_promo_banners_images(self):
# get promo banner list
banner_list = self.driver.find_element_by_class_name("promos")
# get images from the banner_list
banners = banner_list.find_elements_by_tag_name("img")
# check there are 20 tags displayed on the page
self.assertEqual(20, len(banners))
"#header-account" class="skip-link skip-account">
"icon">
"label">ACCOUNT Description
def test_my_account_link_is_displayed(self):
# get the Account link
account_link =
self.driver.find_element_by_link_text("ACCOUNT Description")
# check My Account link is displayed/visible in
# the Home page footer
self.assertTrue(account_link.is_displayed())
def test_account_links(self):
# get the all the links with Account text in it
account_links = self.driver.
find_elements_by_partial_link_text("ACCOUNT")
# check Account and My Account link is
# displayed/visible in the Home page footer
self.assertTrue(2, len(account_links))
?
"promos">
def test_vip_promo(self):
# get vip promo image
vip_promo = self.driver.
find_element_by_xpath("http://img[@alt="Shop Private Sales - Members Only"]")
# check vip promo logo is displayed on home page
self.assertTrue(vip_promo.is_displayed())
# click on vip promo images to open the page
vip_promo.click()
# check page title
self.assertEqual("VIP", self.driver.title)
def test_shopping_cart_status(self):
# check content of My Shopping Cart block on Home page
# get the Shopping cart icon and click to open the
# Shopping Cart section
shopping_cart_icon = self.driver.
find_element_by_css_selector("div.header-minicart
span.icon")
shopping_cart_icon.click()
# get the shopping cart status
shopping_cart_status = self.driver.
find_element_by_css_selector("p.empty").text
self.assertEqual("You have no items in your shopping cart.",
shopping_cart_status)
# close the shopping cart section
close_button = self.driver.
find_element_by_css_selector("div.minicart-wrapper
a.close")
close_button.click()
?
這里是H1,標記了標題
## 查找并定位 iframe
element_frame = driver.find_element_by_css_selector("#iframe-1")
## 切換到剛剛查找到的 iframe
driver.switch_to.frame(element_frame)
## 定位
driver.find_element_by_css_selector("#div-1 > p")
## TODO....
## 退出剛剛切換進去的 iframe
driver.switch_to.default_content()
?
## 查找并定位到 select
element_select = driver.find_element_by_css_selector("#brand")
## 用Select類的構造方法,實例化一個對象 object_select
object_select = Select(element_select)
## 操作 object_select
object_select.select_by_index(3)
## 也可以這樣
object_select.select_by_value("audi")
## 還可以這樣
object_select.select_by_visible_text("Audi")
的第五個
driver.find_element_by_css_selector("#customer_chosen").click()
sleep(1)
driver.find_element_by_css_selector("#customer_list > li:nth-child(5)")
# 方法模擬鼠標右鍵,參考代碼如下:
# 引入ActionChains 類
from selenium.webdriver.common.action_chains import ActionChains
...
# 定位到要右擊的元素
right =driver.find_element_by_xpath("xx")
# 對定位到的元素執行鼠標右鍵操作
ActionChains(driver).context_click(right).perform()
...
# 定位到要雙擊的元素
double = driver.find_element_by_xpath("xxx")
# 對定位到的元素執行鼠標雙擊操作
ActionChains(driver).double_click(double).perform()
代碼
描述
send_keys(Keys.BACKSPACE)
刪除鍵(BackSpace)
send_keys(Keys.SPACE)
空格鍵(Space)
send_keys(Keys.TAB)
制表鍵(Tab)
send_keys(Keys.ESCAPE)
回退鍵(Esc)
send_keys(Keys.ENTER)
回車鍵(Enter)
send_keys(Keys.CONTROL,"a")
全選(Ctrl+A)
send_keys(Keys.CONTROL,"c")
復制(Ctrl+C)
from selenium import webdriver
# 引入Keys 類包
from selenium.webdriver.common.keys import Keys
import time
driver = webdriver.Chrome()
driver.get("http://www.baidu.com")
# 輸入框輸入內容
driver.find_element_by_id("kw").send_keys("selenium")
time.sleep(3)
# 刪除多輸入的一個m
driver.find_element_by_id("kw").send_keys(Keys.BACK_SPACE)
time.sleep(3)
# 輸入空格鍵+“教程”
driver.find_element_by_id("kw").send_keys(Keys.SPACE)
driver.find_element_by_id("kw").send_keys("教程")
time.sleep(3)
# ctrl+a 全選輸入框內容
driver.find_element_by_id("kw").send_keys(Keys.CONTROL,"a")
相關文章
以后有面試官問你跳躍表,你就把這篇文章扔給他
再有人問什么是元類,就把這篇文章扔給他!
如果有人問你ZooKeeper是什么,就把這篇文章發給他。
如何解釋vue的生命周期才能令面試官滿意?
selenium 自動化爬蟲 5分鐘爬取新浪李敖大師1751條微博.
發表評論
0條評論
Keven
男|高級講師
TA的文章
閱讀更多
虛擬主機的瞬時帶寬什么意思-VPS和虛擬主機的區別?
一個子網能容納多少主機-子網掩碼下最多可以容納多少個主機?
命名規范
text-overflow-文本溢出包含樣式設置
ES6入門之對象的新增方法
JS基礎知識:變量對象、作用域鏈和閉包
更快助你弄懂React-高階組件
通俗易懂理解ES6 - ES6的變量類型及Iterator