元素定位
与Web自动化测试一样,app自动化测试过程中最重要一个环节就是元素定位,只有准确定位到了元素才能进行相关元素的操作,如输入、点击、拖拽、滑动等。appium提供了许多元素定位的方法,如id定位、name定位、class定位、层级定位等等.... 接下来将会给大家来实践运用这些定位技巧。
元素定位方式
- id
- name
- class
- List定位
- 相对定位
- Xpath定位
- H5页面元素定位
- Uiautomator定位
id定位
日常生活中身边可能存在相同名字的人,但是每个人的身份证号码是唯一的,在app界面元素中也可以使用id值来区分不同的元素,然后进行定位操作。Appium中可以使用 find_element_by_id() 方法来进行id定位。
测试场景1
- 安装考研帮kaoyan3.1.0.apk
- 点击升级页面取消按钮
- 点击引导页面的跳过按钮
kyb_cancel_skip.py
fromappium
importwebdriver
desired_caps={}
desired_caps[
'platformName']=
'Android'
desired_caps[
'deviceName']=
'127.0.0.1:62025'
desired_caps[
'platforVersion']=
'5.1.1'
desired_caps[
'app']=
r'C:\Users\Shuqing\Desktop\kaoyan3.1.0.apk'
desired_caps[
'appPackage']=
'com.tal.kaoyan'
desired_caps[
'appActivity']=
'com.tal.kaoyan.ui.activity.SplashActivity'
driver=webdriver.Remote(
'http://localhost:4723/wd/hub',desired_caps)
driver.implicitly_wait(
5)
driver.find_element_by_id(
'android:id/button2').click()
driver.find_element_by_id(
'com.tal.kaoyan:id/tv_skip').click()
思考
- 如果安装的版本最新的包,或者升级到了最新的版本,则启动后没有升级弹窗元素该如何处理?
- 跳过引导页面首次启动和非首次启动场景该如何处理?
方案探索1——if条件判断
有同学可能想到用if来做条件判断,判断元素是否存在,存在则点击,不存在则跳过。
kyb_cancel_skip_if.py
fromappium
importwebdriver
desired_caps={}
desired_caps[
'platformName']=
'Android'
desired_caps[
'deviceName']=
'127.0.0.1:62025'
desired_caps[
'platforVersion']=
'5.1.1'
desired_caps[
'app']=
r'C:\Users\Shuqing\Desktop\kaoyan3.1.0.apk'
desired_caps[
'appPackage']=
'com.tal.kaoyan'
desired_caps[
'appActivity']=
'com.tal.kaoyan.ui.activity.SplashActivity'
desired_caps[
'noReset']=
'True'
driver=webdriver.Remote(
'http://localhost:4723/wd/hub',desired_caps)
driver.implicitly_wait(
5)
cancelBtn=driver.find_element_by_id(
'android:id/button2')
skipBtn=driver.find_element_by_id(
'com.tal.kaoyan:id/tv_skip')
if cancelBtn:
cancelBtn.click()
else:
print(
'no cancelBtn')
if skipBtn:
skipBtn.click()
else:
print(
'no skipBtn')
方案探索2——异常捕捉
既然上面的if语句判断无法生效,但是我们发现一个突破口,那就是捕捉NoSuchElementException异常。
kyb_cancel_skip_try.py
fromappium
importwebdriver
fromselenium.common.exceptions
importNoSuchElementException
desired_caps={}
desired_caps[
'platformName']=
'Android'
desired_caps[
'deviceName']=
'127.0.0.1:62025'
desired_caps[
'platforVersion']=
'5.1.1'
desired_caps[
'app']=
r'C:\Users\Shuqing\Desktop\kaoyan3.1.0.apk'
desired_caps[
'appPackage']=
'com.tal.kaoyan'
desired_caps[
'appActivity']=
'com.tal.kaoyan.ui.activity.SplashActivity'
desired_caps[
'noReset']=
'True'
driver=webdriver.Remote(
'http://localhost:4723/wd/hub',desired_caps)
driver.implicitly_wait(
5)
def check_cancelBtn():
print(
"check_cancelBtn")
try
:
cancelBtn = driver.find_element_by_id(
'android:id/button2')
except
NoSuchElementException:
print(
'no CancelBtn')
else
:
cancelBtn.click()
def check_skipBtn():
print(
"check_skipBtn")
try
:
skipBtn = driver.find_element_by_id(
'com.tal.kaoyan:id/tv_skip')
except
NoSuchElementException:
print(
'no skipBtn')
else
:
skipBtn.click()
check_updateBtn()
check_skipBtn()
id定位综合实践——自动登录
测试场景
- 启动App,进入到登录界面
- 在登录页面输入用户名“自学网2018”,密码‘zxw2018’ 然后点击登录。
- 可以把启动后检测升级弹窗和引导页面的模块抽离作为独立的模块被其他模块调用,提高代码复用率。
- 获取用户名密码输入框和登录按钮的元素id属性,另外要考虑启动时App之前是否登录过账号,已经登录过和未登录场景流程不一样。
- 注意:send_keys()传入中文时需要在capability中配置如下内容:
需求分析
desired_caps[
'unicodeKeyboard']=
"True"
desired_caps[
'resetKeyboard']=
"True"
设置之后会有Appium的输入法守护来执行输入操作
代码实现
kby_login.py
fromfind_element.capability
import
driver
,
NoSuchElementException
deflogin
():
driver.find_element_by_id
(
'com.tal.kaoyan:id/login_email_edittext')
.clear()
driver.find_element_by_id
(
'com.tal.kaoyan:id/login_email_edittext')
.send_keys(
'自学网2018')
driver.find_element_by_id
(
'com.tal.kaoyan:id/login_password_edittext')
.send_keys(
'zxw2018')
driver.find_element_by_id
(
'com.tal.kaoyan:id/login_login_btn')
.click()
try:
driver.find_element_by_id
(
'com.tal.kaoyan:id/mainactivity_button_mysefl')
exceptNoSuchElementException
:
login
()
else:
driver.find_element_by_id
(
'com.tal.kaoyan:id/mainactivity_button_mysefl')
.click()
driver.find_element_by_id
(
'com.tal.kaoyan:id/activity_usercenter_username')
.click()
login
()
注意:
使用Appium做了输入操作之后,如果出现输入法无法唤起,可以在系统设置——语言和输入法——将当前输入法替换为系统输入法或者其他输入法。
name定位
根据name进行定位,对于android来说,就是text属性
用法
fromfind_element.capability
import
*
driver.find_element_by_name(
'请输入用户名')
.send_keys(
'自学网2017')
driver.find_element_by_name(
'登录')
.click()
说明:由于text稳定性不是很好,所以appium 1.5开始废弃了该方法。
classname定位
classname定位是根据元素类型来进行定位,但是实际情况中很多元素的classname都是相同的,
如上例中登录页面中的用户名和密码都是clasName属性值都是:“android.widget.EditText” 因此只能定位第一个元素也就是用户名,而密码输入框就需要使用其他方式来定位,这样其实很鸡肋.一般情况下如果有id就不必使用classname定位。
by_classname.py
from find
_element.capability
importdriver
driver.find
_element
_by_class_name(
'android.widget.EditText').send
_keys(
'自学网2018')
driver.find
_element
_by_class_name(
'android.widget.EditText').send
_keys(
'zxw2018')
driver.find
_element
_by_class_name(
'android.widget.Button').click()
相对定位
相对定位是先找到该元素的有对应属性的父元素节点,然后基于父元素进行元素定位。
测试案例
不使用id元素定位方式,在新用户注册界面点击添加头像按钮。
代码实现
by_relative.py
from find
_element.capability
importdriver
driver.find
_element
_by_id(
'com.tal.kaoyan:id/login_register_text').click()
root
_element=driver.find
_element
_by_id(
'com.tal.kaoyan:id/activity_register_parentlayout')
root
_element.find
_element
_by_class_name(
'android.widget.ImageView').click()
xpath定位
xpath定位是一种路径定位方式,主要是依赖于元素绝对路径或者相关属性来定位,但是绝对路径xpath执行效率比较低(特别是元素路径比较深的时候),一般使用比较少。通常使用xpath相对路径和属性定位。
1.xpath路径表达式
表达式 |
描述 |
/ |
从根节点选取。 |
// |
从匹配选择的当前节点选择文档中的节点,而不考虑它们的位置。 |
nodename |
选取此节点的所有子节点。 |
. |
选取当前节点。 |
.. |
选取当前节点的父节点。 |
@ |
选取属性。 |
通配符 |
描述 |
* |
匹配任何元素节点。 |
@* |
匹配任何属性节点。 |
node() |
匹配任何类型的节点。 |
实践案例
使用xpath定位元素来进行登录操作。
by_xpath.py
fromfind_element.capability
import
driver
driver.find_element_by_xpath(
'//android.widget.EditText[@text="请输入用户名"]')
.send_keys(
'zxw1234')
driver.find_element_by_xpath(
'//*[@class="android.widget.EditText" and @index="3"]')
.send_keys(
'zxw123456')
driver.find_element_by_xpath(
'//android.widget.Button')
.click()
#
driver.find_element_by_xpath(
'//*[@class="android.widget.Button"]')
.click()
List定位
前面我们提到相同的classname属性值元素无法区分定位,那么在本节课将使用List定位来解决这个问题。List定位首先是使用find_elements_by_XX获取一组相同的class属性的元素,然后使用数组下标来区分标记不同元素进行相关操作。
测试案例1
在新用户注册界面点击添加头像按钮后,选择指定的图片保存作为头像。
by_list.py
from find_element.capability import
driver
driver.find_element_by_id(
'com.tal.kaoyan:id/login_register_text').click()
driver.find_element_by_id(
'com.tal.kaoyan:id/activity_register_userheader').click()
images=
driver.find_elements_by_id(
'com.tal.kaoyan:id/item_image')
images[
10].click()
driver.find_element_by_id(
'com.tal.kaoyan:id/save').click()
list定位综合案例——用户注册
测试场景
- 进入注册界面设置头像
- 输入注册信息:用户名、密码、邮箱
- 完善院校和专业信息 (院校:上海-同济大学 专业:经济学类-统计学-经济统计学)
- 完成注册
代码实现
kyb_register.py
from find_element.capability
importdriver
import
random
#
进入注册界面选择并设置头像
driver.find_element_by_id(
'com.tal.kaoyan:id/login_register_text').
click()
driver.find_element_by_id(
'com.tal.kaoyan:id/activity_register_userheader').
click()
images=driver.find_elements_by_id(
'com.tal.kaoyan:id/item_image')
images[
10].
click()
driver.find_element_by_id(
'com.tal.kaoyan:id/save').
click()
#
注册信息填写
username=
'zxw2018'+
'FLY'+str(
random.randint(
1000,
9000))
(
'username: %s'%username)
driver.find_element_by_id(
'com.tal.kaoyan:id/activity_register_username_edittext').send_keys(username)
password=
'zxw'+str(
random.randint(
1000,
9000))
(
'password: %s'%password)
driver.find_element_by_id(
'com.tal.kaoyan:id/activity_register_password_edittext').send_keys(password)
email=
'51zxw'+str(
random.randint(
1000,
9000))+
'@163.com'
(
'email: %s'%email)
driver.find_element_by_id(
'com.tal.kaoyan:id/activity_register_email_edittext').send_keys(email)
driver.find_element_by_id(
'com.tal.kaoyan:id/activity_register_register_btn').
click()
#
院校选择
driver.find_element_by_id(
'com.tal.kaoyan:id/perfectinfomation_edit_school_name').
click()
#
选择省份
driver.find_elements_by_id(
'com.tal.kaoyan:id/more_forum_title')[
1].
click()
#
选择具体院校--
同济大学
driver.find_elements_by_id(
'com.tal.kaoyan:id/university_search_item_name')[
1].
click()
#
专业选择
driver.find_element_by_id(
'com.tal.kaoyan:id/activity_perfectinfomation_major').
click()
#
选择经济学类-
统计学-
经济统计学
driver.find_elements_by_id(
'com.tal.kaoyan:id/major_subject_title')[
1].
click()
driver.find_elements_by_id(
'com.tal.kaoyan:id/major_group_title')[
2].
click()
driver.find_elements_by_id(
'com.tal.kaoyan:id/major_search_item_name')[
1].
click()
#
点击“进入考研帮”按钮
driver.find_element_by_id(
'com.tal.kaoyan:id/activity_perfectinfomation_goBtn').
click()
注意:运行前记得将noRest设置为:desired_caps['noReset']='False' 以免之前的注册残留信息干扰。
报错&解决方案
元素定位报错
selenium.common.exceptions.NoSuchElementException: Message: An element could
notbe located on the
pageusing the given search parameters.
【解决方案】检查元素id值是否写错。