企业🤖AI Agent构建引擎,智能编排和调试,一键部署,支持私有化部署方案 广告
[TOC] ## **一、selenium中如何判断元素是否存在?** 首先selenium里面是没有这个方法的,判断元素存在需要自己写一个方法了。 元素存在有几种形式,一种是页面有多个元素属性重复的,这种直接操作会报错的;还有一种是页面隐藏的元素操作也会报错 这个可以说是被问烂的题了,判断元素存在方法有三种: * 方法一,用try...except... ~~~ def is_element_exsist(driver, locator): ''' 判断元素是否存在,存在返回True,不存返回False :param locator: locator为元组类型,如("id", "yoyo") :return: bool值,True or False ''' try: driver.find_element(*locator) return True except Exception as msg: print("元素%s找不到:%s" % (locator, msg)) return False if __name__ == '__main__': loc1 = ("id", "yoyo") # 元素1 print(is_element_exsist(driver, loc1)) ~~~ * 方法二:用elements定义一组元素方法 ~~~ def is_element_exsist1(driver, locator): ''' 判断元素是否存在,存在返回True,不存返回False :param locator: locator为元组类型,如("id", "yoyo") :return: bool值,True or False ''' eles = driver.find_elements(*locator) if len(eles) < 1: return False else: return True if __name__ == '__main__': loc1 = ("id", "yoyo") # 元素1 print(is_element_exsist1(driver, loc1)) ~~~ * (强烈推荐!)方法三:结合WebDriverWait和expected\_conditions判断 ~~~ from selenium import webdriver from selenium.webdriver.support import expected_conditions as EC from selenium.webdriver.support.ui import WebDriverWait def is_element_exsist2(driver, locator): ''' 结合WebDriverWait和expected_conditions判断元素是否存在, 每间隔1秒判断一次,30s超时,存在返回True,不存返回False :param locator: locator为元组类型,如("id", "yoyo") :return: bool值,True or False ''' try: WebDriverWait(driver, 30, 1).until(EC.presence_of_element_located(locator)) return True except: return False if __name__ == '__main__': loc1 = ("id", "yoyo") # 元素1 print(is_element_exsist2(driver, loc1)) ~~~ <br /> ## **二、selenium中hidden或者是display = none的元素是否可以定位到?** 定位是可以定位到的,但是不能操作,可以判断元素is\_displayed() <br /> ## **三、如何提高脚本的稳定性?** 相关类似问题还有“用例在运行过程中经常会出现不稳定的情况,也就是说这次可以通过,下次就没办法通过了,如何去提升用例的稳定性?” “如何提高selenium脚本的执行速度?” “selenium中如何保证操作元素的成功率?也就是说不管网络加载慢还是快” 如果一个元素今天你能定位到,过两天就定位不到了,只要这个页面没变过,说明定位方法是没啥问题的。 优化方向: 1. 不要右键复制xpath(十万八千里那种路径,肯定不稳定),自己写相对路径,多用id为节点查找 2. 定位没问题,第二个影响因素那就是等待了,sleep等待尽量少用(影响执行时间) `driver.implicitly_wait(30)`这个等待也不要用,不要以为是全局的就是好事,有些js加载失败时候会一直等,并且页面跳转时候也无法识别 3. 定位元素方法重新封装,结合WebDriverWait和expected\_conditions判断元素方法,自己封装一套定位元素方法 ~~~ from selenium import webdriver from selenium.webdriver.support import expected_conditions as EC from selenium.webdriver.support.ui import WebDriverWait def find(locator, timeout=30): '''定位元素,参数locator是元祖类型, 如("id", "yoyo")''' element = WebDriverWait(driver, timeout, 1).until(EC.presence_of_element_located(locator)) return element ~~~ <br /> ## **四、selenium中隐藏元素如何定位?(hidden、display: none)** **前言** 面试题:selenium中隐藏元素如何定位?这个是很多面试官喜欢问的一个题,如果单纯的定位的话,隐藏元素和普通不隐藏元素定位没啥区别,用正常定位方法就行了 但是吧~~~很多面试官自己都搞不清楚啥叫定位,啥叫操作元素(如click,clear,send\_keys) ### **隐藏元素** 如下图有个输入框和一个登录的按钮,本来是显示的 ![](http://img2018.cnblogs.com/blog/1070438/201812/1070438-20181224173724546-907774977.png) 元素的属性隐藏和显示,主要是`type="hidden"`和`style="display: none;"`属性来控制的,接下来在元素属性里面让它隐藏 ~~~ <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <p>这里有个按钮,是隐藏的 <!-- type="hidden" --> <br> 输入账号<input id="yoyo" name="hello" type="hidden"> <!-- type="display: none;" --> <br> <button id="yy" name="heo" style="display: none;">登录</button> <br> <a hidden id="baidu" href="https://www.baidu.com">访问百度</a> </p> </body> </html> ~~~ 这样元素就不会显示了,也就是面试官所说的隐藏属性了 ![](https://img.kancloud.cn/ea/f2/eaf2c1d2c012389f3738eaafd13697b5_801x302.png) ### 定位隐藏元素 前面说了,定位隐藏元素和普通的元素没啥区别,接下来就来验证下,是不是能定位到呢? ~~~ from selenium import webdriver driver = webdriver.Firefox() driver.get("http://localhost:63342/test1122/a/b.html") # 定位type="hidden"隐藏元素 ele1 = driver.find_element_by_id("yoyo") print("打印元素信息:%s" % ele1) # 获取元素属性 print(ele1.get_attribute("name")) # 判断元素是否隐藏 print(ele1.is_displayed()) ~~~ 运行结果: ~~~ 打印元素信息:<selenium.webdriver.remote.webelement.WebElement (session="1debdd46-21b1-451e-b8a7-5aeff1d74f9d", element="{28628a87-7f22-4574-9e14-931f9c6f20e1}")> hello False ~~~ **运行结果可以看出,隐藏元素用普通定位方法,事实上是定位到了呢!** ### **操作隐藏元素** 隐藏元素可以正常定位到,只是不能操作(定位元素和操作元素是两码事,很多初学者傻傻分不清楚),操作元素是click,clear,send\_keys这些方法 ~~~ # 隐藏输入框元素输入文本 ele1 = driver.find_element_by_id("yoyo") ele1.send_keys("yoyo") ~~~ 隐藏元素用send\_keys()方法会抛异常'ElementNotVisibleException': Message: Element is not currently visible and so may not be interacted with 这个报错是说元素不可见,不可以被操作,同样的对“登录”按钮点击操作也是会报'ElementNotVisibleException' ~~~ # 点击隐藏登录框 ele2 = driver.find_element_by_id("yy") ele2.click() ~~~ ### **JS操作隐藏元素** 如果面试官想问的是定位后操作隐藏元素的话,本质上说这个问题就是毫无意义的,web自动化的目的是模拟人的正常行为去操作。 如果一个元素页面上都看不到了,你人工也是无法操作的是不是?人工都不能操作,那你自动化的意义又在哪呢?所以这个只是为了单纯的考察面试者处理问题的能力,没啥实用性!(面试造飞机,进去拧螺丝) 既然面试官这么问了,那就想办法回答上给个好印象吧! 首先selenium是无法操作隐藏元素的(但是能正常定位到),本身这个框架就是设计如此,如果非要去操作隐藏元素,那就用js的方法去操作,selenium提供了一个入口可以执行js脚本。 js和selenium不同,只有页面上有的元素(在dom里面的),都能正常的操作,接下来用js试试吧! `<a hidden id="baidu" href="https://www.baidu.com">访问百度</a>`这个链接是隐藏的,但是能用js点到 ~~~ from selenium import webdriver driver = webdriver.Firefox() driver.get("http://localhost:63342/test1122/a/b.html") # js点击hidden元素 js = 'document.getElementById("baidu").click()' driver.execute_script(js) ~~~ 运行完之后,会发现页面正常的点击,跳转到百度页面了 备注:百度搜到的可能方法是先用js去掉hidden属性,再用selenium操作,这个有点多此一举,你既然都已经会用js了,何必不一次性到位直接click呢? <br /> ## 五、你的自动化用例的执行策略是什么? 1. 集成到jenkins一键执行,可以手动执行,也可以定时执行 <br /> ## 六、什么是持续集成? 1.持续集成是一种软件开发实践,即团队开发成员经常集成它们的工作,通过每个成员每天至少集成一次,也就意味着每天可能会发生多次集成。 每次集成都通过自动化的构建(包括编译,发布,自动化测试)来验证,从而尽早地发现集成错误。 <br /> ## 七、自动化测试的时候是不是需要连接数据库做数据校验? 1. UI自动化不需要 2. 接口测试会需要 <br /> ## 八、id,name,class,xpath, css selector这些属性,你最偏爱哪一种,为什么? 1.css妥妥的,css语法简洁,定位快(xpath语法长,定位慢,还不稳定) <br /> ## 九、如何定位动态元素? 动态元素有2种情况,一个是属性动态,比如id是动态的,定位时候,那就不要用id定位就是了 ~~~ <p id="yo" class="hello world"> <button id="yy_auto_1929292" name="heo" >登录</button> <br> </p> ~~~ 比如上面这个button元素,id是动态的,定位方法千千万,何必死在id上,可以用name定位, 哪怕这个元素属性都是动态的,它的标签不可能动态吧,那就定位父元素id="yo"啊:`.//*[@id='yo']/button` 还有一种情况动态的,那就是这个元素一会在页面上方,一会在下方,飘忽不定的动态元素,定位方法也是一样,按f12,根据元素属性定位(元素的tag、name的步伐属性是不会变的,动的只是class属性和styles属性) <br /> ## 十、点击链接以后,selenium是否会自动等待该页面加载完毕? 1. 这个不会等的,没加载完也可以下一步操作 <br /> ## 十一、webdriver client的原理是什么? 通过研究selenium-webdriver的源码,发现其实webdriver的实现原理并不高深莫测无法揣度。在这里以webdriver **ruby binding**的**firefox-webdriver**实现为例,简单介绍一下webdriver的工作原理。 * 当测试脚本启动firefox的时候,selenium-webdriver 会首先在新线程中启动firefox浏览器。如果测试脚本指定了firefox的profile,那么就以该profile启动,否则的话就新启1个profile,并启动firefox; * firefox一般是以-no-remote的方法启动,启动后selenium-webdriver会将firefox绑定到特定的端口,绑定完成后该firefox实例便作为webdriver的remote server存在; * 客户端(也就是测试脚本)创建1个session,在该session中通过http请求向remote server发送restful的请求,remote server解析请求,完成相应操作并返回response; * 客户端接受response,并分析其返回值以决定是转到第3步还是结束脚本; 这就是webdriver的工作流程,看起来很复杂实际上当了解了webdriver的实现原理后,理解上述问题应该比较简单。 webdriver是按照server – client的经典设计模式设计的。 server端就是remote server,可以是任意的浏览器。当我们的脚本启动浏览器后,该浏览器就是remote server,它的职责就是等待client发送请求并做出相应; client端简单说来就是我们的测试代码,我们测试代码中的一些行为,比如打开浏览器,转跳到特定的url等操作是以http请求的方式发送给被 测试浏览器,也就是remote server;remote server接受请求,并执行相应操作,并在response中返回执行状态、返回值等信息; <br /> ## 十二、启动浏览器的时候用到的是哪个webdriver协议? 1. http <br /> ## 十三、什么是page object设计模式? 1. 通俗来讲,就是页面元素放到一个脚本,封装页面元素操作方法,每个页面但是一个page对象方便页面元素的维护 2. 测试用例从这个页面对象脚本里面去调用对应方法 **page object设计模式中,如何实现页面的跳转?** 1. click方法 <br /> ## 十四、怎样去选择一个下拉框中的value=xx的option? 1. select类里面提供的方法:select\_by\_value(“xxx”) 2. xpath的语法也可以定位到 <br /> ## 十五、什么是断言? assert,判断测试结果与期望结果是否一致 <br />