前言
雖然一直在做app自動化,但沒能總結過appium中的元素定位方式,appium中一部分定位方式和selenium類似,一部分又不相同。因此本文將總結appium中常用的元素定位方式,同時與selenium做一個橫向的比較
通過resource-id定位
resource-id也稱為id,resource-id是唯一的

代碼實現
# 元素定位
agree_continue_id = "com.baidu.searchbox:id/positive_button"
WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.ID, agree_continue_id)))
driver.find_element_by_id(agree_continue_id).click()
通過className定位
這里的className對應的元素有2個,看上面的index是2,不同意是1,同意並繼續是2

代碼實現
# 元素定位,注意我這里的寫法,用的是find_elements_by_class_name,另外還要加索引
agree_continue_class = "android.widget.Button"
WebDriverWait(driver, 10, 1).until(EC.visibility_of_all_elements_located((MobileBy.CLASS_NAME, agree_continue_class)))
driver.find_elements_by_class_name(agree_continue_class)[1].click()
通過AccessibilityId定位
AccessibilityId也稱為content-desc

代碼實現
# 元素定位
tiku_AccessibilityId = "題庫"
WebDriverWait(driver, 10, 1).until(EC.visibility_of_all_elements_located((MobileBy.ACCESSIBILITY_ID, tiku_AccessibilityId)))
driver.find_element_by_accessibility_id(tiku_AccessibilityId).click()
通過xpath定位
常規定位
比如我要用text屬性的文本值定位,這時候只需要寫成xpath表達式(注意appium中沒有單獨的text定位方式,必須要結合xpath)

代碼實現
# 元素定位
agree_continue_xpath = "//*[@text='同意並繼續']"
WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.XPATH, agree_continue_xpath)))
driver.find_element_by_xpath(agree_continue_xpath).click()
contains模糊定位
這種經常用於獲取toast的時候,toast文本內容較長,可以采用contains包含部分文本的匹配方式。當然,可以用來模糊匹配上面的文本屬性“同意並繼續”

代碼實現
# 元素定位
agree_continue_xpath = "//android.widget.Button[contains(@text, '並繼續')]"
WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.XPATH, agree_continue_xpath)))
driver.find_element_by_xpath(agree_continue_xpath).click()
組合定位
比如xpath中同時包含class和text兩個屬性

代碼實現
# 元素定位
agree_continue_xpath = "//*[@class='android.widget.Button' and @text='同意並繼續']"
WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.XPATH, agree_continue_xpath)))
driver.find_element_by_xpath(agree_continue_xpath).click()
層級定位
使用lazy uiautomatorviewer,可以看到底下有個fullIndexXpath,這種是全路徑的形式,也是層級的形式

代碼實現
# 元素定位
agree_continue_xpath = "//android.widget.FrameLayout[1]/android.widget.FrameLayout[1]/android.widget.FrameLayout[1]/android.widget.FrameLayout[1]/android.widget.RelativeLayout[1]/android.widget.FrameLayout[1]/android.widget.FrameLayout[1]/android.widget.LinearLayout[1]/android.widget.RelativeLayout[1]/android.widget.Button[2]"
WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.XPATH, agree_continue_xpath)))
driver.find_element_by_xpath(agree_continue_xpath).click()
通過AndroidUIAutomator定位
使用lazy uiautomatorviewer,可以看到底下有個uiaSelector,將其拷貝到代碼

代碼實現
# 元素定位
agree_continue_android_uiautomator = "new UiSelector().className(\"android.widget.Button\").textContains(\"同意並繼續\").resourceId(\"com.baidu.searchbox:id/positive_button\")"
WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.ANDROID_UIAUTOMATOR, agree_continue_android_uiautomator)))
driver.find_element_by_android_uiautomator(agree_continue_android_uiautomator).click()
appium與selenium元素定位之比較
| 框架 | 常用的元素定位方式 | 備注 |
|---|---|---|
| appium | id, className, AccessibilityId, xpath, AndroidUIAutomator | 對於h5頁面,也支持selenium的name, link text, css等定位方式 |
| selenium | id, className, name, tag name, link text, paratial link text, xpath ,css |
