Appium(六):元素定位

1. 元素定位

对于自动化测试来说,核心技能就是对象的定位了。不管是web页面上的按钮或输入框,还是移动app上的一个按钮或输入框,我们要想对其进行点击或输入操作,前提是要先找到这个对象。

webdriver提供了八种元素定位的方法:

  • id
  • name
  • class name
  • tag name
  • link text
  • partial link text
  • xpath
  • css selector

在python语言中对应的定位方法如下:

find_element_by_id()

find_element_by_name()

find_element_by_class_name()

find_element_by_tag_name()

find_element_by_link_text()

find_element_by_partial_link_text()

find_element_by_xpath()

find_element_by_css_selector()

一组元素定位的方法如下:

find_elements_by_id()

find_elements_by_name()

find_elements_by_class_name()

find_elements_by_tag_name()

find_elements_by_link_text()

find_elements_by_partial_link_text()

find_elements_by_xpath()

find_elements_by_css_selector()

Appium完全继承了WebDriver中所定义的这些方法,除此之外对其进行了扩展,以便适合移动端对象的定位与操作。

由Mobile JSON Wire Protocol协议中定义的方法,更适合移动设备上的控件定位。

  • ios uiautomation:一个递归的、对应使用UIAutomation library搜索元素的字符串(IOS-only)。
  • android uiautomator:一个递归的、对应使用UIAutomation Api搜索元素的字符串(Android-only)。
  • accessibility id:一个递归的、对应本地Accessibility选项实现的Id/Name进行元素搜索的字符串。

 对于python来说,在WebDriver的方法的基础上增加了下列方法:

find_element_by_accessibility_id()

find_elements_by_accessibility_id()

find_element_by_android_uiautomator()

find_elements_by_android_uiautomator()

但是我们对照查询到的元素,可以发现很多的定位方法无法实现。

点击审查元素,发现根本就没有name什么的,最多的还是class、id元素。

所以我们只需要学习id、class、xpath定位就可以完成定位元素的目的了。 

1.1 id定位

经过我几次测试,发现app的id值有两个属性可以表示,分别是id和resource-id。

如果id只有一个,那么就使用find_element_by_id来查询。

# coding:utf-8
import time
from appium import webdriver

# 初始化
desired_caps = {}
# 使用哪种移动平台
desired_caps['platformName'] = 'Android'
# Android版本
desired_caps['platformVersion'] = '5.1.1'
#使用adb devices -l 查询,当有多台设备时,需要声明
desired_caps['deviceName'] = '127.0.0.1:62001'
#包名
desired_caps['appPackage'] = 'com.android.settings'
# com.tencent.mobileqq
#界面名
desired_caps['appActivity'] = '.Settings'

# 启动服务
driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps)

driver.find_element_by_id('com.android.settings:id/search').click()

time.sleep(5)

driver.quit()

启动这个程序,我们可以看到点击了搜索按钮,进入了搜索界面。

而id值不是每个元素都会显示的,而resource-id可以查询到,但是在大多数时候resource-id是一系列元素的id,所以我们需要使用find_elements_by_id方法来查询id。

# coding:utf-8
import time
from appium import webdriver

# 初始化
desired_caps = {}
# 使用哪种移动平台
desired_caps['platformName'] = 'Android'
# Android版本
desired_caps['platformVersion'] = '5.1.1'
#使用adb devices -l 查询,当有多台设备时,需要声明
desired_caps['deviceName'] = '127.0.0.1:62001'
#包名
desired_caps['appPackage'] = 'com.android.settings'
# com.tencent.mobileqq
#界面名
desired_caps['appActivity'] = '.Settings'
# 启动服务
driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps)

titles = driver.find_elements_by_id("com.android.settings:id/title")
for title in titles:
    print(title.text)

time.sleep(5)

driver.quit()

1.2 class定位

class一般都不止一个,所以应该需要遍历一遍得到的元素,然后缩小搜索条件来获得目标元素。

我们可以借助python中的pop()方法确定想要这一组元素中的第几个,并对它进行点击或输入操作。

pop(0) 或pop(-1):默认获得一组元素中的最后一个。

pop(n):获得一组元素中的第n-1个。

一个class元素,使用find_element_by_class_name来查询。

# coding:utf-8
import time
from appium import webdriver

# 初始化
desired_caps = {}
# 使用哪种移动平台
desired_caps['platformName'] = 'Android'
# Android版本
desired_caps['platformVersion'] = '5.1.1'
#使用adb devices -l 查询,当有多台设备时,需要声明
desired_caps['deviceName'] = '127.0.0.1:62001'
#包名
desired_caps['appPackage'] = 'com.android.settings'
# com.tencent.mobileqq
#界面名
desired_caps['appActivity'] = '.Settings'

# 启动服务
driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps)

driver.find_element_by_id('com.android.settings:id/search').click()
driver.find_element_by_class_name("android.widget.EditText").send_keys("hello")

time.sleep(5)

driver.quit()

可以看到点击搜索按钮后出现后,在输入框输入了hello。

 

 

如果class元素有很多,就可以使用find_elements_by_class_name来查询。

# coding:utf-8
import time
from appium import webdriver

# 初始化
desired_caps = {}
# 使用哪种移动平台
desired_caps['platformName'] = 'Android'
# Android版本
desired_caps['platformVersion'] = '5.1.1'
#使用adb devices -l 查询,当有多台设备时,需要声明
desired_caps['deviceName'] = '127.0.0.1:62001'
#包名
desired_caps['appPackage'] = 'com.android.settings'
#界面名
desired_caps['appActivity'] = '.Settings'

# 启动服务
driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps)

text_views = driver.find_elements_by_class_name("android.widget.TextView")
for text_view in text_views:
    print(text_view.text)

time.sleep(5)

driver.quit()

1.3 xpath 

在webdriver上xpath定位是功能强大的一种定位方式。我个人惯用于此方法来定位web页面上的元素。但是在android上要定位的是控件,而非页面元素,xpath定位的写法也会有所不同。 

# coding:utf-8
import time
from appium import webdriver

# 初始化
desired_caps = {}
# 使用哪种移动平台
desired_caps['platformName'] = 'Android'
# Android版本
desired_caps['platformVersion'] = '5.1.1'
#使用adb devices -l 查询,当有多台设备时,需要声明
desired_caps['deviceName'] = '127.0.0.1:62001'
#包名
desired_caps['appPackage'] = 'com.android.settings'
# com.tencent.mobileqq
#界面名
desired_caps['appActivity'] = '.Settings'

# 启动服务
driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps)
driver.implicitly_wait(5)#隐式等待,下一章会讲解
driver.find_element_by_id("com.android.settings:id/search").click()
driver.find_element_by_class_name("android.widget.EditText").send_keys("hello")
driver.find_element_by_xpath("//*[@content-desc='收起']").click()

time.sleep(5)

driver.quit()

就目前而言吧,我觉得appium的xpath定位很不好使用,可能是因为还不够熟练的原因吧。

# coding:utf-8
import time
from appium import webdriver

# 初始化
desired_caps = {}
# 使用哪种移动平台
desired_caps['platformName'] = 'Android'
# Android版本
desired_caps['platformVersion'] = '5.1.1'
#使用adb devices -l 查询,当有多台设备时,需要声明
desired_caps['deviceName'] = '127.0.0.1:62001'
#包名
desired_caps['appPackage'] = 'com.android.settings'
# com.tencent.mobileqq
#界面名
desired_caps['appActivity'] = '.Settings'

# 启动服务
driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps)

elements = driver.find_elements_by_xpath("//*[contains(@text,'设')]")
for element in elements:
    print(element.text)

time.sleep(5)

driver.quit()

和webdriver一样,如果使用find_element_by_xx方法,如果传入一个没有的特征,会报NoSuchElementException的错误。如果使用find_elements_by_xx方法,如果传入一个没有的特征,不会报错,会返回一个空列表。

猜你喜欢

转载自www.cnblogs.com/liuhui0308/p/12015458.html