05-对混合app应用中的元素进行定位

        本文介绍对于混合app应用中的元素如何进行定位。

一、app的类型

1)Native App(原生应用)

        原生应用是指利用Android、IOS平台官方的开发语言、开发类库、工具等进行开发的app应用,在应用性能和交互体验上应该是最好的。

        通俗点来讲,原生开发就像盖房子一样,先打地基然后浇地梁、房屋结构、一砖一瓦、钢筋水泥、电路走向等,都是经过精心的设计。原生app也一样:将每个页面、每个功能、每个效果、每个逻辑、每个步骤全部用代码一层层、一段段的写出来。

优点:

  • 可访问手机所有功能、可实现功能最齐全。
  • 运行速度快、性能高,绝佳的用户体验。
  • 支持大量图形和动画。不卡,反应快。
  • 比较快捷地使用设备端提供的接口,处理速度上有优势。

缺点:

  • 在过去主要是成本高、周期长,Android和iOS都需要单独开发。
  • 更新版本需要重新下载安装包。        
2)Web App(Web版应用)

        Web版应用是利用Web技术进行开发的,通过浏览器进行访问。Web版应用需要浏览器的支持才能进行展示和用户交互,因此主要用到的技术是HTML5、Javascript、CSS等。HTML5支持一些新标签和脚本,可以做出类似原生应用的效果和动画。

优势:

  • 支持范围广。
  • 开发成本低、周期短。

缺点:

  • 对联网要求高,离线不能做任何操作。
  • 功能有限。
  • 运行速度慢,页面不能承载太多东西。
  • 图片和动画支持性不高。
  • 如果用户使用更多的新型浏览器,那么运行过程中容易出现问题。
3)Hybrid App(混合应用)

        混合应用是利用了原生app的开发技术和HTML5技术进行开发的app应用,是原生和HTML5技术的混合应用,混合比例不限。

        混合开发是一种取长补短的开发模式,原生代码中利用Web View插件或者其它框架为H5提供容器,程序主要的业务实现、界面展示都是利用与H5相关的Web技术进行实现的。比如京东、淘宝、今日头条等app都是利用混合开发模式而成的。

优点:

  • 开发周期短。
  • 功能更新发布快。

缺点:

  • 用户体验不如原生应用。
  • 性能稍慢(需要连接网络)。

二、webview的概述

        现在大部分app都是混合式的native+webview,对于native上的元素通过uiautomatorviewer很容易定位到,而对于webview上的元素就无法识别了。

1)什么是webview?

        WebView直译是网页视图,是一个基于webkit内核的,用于显示网页的控件,具备渲染Web页面的功能。Android的Webview在低版本和高版本采用了不同的webkit版本内核,4.4后直接使用了Chrome浏览器的内核。

        WebView可以将Web网页内嵌在移动端,实现前端的混合式开发,大多数混合式开发框架都是基于WebView模式进行二次开发的,比如:uni-app、Hybrid app等。

2)webview的作用
  • 显示和渲染Web页面。
  • 直接使用html文件(网络上或本地assets中)作布局。
  • 可以和JavaScript交互调用。

        WebView控件功能强大,除了具有一般View的属性和设置外,还可以对url请求、页面加载、渲染、页面交互进行强大的处理。

3)webview的优点
  • 一次开发,多系统适配,节省人力成本和时间成本。

        用H5开发的页面内容可以跨平台使用。无论你用的安卓还是苹果,你的淘宝app里都是H5页面。也就是一次开发,多系统适配,是不是节省了人力成本和时间成本?

        如果不用混合开发这种方式,就得找安卓工程师开发安卓端的、苹果IOS工程师开发苹果的,需要开发两次,是不是很麻烦?

  • Web更新方式为线上即时更新,不用下载安装补丁包。

        你想想,你玩王者或吃鸡吗?这些游戏里的游戏内容都是原生开发的,每次更新你都得先下载安装包吧?当然,打开游戏经常会有一些类似活动公告、说明公告等一些窗口,这些可能也是由WebView嵌入的Web页面。也就是说,原生app里嵌入的Web内容所占比例可大可小,根据需求,对于适合WebView的模块,我们就可以选择用WebView实现它。

        而WebView中嵌入的网页更新需要下载安装包吗?更新后的内容即时上线,根本不需要下载安装包,这样对于用户来讲就很舒服。

  • H5性能不断提高。

  实际上,H5页面的交互、渲染性能肯定是比不上原生的。但随着H5技术的加快发展,加上5G网络的普及化,性能这一块就逐渐不再成问题了。典型的案例就是电商类app应用,性能差距基本看不出来,用户体验还是很舒服的。

        当然,像游戏这类应用,牵扯到渲染引擎性能的问题,对于这点,Web渲染引擎还是不太够用的,对比安卓原生或IOS原生应用,效果太差。

三、识别webview

        用uiautomatorviewer定位工具查看页面,发现页面上有些区域是无法定位到的,如下图左边红色区域,只能定位到这个大红框,红框里面的元素是无法识别的。

        查看该红色框的属性,它的class属性值为WebView,那毫无疑问这种页面就是webview了。

四、获取contexts

        contexts是上下文环境,类似于selenium中的句柄(handle),用于区分不同的环境。

        获取应用当前显示页面的contexts上下文,如下图红框内容所示,获取到的是一个list列表:

  • NATIVE_APP:这个就是native。
  • WEBVIEW_com.xxxx:这个就是webview。

        当看到这样的打印结果,就说明获取到了webview的上下文了(当然有的app有坑,明明有webview,却通过contexts获取不到, 那是开发没打开webview调试)。

from appium import webdriver
import time

# 设置启动参数
desired_cap = {}
desired_cap['platformName'] = 'Android'
desired_cap['platformVersion'] = '6.0.1'
desired_cap['deviceName'] = '127.0.0.1:7555'
# 必须参数,指定被测软件的包名
desired_cap['appPackage'] = 'com.android.browser'
# 必须参数,指定要打开app的哪个页面
desired_cap['appActivity'] = '.BrowserActivity'
desired_cap['automationName']='Uiautomator2'
desired_cap['noReset'] = True
desired_cap['newCommandTimeout'] = 6000
desired_cap['unicodeKeyboard'] = True
desired_cap['resetKeyboard'] = True

driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_cap)
time.sleep(5)

# 获取当前页面所有的上下文环境
print(driver.contexts)

五、从native切换到webview

        要想操作webview上的元素,第一步需要切换环境(与selenium中切换iframe、切换handle思路是一样的)。

        切换方法是:switch_to.context(参数是webview的上下文)

        根据上述打印结果可知,获取到contexts是一个list列表对象,所以这里取这个列表中的第二个参数即可,也就是contexts[1]。

from appium import webdriver
import time

# 设置启动参数
desired_cap = {}
desired_cap['platformName'] = 'Android'
desired_cap['platformVersion'] = '6.0.1'
desired_cap['deviceName'] = '127.0.0.1:7555'
# 必须参数,指定被测软件的包名
desired_cap['appPackage'] = 'com.android.browser'
# 必须参数,指定要打开app的哪个页面
desired_cap['appActivity'] = '.BrowserActivity'
desired_cap['automationName']='Uiautomator2'
desired_cap['noReset'] = True
desired_cap['newCommandTimeout'] = 6000
desired_cap['unicodeKeyboard'] = True
desired_cap['resetKeyboard'] = True

driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_cap)
time.sleep(5)

# 1.获取当前页面所有的上下文环境
all_context = driver.contexts
print(all_context)

# 2.从native切换到webview
driver.switch_to.context(all_context[1])

# 3.打印当前的上下文环境,查看是否切换成功
print(driver.current_context)
1)报错分析

        执行上面程序,报错:Unable to automate Chrome version because it is too old for this version of Chromedriver.

        Android的webview使用的是chrome浏览器的内核,这里的报错是说webview的chrome版本号和chromedriver驱动的版本号不匹配。

2)解决办法
  • 首先,查看webview的chrome版本号是多少?

        在电脑chrome浏览器中输入:chrome://inspect/#devices,点开手机设备的浏览器,随便打开一个网页,点击电脑chrome浏览器的刷新按钮,就会出现webview的chrome版本号信息。

  • 然后,下载chrome版本号相对应的chromedriver驱动。

        找到webview的chrome版本号之后,下载对应的chromedriver驱动就可以了。下载地址:chromedriver.storage.googleapis.com/index.html。

  • 最后,将不匹配的chromedriver驱动进行替换。

3)执行结果

        解决报错之后,重新执行代码,执行结果如下:

六、从webview切换到native

        在webview上操作完之后,想返回到native上操作,这时候需要先切回来,切回native有两个方法:

  • driver.switch_to.context("NATIVE_APP") # 这个NATIVE_APP是固定的参数
  • driver.switch_to.context(contexts[0])        # 从contexts列表中取第一个参数
from appium import webdriver
import time

# 设置启动参数
desired_cap = {}
desired_cap['platformName'] = 'Android'
desired_cap['platformVersion'] = '6.0.1'
desired_cap['deviceName'] = '127.0.0.1:7555'
# 必须参数,指定被测软件的包名
desired_cap['appPackage'] = 'com.android.browser'
# 必须参数,指定要打开app的哪个页面
desired_cap['appActivity'] = '.BrowserActivity'
desired_cap['automationName']='Uiautomator2'
desired_cap['noReset'] = True
desired_cap['newCommandTimeout'] = 6000
desired_cap['unicodeKeyboard'] = True
desired_cap['resetKeyboard'] = True

driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_cap)
time.sleep(5)

# 1.获取当前页面所有的上下文环境
all_context = driver.contexts
print(all_context)

# 2.从native切换到webview
driver.switch_to.context(all_context[1])

# 3.打印当前的上下文环境,查看是否已切换到webview
print(driver.current_context)

# 4.从webview切换到native
driver.switch_to.context(all_context[0])

# 5.打印当前的上下文环境,查看是否已切换回native
print(driver.current_context)

七、使用DevTools工具对webview上的元素进行定位

        app应用中webview内嵌的页面是基于chrome浏览器的内核加载的,在电脑上的chrome浏览器中有个开发者调试工具DevTools,可以很方便的对webview上的元素进行定位。

1)启动DevTools工具
  1. 电脑上打开chrome浏览器,输入:chrome://inspect/#devices。
  2. Discover USB devices 这个要勾选的,可以检测到手机设备。
  3. Discover network targets 这个要勾选的,可以检测到网络。
  4. 下图箭头指向的,是手机的设备名称。
  5. WebView in com.android.browser (91.0.4472.114) 这个是手机浏览器中的chrome内核版本号。

2)定位元素

        上图红框中的地址就是电脑chrome浏览器加载到的手机设备中webview的网页地址,点击地址下的inspect按钮,进入调试界面。

        接下来的元素定位就跟selenium的定位是一回事了。

3)示例代码

        这里首先定位webview中的百度搜索输入框,然后输入内容,最后点击 “百度一下” 按钮。

from appium import webdriver
import time

# 设置启动参数
desired_cap = {}
desired_cap['platformName'] = 'Android'
desired_cap['platformVersion'] = '6.0.1'
desired_cap['deviceName'] = '127.0.0.1:7555'
# 必须参数,指定被测软件的包名
desired_cap['appPackage'] = 'com.android.browser'
# 必须参数,指定要打开app的哪个页面
desired_cap['appActivity'] = '.BrowserActivity'
desired_cap['automationName']='Uiautomator2'
desired_cap['noReset'] = True
desired_cap['newCommandTimeout'] = 6000
desired_cap['unicodeKeyboard'] = True
desired_cap['resetKeyboard'] = True

driver = webdriver.Remote('http://localhost:4723/wd/hub', desired_cap)
time.sleep(5)

# 1.获取当前页面所有的上下文环境
all_context = driver.contexts
print(all_context)

# 2.从native切换到webview
driver.switch_to.context(all_context[1])

# 3.打印当前的上下文环境,查看是否已切换到webview
print(driver.current_context)

# 4.定位输入框后,输入内容,并进行搜索
driver.find_element('id', 'index-kw').clear()
driver.find_element('id', 'index-kw').send_keys('python')
driver.find_element('id', 'index-bn').click()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/724410.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【论文精读】RayMVSNet

今天读的是发表在CVPR2022上的无监督MVS文章,作者来自于国防科大。 文章链接:RayMVSNet 项目地址:Github Abstract 作者希望直接优化每个camera ray上的深度值,所以提出这个RayMVSNet来学习1D implicit field的序列预测。使用了…

springboot+vue+mybatis酒店管理系统+PPT+论文+讲解+售后

基于Spring框架的小型宾馆旅客信息管理系统采用B/S结构、java开发语言、以及Mysql数据库等技术。系统主要分为管理员和用户二部分,管理员:首页、个人中心、用户管理、客房类型管理、客房信息管理、客房预订管理、入住登记管理、退房评价管理、系统管理&a…

基于Redis提高查询性能(保持数据一致性)

Redis实战篇 | Kyles Blog (cyborg2077.github.io) 目录 背景 商户查询缓存(根据ID查询) 根据店铺类型查询(List型) 缓存更新策略(保证数据一致性) 案例(利用缓存更新策略) 背景 起初客户端…

2024-06-19,面试官问的问题

文章目录 1、采用minIO完成了图片存储,采用阿里云OSS服务器存储图片这两个功能面试官理解为重复,面试官又问minIO怎么同步到OSS?2、讲一下ThreadLocal?3、为什么用ThreadLocal存数据?4、redis有几种数据结构&#xff1…

Office--加载宏-CS上线

免责声明:本文仅做技术交流与学习... 目录 关于宏上线的格式: 操作流程: 其他office文本的上线格式一样: 关于宏上线的格式: doc宏病毒: 1-生成格式为dotm 只要点启用宏就上线 2-生成格式为()word 97-2003 .doc) 被杀 操作流…

13.1.资源清单的管理工具-helm

目录 一、helm的介绍 1.helm的价值概述 2.helm的关键名词 二、安装部署helm 1.解压安装包并设置全局命令 2.添加命令补全 三、使用helm部署服务管理 1.使用helm创建chart 1.1创建工作目录 1.2.helm创建chart 2.响应式创建名称空间 3.安装chart到名称空间 4.使用hel…

机器人运动学笔记

一、建模 参考资料:https://zhuanlan.zhihu.com/p/137960186 1、三维模型和连杆、关节定义 2、设置z轴 SDH和MDH会不一样,主要的区别在于SDH中坐标系在连杆末端,MDH中坐标系在连杆首端。虽然这里只是给出z轴,但是由于后面原点位…

鸿蒙开发通信与连接:【@ohos.connectedTag (有源标签)】

有源标签 说明: 本模块首批接口从API version 8开始支持。后续版本的新增接口,采用上角标单独标记接口的起始版本。 导入模块 import connectedTag from ohos.connectedTag;connectedTag.init init(): boolean 初始化有源标签芯片。 需要权限&#…

Python学习笔记14:进阶篇(三)。类的终结篇,类的导入和模块的导入。

前言 这篇文章属于类知识的最后一篇,带一点点其他知识,学习内容来自于Python crash course。 关注我私信发送Python crash course,分享一份中文版PDF。 类的导入 在学习的时候,包括之前,我都是在一个文件中把所有代…

48-2 内网渗透 - 利用Metasploit提权

一、Metasploit提权过程概述 Metasploit是一个开源的安全漏洞检测工具,广泛用于安全和IT专业人士识别、验证和利用安全漏洞,同时也支持专家驱动的安全评估和管理。 提权过程详解 1)生成后门 在Kali Linux上使用msfvenom生成反向连接的后门文件。 # ip 要改成自…

第七节 文本框的默认输入设置

通常我们常见设置一些文本框,默认提示一段信息,在文本框输入值变化值切换实际输入值,那么这时怎么添加情形进行设置呢,请看下面这个案例说明。 第一种,添加文本框不为空,显示为当前输入值 1、添加文本框&…

什么是进程

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 在了解进程之前,我们需要知道多任务的概念。多任务,顾名思义,就是指操作系统能够执行多个任务。例如,…

为什么带上符号位计算,仍然可以算出正负数对应的补码

文章目录 🚀前言🚀 为什么要引入原反补码✈️ 利用加法表示减法✈️ 关于数字在计算机中的存储🚀 数学解释 🚀 为什么带上符号位计算,仍然可以算出正负数对应的补码 🚀前言 为应付期末,速成数电…

ROS 机器人运动控制

ROS 机器人运动控制 机器人运动 当我们拿到一台机器人,其配套的程序源码中,通常会有机器人核心节点,这个核心节点既能够驱动机器人的底层硬件,同时向上还会订阅一个速度话题。我们只需要编写一个新的节点(速度控制节点…

(四十二)Vue之路由及其基本使用Vue Router

文章目录 概念基本使用demo几个注意点 上一篇:(四十一)Vuex之模块化与命名空间 概念 在计算机科学和网络领域中,路由(Routing)是指确定数据包在网络中传输的路径和方式的过程。它涉及到根据源地址和目标地…

杂说咋说-关于城市化发展和城市治理的几点建议(浙江借鉴)

杂说咋说-关于城市化发展和城市治理的几点建议(浙江借鉴) 近年来,浙江省坚持一张蓝图绘到底,推动城市化发展和城市治理不断迈上新台阶,全省城市化水平和城市治理能力牢牢居于全国第一方阵。当前,国内外环境…

DFS(一)

问题一(指数级选择) 从1~n这n个整数中任意选取多个,输出所有可能的选择方案。 首先想一下,在现实世界中,我们要如何解决这个问题。 应该是一个一个枚举,即每个数都可以有两个选择(选/不选)。共有种结果。 想一下,如…

JavaSE 面向对象程序设计高级 方法引用 2024详解

在编程中,方法引用(Method Reference)是一种技术,它让你能够直接引用一个现有的函数或方法,而无需通过对象实例来调用。这种方法在函数式编程和高阶函数中非常有用,因为它提供了简洁的方式来传递函数行为&a…

洛谷 P1726:上白泽慧音 ← Tarjan算法

【题目来源】https://www.luogu.com.cn/problem/P1726【题目描述】 在幻想乡,上白泽慧音是以知识渊博闻名的老师。春雪异变导致人间之里的很多道路都被大雪堵塞,使有的学生不能顺利地到达慧音所在的村庄。因此慧音决定换一个能够聚集最多人数的村庄作为新…

【Java】已解决java.util.concurrent.RejectedExecutionException异常

文章目录 一、问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项 已解决java.util.concurrent.RejectedExecutionException异常 一、问题背景 java.util.concurrent.RejectedExecutionException是Java并发编程中常见的一个异常,它通常发生…