selenium常用API

1.八大元素定位方式

driver.find_element_by_id(‘text’)

通过id去定位控件【在web端ID一般是唯一的】

# 0
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

driver.find_element_by_id('cart_num').click() # driver.find_element_by_id_text('cart_num')仅仅是定位到car_num控件,click()才是点击的动作
driver.find_element_by_name(‘text’)

通过name去定位控件【在web端name有可能是重复的,默认找的是第一个,如果报错一般会换其他方式】

# 
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

driver.find_element_by_name('key').send_keys('王麻子')
driver.find_element_by_class_name(‘text’)

通过class去定位控件【对于属性值中间有空格的如’but bg’为复合类,不建议用class去定位,因为会报错】【所用控件必须具有链接属性】

# 
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

driver.find_element_by_class_name('but2').click()
driver.find_element_by_link_text(‘text’)

【传入值为a标签中间加的值】

# 家装节
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

driver.find_element_by_link_text('家装节').click()
driver.find_element_by_partial_link_text(‘text’)

【传入值部分匹配,如’T恤男’,‘男2016’,‘T恤男2’;注意传入值条件:1.必须唯一;2.必须连续】

# T恤男2016
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

driver.find_element_by_partial_link_text('T恤男').click()
driver.find_element_by_xpath
# 绝对路径不适用于页面层级太深的
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

driver.find_element_by_xpath("/html/body/div/div/div/div/form/input[1]").click()  # 绝对路径/
driver.find_element_by_xpath("//input[@class='but1']").click()  # 相对路径// 1.(单条件定位)用属性定位@后跟属性class或其他,=后为唯一值;2.(多条件定位)('//input[@name='key' and @class='but1']')
driver.find_element_by_xpath("//input[@class='but1']").click()  # 单条件定位
driver.find_element_by_xpath("//input[@name='key'and @class='but1']").click()  # 多条件定位
driver.find_element_by_xpath("//input[contains(@placeholder,'请输入')]").click()  # 属性值模糊定位
driver.find_element_by_xpath("//a[text()='家装节']").click()  # 值条件定位
driver.find_element_by_xpath("//*[text()='家装节']").click()  # 值条件定位,*代表正则匹配,只要是text值为家装节的就可以被搜索到
driver.find_element_by_xpath("//div[@class='schbox']/form/input[1]").click()  # 父级定位 爸找儿子
driver.find_element_by_xpath("//input[@class='but1']/..").click()  # 子级定位 儿子找爸
driver.find_element_by_xpath("//input[@class='but1']/../input[2]").click()  # 兄弟定位,子级定位再父级定位
driver.find_element_by_css_selector

注意不同标签,css-selector也会一同排序
写绝对路径最好用xpath

# 
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

driver.find_element_by_css_selector("html>body>div>div>div>div>form>input").click()  # 绝对路径
driver.find_element_by_css_selector("i#cart_num").click()  # id定位 【i是标签名,用id定位必须是#】
driver.find_element_by_css_selector("input.but1").click()  # class定位
driver.find_element_by_css_selector("input[placeholder='请输入你要查找的关键字']").click()  # 搜索值唯一定位
driver.find_element_by_css_selector("input[name='key'][class='but1']").click()  # 多条件定位
driver.find_element_by_css_selector("div.schbox>form>input:nth-child(1)").click()  # 父级定位
driver.find_element_by_css_selector("div.schbox>form>input:first-child").click()  # 父级定位-第一个
driver.find_element_by_css_selector("div.schbox>form>input:last-child").click()  # 父级定位-最后一个
driver.find_element_by_css_selector("div.schbox>form>input:nth-last-child(2)").click()  # 父级定位-倒数第二个
driver.find_element_by_css_selector("form#form>span:nth-child(9)>input").click()  # css是不同标签一同排序,比如span和div【此处id为form,第一个form为标签名】
driver.find_element_by_tag_name

页面的tag实在是太多了,因此一般用做计数标签数量,

# 
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

aa = driver.find_elements_by_tag_name('div')
print(len(aa))

2.操作

(1)获取验证信息

driver.current_url:对于要获取点击的文本链接信息,则需要先点击,才能再获取。
driver.title:直接获取当前网页的title信息

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

title = driver.title  # 获取网页的title信息【网页标签的名字】

driver.find_element_by_link_text("联系客服").click()
url = driver.current_url  # 获取当前跳转后的页面地址

print(title)
print(url)
(2)页面控制

driver.refresh():刷新
driver.back():回退
driver.forward():前进

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()

driver.refresh()  # 刷新
time.sleep(2)
driver.back()  # 回退
time.sleep(2)
driver.forward()  # 前进
time.sleep(2)
(3)窗体设置

driver.maximize_window():窗体最大化
driver.set_windows_size(1200,720):设置窗体长1200宽720

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window()  # 窗体最大化
driver.set_windows_size(1200,720)  # 设置窗体宽1200高720
(4)动作事件

定位控件.send_keys(‘text’):对定位控件输入内容text
定位控件.clear():对定位空间清空内容
定位控件.click():点击定位控件

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window() 

driver.find_element_by_name('key').send_keys('王麻子')  # 输入内容
driver.find_element_by_name('key').clear()  # 清空输入内容
driver.find_element_by_partial_link_text('T恤男').click()  # 点击
(5)获取属性数据

定位控件.size:获取控件长宽数据,数据类型为字典{‘height’:’’,‘weihght’:’’}
定位控件.text:获取控件上文本信息
定位控件.get_attribute(“href”):获取控件内默认信息,href为属性名
定位控件.get_attribute(‘value’) : 固定传value,在输入内容后获取其回显信息【先输入内容在获取回显信息】
定位控件.is_displayed():判断控件是否显示

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window() 

size = driver.find_element_by_name('key').size  # 获取控件长宽数据,获取的数据类型为字典{'height':'','weihght':''}
text = driver.find_element_by_name('key').text  # 获取控件上文本信息
at = driver.find_element_by_name('key').get_attribute("href")  # 获取控件内默认信息,href为属性名
dis = driver.find_element_by_name('key').is_displayed()  # 判断是否显示,true/false
text = driver.find_element_by_name('key').get_attribute('value')  # 固定传value拿到输入后的回显信息
(6)鼠标事件

ele = 定位控件
ActtionChains(driver).move_to_element(ele).perform():将鼠标移动到定位控件上悬停【类似于hover】
ActtionChains(driver).context_click(ele).perform():鼠标右击定位控件
ActtionChains(driver).double_click(ele).perform():鼠标双击定位控件

scour = driver.find_element_by_xpath(""):确定拖拽目标的起点
target = driver.find_element_by_xpath(""):确定拖拽目标的终点
ActionChains(driver).drag_and_drop(scour,target):拖拽目标从起点到终点

ActionChains(driver).drag_and_drop_by_offset(scour,100,0):拖拽目标(起点,x值,y值)

from selenium.webdriver.common.action_chains import ActionChains  # 引入 ActionChains 类
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window() 

ele = driver.find_element_by_link_text("母婴玩具")
ActtionChains(driver).move_to_element(ele).perform()  # 把鼠标移到某个控件上.perform()使动作生效
ActtionChains(driver).context_click(ele).perform()  # 右击
ActtionChains(driver).double_click(ele).perform()  # 双击
# .perform() 运行动作

# 拖动控件
scour = driver.find_element_by_xpath("")
target = driver.find_element_by_xpath("")
ActionChains(driver).drag_and_drop(scour,target)

ActionChains(driver).drag_and_drop_by_offset(scour,100,0)  # 偏移
(7)键盘事件

定位控件.send(‘女装’):输入数据
定位控件.keys(Keys.Back_SPACE):删除1个字符
定位控件.keys(Keys.SPACE):输入空格
定位控件.keys(Keys.CONTROL,‘a’):传入CTRL+a
定位控件.keys(Keys,ENTER):传入enter
注意:复制后需要重新选中粘贴才看得到效果【查看keys方法可以看见相应的键盘输入内容】

from selenium.webdriver.common.keys import Keys  # 引入keys
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window() 

text = driver.find_element_by_name('key').send('女装')  # 输入数据
text = driver.find_element_by_name('key').keys(Keys.Back_SPACE)  # 删除1个字符
text = driver.find_element_by_name('key').keys(Keys.SPACE)  # 输入空格
(8)等待时间

因为代码执行的速度非常快,因此若不给以适当的等待时间,由于页面未加载完毕,可能会出现漏报甚至错报,所以加入等待时间。
time.sleep(4):强制等待时间【线程休眠,不进行执行操作,直到时间等待完才继续执行】
dirver.implicitly_wait(10):隐式等待时间,相当于最大等待时间(全局),如果只用了2s则无需等待10秒,便可继续执行后面操作【缺点是第一个是如果第一页的元素第二页也有,因为网络原因或其他原因加载慢了或是没加载出来,原本需要在第二个页面查找的元素,返回的元素却是第一个页面的元素,而且在该函数之后如果出现了强制等待时间,则会打破其的全局性,接下来的代码将不会有等待时间】
ele = WebDriverWait(driver,15,0.5).until(EC.presence_of_element_located((By.XPATH,“定位控件”))) /n ele.控件操作:显式等待时间【最大等待时间15s,每0.5秒检测一次,知直到当前页面加载出来xpath的xx的控件,才对其做相应操作】XPATH可为其他定位方法名如ID、CSS_SELECTOR、…

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window() 

time.sleep(2)  # 强制等待时间,时间等待完才继续执行

# 隐式等待时间
dirver.implicitly_wait(10)  # 隐式等待时间,相当于最大等待时间(全局),如果只用了2s则无需等待10秒,便可继续执行后面操作【缺点是第一个是如果第一页的元素第二页也有,因为网络原因或其他原因加载慢了或是没加载出来,原本需要在第二个页面查找的元素,返回的元素却是第一个页面的元素】
time.sleep(3)  # 而且在隐式等待时间下有time.sleep()则会打破全局的规则,而且之后代码执行也不会有时间等待

显式等待时间
from selenium.webdriver.support.ui import WebDriverWait # 显式等待时间所需要引入的模块
from selenium.webdriver.common.by import By # 显式等待时间所需要引入的模块
from selenium.webdriver.support import expected_conditions as EC # 显式等待时间所需要引入的模块

from selenium.webdriver.support.ui import WebDriverWait  # 显式等待时间所需要引入的模块
from selenium.webdriver.common.by import By  # 显式等待时间所需要引入的模块
from selenium.webdriver.support import expected_conditions as EC  # 显式等待时间所需要引入的模块
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php') # 打开项目地址
driver.maximize_window() 

# 最大等待时间15s,每0.5秒检测一次,知直到当前页面加载出来xpath的xx的控件
ele = WebDriverWait(self.driver,15,0.5).until(EC.presence_of_element_located(
	(By.XPATH,"//div[@class='schbox']/form/input[1]")))
ele.send_keys("123456")

ele = WebDriverWait(self.driver,15,0.5).until_not(EC.presence_of_element_located(
	(By.XPATH,"//div[@class='schbox']/form/input[1]")))
ele.send_keys("123456")

等待时间的时间怎么确认?以多次测试的最大等待时间+1s-2s为准

(9)多窗口

driver.switch_to.window(driver.window_handles[-1]):切换最新打开的窗体
driver.close():关闭当前句柄所在窗体
driver.quit():退出所有窗体

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://www.baidu.com') # 打开项目地址
driver.maximize_window() 
print(driver.windiws_handles)  # 打印当前所有窗体的信息
driver.find_element_by_id('kw'),send_keys('123')
driver.find_element_by_id('su').click()

driver.find_element_by_partial_link_text('123网址').click()
print(driver.windiws_handles)  # 打印当前所有窗体的信息
print(driver.current_window_handle)  # 打印当前窗体所在句柄的信息
driver.switch_to.window(driver.window_handles[1])  #切换句柄为1的新窗体
driver.close()  #关闭当前句柄所在窗体
driver.quit()  #所有浏览器退出
(10)弹窗处理

鼠标右击查看不了的弹出框就是alert
driver.switch_to.alert.send_keys(“test”):输入信息
print(driver.switch_to.alert.text) :获取信息
driver.switch_to.alert.accept() :点击确定【正向信息,如确定、提交】
driver.switch_to.alert.dismiss():点击取消【反向信息,如取消,删除等】

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome() # 初始化webdriver类的对象
driver.get('http://') # 打开项目地址
driver.maximize_window() 

driver.switch_to.alert.send_keys("test")  #输入信息
print(driver.switch_to.alert.text)  # 获取信息
driver.switch_to.alert.accept()  # 点击确定
driver.switch_to.alert.dismiss()  #点击取消

(11)截图

存储截图便于分析
driver.get_screenshot_as_file(“C:/Users/Hades/Desktop/name.png”):截图(当前句柄所在窗体)到真实地址,有路径和名称【png格式】

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome()  # 初始化webdriver类的对象
driver.get('http://101.133.169.100/yuns/index.php')  # 打开项目地址
driver.maximize_window() 

driver.get_screenshot_as_file("C:/Users/Hades/Desktop/name.png")   # 截图(当前句柄所在窗体)到真实地址
(12)iframe网页嵌套

iframe是控件,frame是方法
有id/name且不变化,可用。
driver.switch_to.frame(‘x-URS-iframe’)括号里跟id/name直接切入iframe里【x-URS-iframe是id】

无id/name,先定位元素位置
dd = driver.find_element_by_xpath("//div[@id=‘loginDiv’]/iframe"):获取iframe的id或其他定位信息
driver.switch_to.frame(dd):先定位位置,再定位内部信息 ,

driver.switch_to.parent_frame():从子frame切回到父frame【层层嵌套】
driver.switch_to.default_content():切回主文档

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome()  # 初始化webdriver类的对象
driver.get('https://mail.163.com')  # 打开项目地址
driver.maximize_window() 


dd = driver.find_element_by_xpath("//div[@id='loginDiv']/iframe")  #获取id
driver.switch_to.frame(dd)  #先定位位置,再定位内部信息 

driver.switch_to.frame('x-URS-iframe')  #有id,name 且不变化可用

driver.switch_to.parent_frame()#从子frame切回到父frame
driver.switch_to.default_content()#切回主文档
(13)select下拉选择框

s = driver.find_element_by_id(“j_roomCountList”):找到select控件,将其赋值给s
Select(s).select_by_visible_text(“2间”):定位到可见的文本信息【2间】
Select(s).select_by_index(3):用index定位,类似于列表,从0开始
Select(s).select_by_value(6)::用value的值定位

from selenium.webdriver.support.select import Select  # 导入下拉选择框类
from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome()  # 初始化webdriver类的对象
driver.get('https://ctrip.com')  # 打开项目地址
driver.maximize_window() 

emailname=driver.find_element_by_name('email')
emaiiname.send_keys('testiframe')

s = driver.find_element_by_id("j_roomCountList") # 找到select控件,将其赋值给s
Select(s).select_by_visible_text("2间")  # 定位到可见的文本信息【2间】
Select(s).select_by_index(3)  # 用index定位,类似于列表,从0开始
Select(s).select_by_value(6)  # 用value的值定位

(14)时间控件处理

可输入内容
ele = d.find_element_by_id(‘HD_CheckIn’):定位控件
ele.clear():控件内容清空
ele.send_keys(‘2021-05-16’):重新输入值

时间控件属性readonly='readonly’的
js = “document.getElementById(‘noticeEndTime’).removeAttribute(‘readonly’)” :移除readonly属性
d.execute_script(js) :执行js
d.find_element_by_name(“noticeEndTime”).send_keys(“2019-06-21 10:52:52”) :按照正常时间控件输入处理

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome()  # 初始化webdriver类的对象
driver.get('https://ctrip.com')  # 打开项目地址
driver.maximize_window() 

ele = d.find_element_by_id('HD_CheckIn')  # 定位控件
ele.clear()  # 控件内容清空
ele.send_keys('2021-05-16')  #重新输入值

js = "document.getElementById('noticeEndTime').removeAttribute('readonly')" :移除readonly属性
#js = "document.getElementsByName('noticeEndTime')[0].removeAttribute('readonly')"
#js = "document.getElementsByTagName('input')[0].removeAttribute('readonly')"
d.execute_script(js) :执行js
d.find_element_by_name("noticeEndTime").send_keys("2019-06-21 10:52:52") :按照正常时间控件输入处理
**(15)**滚动条

js=“var q=document.documentElement.scrollTop=10000” # 一个js脚本给10000(足够大的数)滚动到页面最底部
driver.execute_script(js) # 执行js
driver.execute_script(“var q=document.documentElement.scrollTop=0”) # 执行js-滚动到页面最顶部
driver.execute_script(“window.scrollTo(0, document.body.scrollHeight*0.5)”) # 滚动到窗体高度的50%【scrollTo:到哪里;x方向是0,y方向为窗口body高度的50%,】
driver.execute_script(‘window.scrollBy(0,200)’) # 以当前位置为准【相对】(x方向为0,y方向向上偏移200)
driver.execute_script(‘window.scrollTo(0,1500)’) # 以当前位置为准【绝对】(x方向为0,y方向为1500)

from selenium import webdriver # 导入selenium库里的webdriver模块
import time

driver = webdriver.Chrome()  # 初始化webdriver类的对象
driver.get('https://ctrip.com')  # 打开项目地址
driver.maximize_window() 

js="var q=document.documentElement.scrollTop=10000"  # 一个js脚本给10000(足够大的数)滚动到页面最底部
driver.execute_script(js)  # 执行js

js="var q=document.documentElement.scrollTop=0"  # 滚动到页面最顶部
driver.execute_script(js)  # 执行js

driver.execute_script("window.scrollTo(0, document.body.scrollHeight*0.5)")  # 滚动到窗体高度的50%【scrollTo:到哪里;x方向是0,y方向为窗口body高度的50%,】

driver.execute_script('window.scrollBy(0,200)')  # 以当前位置为准【相对】(x方向为0,y方向向上偏移200)

driver.execute_script('window.scrollTo(0,1500)')  # 以当前位置为准【绝对】(x方向为0,y方向为1500)
(16)文件上传

input标签
我们在做selenium自动化时,必须要面临的就是页面元素的定位,如果该上传文件功能的标签类型为input,例如:

以上类型的上传文件功能吗,我们可以直接使用selenium中的元素定位+send_keys()方法,括号内传入文件路径,例如:
driver.find_element_by_xpath("").send_keys(r"c:/test.png")
以上是针对于元素类型为input可直接使用selenium上传。

验证码

用selenium来获取获取cookies。
因为经常会出现验证码,导致我们ui自动化测试工作无法更好开展,那么如何处理这种验证码呢?
selenium常用API_第1张图片

像这种滑块和手机验证码的情况下,无法通过正常办法是没有办法进行处理,那么 一般对这种处理有以下几种思路:
1、 通过接口请求,拿到对应验证码信息
2、 让开发配合把验证码搞成万能验证码
3、 注入cookies

那么如何通过注入cookies的形式来实现呢?
首先我们先通过自动化形式打开我们登录页面,此时我们加一定的延时时间,手动登录后,打印出来cookies信息,此时我们就知道所需要的cookies信息了。

driver = webdriver.Chrome()
driver.get("https://fly.layui.com/user/login/")
time.sleep(30)
cookies = driver.get_cookies()
print (cookies)

复制代码
这样你之前的cookies信息就会存在cookies的json文件中,下次要去使用就直接读文件就好了,具体实现如下:
2、自动化代码的注入cookies办法:

driver = webdriver.Chrome()
driver.get("https://fly.layui.com/user/login/")
time.sleep(3)
with open('cookiesFile.json','r') as filemy:
     cookiesInfo=json.loads(filemy.read())
for cc in range(0,len(cookiesInfo)):
    driver.add_cookie(cookiesInfo[cc])
driver.refresh()

复制代码此时通过refresh后,我们就直接登入了系统,是不是很方便呢?
另外需要注意的是:
77版本的Chrome浏览器:对于 ‘expiry’ 参数,不支持float类型上传,这参数表示过期时间,给浏览器添加时,
浏览器只识别整形的,多少秒。 需要将driver.get_cookies() 得到的Cookies,变为整型,才可做cookie绕过验证码登录。

for i in cookies:
    if 'expiry' in i and i['expiry'] is  not None
        i['expiry']= int(i['expiry'])

如有侵权,邮箱联系,实属抱歉。
此只为学习个人笔记整理,同时如有转载请注明出处。
联系邮箱:[email protected]
一同学习测开技企鹅群(闲聊,水群,广告勿扰):826471103

你可能感兴趣的:(selenium,python,测试工具)