当前位置: 首页 > news >正文

Selenium自动化测试框架(超详细总结分享)

设计思路

本文整理归纳以往的工作中用到的东西,现汇总成基础测试框架提供分享。

框架采用python3 + selenium3 + PO + yaml + ddt + unittest等技术编写成基础测试框架,能适应日常测试工作需要。

1、使用Page Object模式将页面定位和业务操作分开,分离测试对象(元素对象)和测试脚本(用例脚本),一个页面建一个对象类,提高用例的可维护性;

2、使用yaml管理页面控件元素数据和测试用例数据。例如元素ID等发生变化时,不需要去修改测试代码,只需要在对应的页面元素yaml文件中修改即可;

3、分模块管理,互不影响,随时组装,即拿即用。

GitHub项目地址:GitHub - yingoja/DemoUI: selenium UI自动化测试框架

测试框架分层设计

  • 把常见的操作和查找封装成基础类,不管是什么产品,可直接拿来复用
  • 业务层主要是封装对象页面类,一个页面建一个类,业务层页面继承基础层
  • 用例层针对产品页面功能进行构造摸拟执行测试
  • 框架层提供基础组件,支撑整个流程执行及功能扩展,给用例层提供各页面的元素数据、用例测试数据,测试报告输出等

测试框架目录结构


如下思维导图目录结构介绍:

在这我为大家准备了一份软件测试视频教程(含面试、接口、自动化、性能测试等),就在下方,需要的可以直接去观看,也可以直接【点击文末小卡片免费领取资料文档】

软件测试视频教程观看处:

软件测试工程师大忌!盲目自学软件测试真的会毁终生,能救一个是一个......

编写用例方法

 1 testinfo:2       - id: test_login0013         title: 登录测试4         info: 打开抽屉首页5 testcase:6       - element_info: login-link-a7         find_type: ID8         operate_type: click9         info: 打开登录对话框
10       - element_info: mobile
11         find_type: ID
12         operate_type: send_keys
13         info: 输入手机号
14       - element_info: mbpwd
15         find_type: ID
16         operate_type: send_keys
17         info: 输入密码
18       - element_info: //input[@class='keeplogin']
19         find_type: XPATH
20         operate_type: click
21         info: 单击取消自动登录单选框
22       - element_info: //span[text()='登录']
23         find_type: XPATH
24         operate_type: click
25         info: 单击登录按钮
26       - element_info: userProNick
27         find_type: ID
28         operate_type: perform
29         info: 鼠标悬停账户菜单
30       - element_info: //a[@class='logout']
31         find_type: XPATH
32         operate_type: click
33         info: 选择退出
34 check:
35       - element_info: //div[@class='box-mobilelogin']/div[1]/span
36         find_type: XPATH
37         info: 检查输入手机号或密码,登录异常提示
38       - element_info: userProNick
39         find_type: ID
40         info: 成功登录
41       - element_info: reg-link-a
42         find_type: ID
43         info: 检查退出登录是否成功

例如,我们要新增登录功能测试用例:首先,只需在testyaml目录下新增一个页面对象yaml文件,参考login.yaml格式编写即可。这些文件是提供给封装页面对象类调用并执行定位识别操作。

 1 -2   id: test_login001.13   detail : 手机号和密码为空登录4   screenshot : phone_pawd_empty5   data:6     phone: ""7     password: ""8   check :9      - 手机号不能为空
10 -
11   id: test_login001.2
12   detail : 手机号为空登录
13   screenshot : phone_empty
14   data :
15     phone: ""
16     password : aa
17   check :
18     - 手机号不能为空
19 -
20   id: test_login001.3
21   detail : 密码为空登录
22   screenshot : pawd_empty
23   data :
24     phone : 13511112222
25     password: ""
26   check :
27     - 密码不能为空
28 -
29   id: test_login001.4
30   detail : 非法手机号登录
31   screenshot : phone_error
32   data :
33     phone : abc
34     password: aa
35   check :
36     - 手机号格式不对
37 -
38   id: test_login001.5
39   detail : 手机号或密码不匹配
40   screenshot : pawd_error
41   data :
42     phone : 13511112222
43     password: aa
44   check :
45     - 账号密码错误
46 -
47   id: test_login001.6
48   detail : 手机号和密码正确
49   screenshot : phone_pawd_success
50   data :
51     phone : 13865439800
52     password: ********
53   check :
54     - yingoja
55 
56 login_data.yaml

其次,在testdata目录下新增一个login_data.yaml文件提供给登录接口传参的测试数据,编写格式参考login_data.yaml文件。

 1 #!/usr/bin/env python2 # _*_ coding:utf-8 _*_3 __author__ = 'YinJia'4 5 import os,sys6 sys.path.append(os.path.dirname(os.path.dirname(os.path.dirname(__file__))))7 from config import setting8 from selenium.webdriver.support.select import Select9 from selenium.webdriver.common.action_chains import ActionChains10 from selenium.webdriver.common.by import By11 from public.page_obj.base import Page12 from time import sleep13 from public.models.GetYaml import getyaml14 15 testData = getyaml(setting.TEST_Element_YAML + '/' + 'login.yaml')16 17 class login(Page):18     """19     用户登录页面20     """21     url = '/'22     dig_login_button_loc = (By.ID, testData.get_elementinfo(0))23     def dig_login(self):24         """25         首页登录26         :return:27         """28         self.find_element(*self.dig_login_button_loc).click()29         sleep(1)30 31     # 定位器,通过元素属性定位元素对象32     # 手机号输入框33     login_phone_loc = (By.ID,testData.get_elementinfo(1))34     # 密码输入框35     login_password_loc = (By.ID,testData.get_elementinfo(2))36     # 取消自动登录37     keeplogin_button_loc = (By.XPATH,testData.get_elementinfo(3))38     # 单击登录39     login_user_loc = (By.XPATH,testData.get_elementinfo(4))40     # 退出登录41     login_exit_loc = (By.ID, testData.get_elementinfo(5))42     # 选择退出43     login_exit_button_loc = (By.XPATH,testData.get_elementinfo(6))44 45     def login_phone(self,phone):46         """47         登录手机号48         :param username:49         :return:50         """51         self.find_element(*self.login_phone_loc).send_keys(phone)52 53     def login_password(self,password):54         """55         登录密码56         :param password:57         :return:58         """59         self.find_element(*self.login_password_loc).send_keys(password)60 61     def keeplogin(self):62         """63         取消单选自动登录64         :return:65         """66         self.find_element(*self.keeplogin_button_loc).click()67 68     def login_button(self):69         """70         登录按钮71         :return:72         """73         self.find_element(*self.login_user_loc).click()74 75     def login_exit(self):76         """77         退出系统78         :return:79         """80         above = self.find_element(*self.login_exit_loc)81         ActionChains(self.driver).move_to_element(above).perform()82         sleep(2)83         self.find_element(*self.login_exit_button_loc).click()84 85     def user_login(self,phone,password):86         """87         登录入口88         :param username: 用户名89         :param password: 密码90         :return:91         """92         self.open()93         self.dig_login()94         self.login_phone(phone)95         self.login_password(password)96         sleep(1)97         self.keeplogin()98         sleep(1)99         self.login_button()
100         sleep(1)
101 
102     phone_pawd_error_hint_loc = (By.XPATH,testData.get_CheckElementinfo(0))
103     user_login_success_loc = (By.ID,testData.get_CheckElementinfo(1))
104     exit_login_success_loc = (By.ID,testData.get_CheckElementinfo(2))
105 
106     # 手机号或密码错误提示
107     def phone_pawd_error_hint(self):
108         return self.find_element(*self.phone_pawd_error_hint_loc).text
109 
110     # 登录成功用户名
111     def user_login_success_hint(self):
112         return self.find_element(*self.user_login_success_loc).text
113 
114     # 退出登录
115     def exit_login_success_hint(self):
116         return self.find_element(*self.exit_login_success_loc).text

然后,在page_obj目录下新增一个loginPage.py文件,是用来封装登录页面对象类,执行登录测试流程操作。

 1 #!/usr/bin/env python2 # _*_ coding:utf-8 _*_3 __author__ = 'YinJia'4 5 6 import os,sys7 sys.path.append(os.path.dirname(os.path.dirname(__file__)))8 import unittest,ddt,yaml9 from config import setting
10 from public.models import myunit,screenshot
11 from public.page_obj.loginPage import login
12 from public.models.log import Log
13 
14 try:
15     f =open(setting.TEST_DATA_YAML + '/' + 'login_data.yaml',encoding='utf-8')
16     testData = yaml.load(f)
17 except FileNotFoundError as file:
18     log = Log()
19     log.error("文件不存在:{0}".format(file))
20 
21 @ddt.ddt
22 class Demo_UI(myunit.MyTest):
23     """抽屉新热榜登录测试"""
24     def user_login_verify(self,phone,password):
25         """
26         用户登录
27         :param phone: 手机号
28         :param password: 密码
29         :return:
30         """
31         login(self.driver).user_login(phone,password)
32 
33     def exit_login_check(self):
34         """
35         退出登录
36         :return:
37         """
38         login(self.driver).login_exit()
39 
40     @ddt.data(*testData)
41     def test_login(self,datayaml):
42         """
43         登录测试
44         :param datayaml: 加载login_data登录测试数据
45         :return:
46         """
47         log = Log()
48         log.info("当前执行测试用例ID-> {0} ; 测试点-> {1}".format(datayaml['id'],datayaml['detail']))
49         # 调用登录方法
50         self.user_login_verify(datayaml['data']['phone'],datayaml['data']['password'])
51         po = login(self.driver)
52         if datayaml['screenshot'] == 'phone_pawd_success':
53             log.info("检查点-> {0}".format(po.user_login_success_hint()))
54             self.assertEqual(po.user_login_success_hint(), datayaml['check'][0], "成功登录,返回实际结果是->: {0}".format(po.user_login_success_hint()))
55             log.info("成功登录,返回实际结果是->: {0}".format(po.user_login_success_hint()))
56             screenshot.insert_img(self.driver, datayaml['screenshot'] + '.jpg')
57             log.info("-----> 开始执行退出流程操作")
58             self.exit_login_check()
59             po_exit = login(self.driver)
60             log.info("检查点-> 找到{0}元素,表示退出成功!".format(po_exit.exit_login_success_hint()))
61             self.assertEqual(po_exit.exit_login_success_hint(), '注册',"退出登录,返回实际结果是->: {0}".format(po_exit.exit_login_success_hint()))
62             log.info("退出登录,返回实际结果是->: {0}".format(po_exit.exit_login_success_hint()))
63         else:
64             log.info("检查点-> {0}".format(po.phone_pawd_error_hint()))
65             self.assertEqual(po.phone_pawd_error_hint(),datayaml['check'][0] , "异常登录,返回实际结果是->: {0}".format(po.phone_pawd_error_hint()))
66             log.info("异常登录,返回实际结果是->: {0}".format(po.phone_pawd_error_hint()))
67             screenshot.insert_img(self.driver,datayaml['screenshot'] + '.jpg')
68 
69 if __name__=='__main__':
70     unittest.main()

最后,在testcase目录下创建测试用例文件login_sta.py,采用ddt数据驱动读取yaml测试数据文件

综上所述,编写用例方法只需要按以上四个步骤创建->编写即可。执行如下主程序,可看输出的实际结果。

 1 #!/usr/bin/env python2 # _*_ coding:utf-8 _*_3 __author__ = 'YinJia'4 5 import os,sys6 sys.path.append(os.path.dirname(__file__))7 from config import setting8 import unittest,time9 from package.HTMLTestRunner import HTMLTestRunner
10 from public.models.newReport import new_report
11 from public.models.sendmail import send_mail
12 
13 # 测试报告存放文件夹,如不存在,则自动创建一个report目录
14 if not os.path.exists(setting.TEST_REPORT):os.makedirs(setting.TEST_REPORT + '/' + "screenshot")
15 
16 def add_case(test_path=setting.TEST_DIR):
17     """加载所有的测试用例"""
18     discover = unittest.defaultTestLoader.discover(test_path, pattern='*_sta.py')
19     return discover
20 
21 def run_case(all_case,result_path=setting.TEST_REPORT):
22     """执行所有的测试用例"""
23     now = time.strftime("%Y-%m-%d %H_%M_%S")
24     filename =  result_path + '/' + now + 'result.html'
25     fp = open(filename,'wb')
26     runner = HTMLTestRunner(stream=fp,title='抽屉新热榜UI自动化测试报告',
27                             description='环境:windows 7 浏览器:chrome',
28                             tester='Jason')
29     runner.run(all_case)
30     fp.close()
31     report = new_report(setting.TEST_REPORT) #调用模块生成最新的报告
32     send_mail(report) #调用发送邮件模块
33 
34 if __name__ =="__main__":
35     cases = add_case()
36     run_case(cases)

测试结果展示

HTML报告日志

HTML报告点击截图,弹出截图

测试报告通过的日志

自动截图存放指定的目录

邮件测试报告

感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走

这些资料,对于做【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!凡事要趁早,特别是技术行业,一定要提升技术功底。

 

相关文章:

Selenium自动化测试框架(超详细总结分享)

设计思路 本文整理归纳以往的工作中用到的东西,现汇总成基础测试框架提供分享。 框架采用python3 selenium3 PO yaml ddt unittest等技术编写成基础测试框架,能适应日常测试工作需要。 1、使用Page Object模式将页面定位和业务操作分开&#xff…...

STM32 DAC+串口

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、DAC是什么?二、STM32 DAC1.什么型号有DAC2. 简介3. 主要特点4. DAC框图5. DAC 电压范围和引脚 三、程序步骤1. 开启DAC时钟2. 配置引脚 PA4 PA5…...

SolidWorks二次开发 C#-读取基于Excel的BOM表信息

SolidWorks二次开发 C#-读取基于Excel的BOM表信息 问题点来源解决方案及思路相关引用链接 问题点来源 这是一位粉丝问的一个问题,他说到: 老师,请问Solidworks二次开发工程图中"基于Excel的材料明细表"怎么读取里面的数据? Ps:这…...

maui中实现加载更多 RefreshView跟ListView(2)

一个类似商品例表的下拉效果&#xff1a; 代码 新增个类为商品商体类 public class ProductItem{public string ImageSource { get; set; }public string ProductName { get; set; }public string Price { get; set; }}界面代码&#xff1a; <?xml version"1.0&quo…...

win10环境下git安装和基础操作

简述 关于git的作用就不多赘述了&#xff0c;配合GitHub&#xff0c;达到方便人们日常项目维护和管理&#xff0c;每一次项目增删改查都可以看的清清楚楚&#xff0c;方便团队协作和个人项目日常维护。 下载git 首先我们自然是要到官网下载git&#xff0c;下载地址为https:/…...

将yolo格式转化为voc格式:txt转xml(亲测有效)

1.文件目录如下所示&#xff1a; 对以上目录的解释&#xff1a; 1.dataset下面的image文件夹&#xff1a;里面装的是数据集的原图片 2.dataset下面的label文件夹&#xff1a;里面装的是图片对应得yolo格式标签 3.dataset下面的Annotations文件夹&#xff1a;这是一个空文件夹&…...

字符串 - 541.反转字符串II(C#和C实现)

字符串 - 541.反转字符串II(C#和C实现) 题目描述 给定一个字符串 s 和一个整数 k&#xff0c;你需要对从字符串开头算起的每隔 2k 个字符的前 k 个字符进行反转。 如果剩余字符少于 k 个&#xff0c;则将剩余字符全部反转。如果剩余字符小于 2k 但大于或等于 k 个&#xff0…...

机器视觉技术与应用实战(开运算、闭运算、细化)

开运算和闭运算的基础是膨胀和腐蚀&#xff0c;可以在看本文章前先阅读这篇文章机器视觉技术与应用实战&#xff08;Chapter Two-04&#xff09;-CSDN博客 开运算&#xff1a;先腐蚀后膨胀。开运算可以使图像的轮廓变得光滑&#xff0c;具有断开狭窄的间断和消除细小突出物的作…...

云原生之深入解析云原生架构的日志监控

一、什么是云原生架构的日志监控&#xff1f; 云原生架构的日志监控要求现代 Web 应用程序采用与传统应用程序略有不同的方法。部分原因是应用程序环境要复杂得多&#xff0c;包括从微服务中获取数据、使用 Kubernetes 和其他容器技术&#xff0c;以及在许多情况下集成开源组件…...

基于hfl/rbt3模型的情感分析学习研究——文本挖掘

参考书籍《HuggingFace自然语言处理详解 》 什么是文本挖掘 文本挖掘&#xff08;Text mining&#xff09;有时也被称为文字探勘、文本数据挖掘等&#xff0c;大致相当于文字分析&#xff0c;一般指文本处理过程中产生高质量的信息。高质量的信息通常通过分类和预测来产生&…...

计算机网络基础——常用的中英文网络述语大全,强烈建议收藏

系统网络体系结构(System Network Architecture&#xff0c;SNA) 国际标准化组织(International Organization for Standardization&#xff0c;ISO) 开放系统互连基本参考模型(Open System Interconnection Reference Model。OSI/RM) 物理层(Physical Layer) 数据终端设备…...

c++如何自定义类及成员函数

#include <iostream>using namespace std;class Box {public:double length; // 长度double breadth; // 宽度double height; // 高度// 成员函数声明double get(void);void set( double len, double bre, double hei ); }; // 成员函数定义 double Box::get(void) …...

100G云数据中心网络建设解决方案

随着数据和流量的快速增长&#xff0c;近年来数据中心已经进入了一个全新的100G时代。为了更高效地提供包括人工智能、虚拟现实、4K视频等在内的云计算服务&#xff0c;全球范围内正在大规模建设众多大型100G数据中心&#xff0c;如云数据中心。作为一种新型高效的基础设施&…...

Zoho Desk为何受到跨境电商企业青睐:优势与特点解析

现如今&#xff0c;跨境电商已成为中国外贸发展的一支重要力量&#xff0c;正从一种新业态成长为外贸的新常态。越来越多的国内电商玩家加入了跨境电商这个战场。跨境电商自有其特殊性&#xff0c;海外客户服务不好一样惨遭投诉&#xff0c;Zoho Desk可以帮助您赢得客户满意度&…...

git 删除仓库中多余的文件或者文件夹

目录 问题 解决方案 第一步&#xff1a;同步代码 第二步&#xff1a;删除文件 第三步&#xff1a;提交 第四步&#xff1a;推送远端 问题 在项目开发测试阶段&#xff0c;将无意间将本地敏感的、或无用的文件或目录不小心提交到远程仓库&#xff0c;该怎么解决呢。 解决方…...

搭建git服务器(本地局域网)

搭建git服务器&#xff08;本地局域网&#xff09; 创建仓库 (假定在/home/git目录下创建仓库) git init --bare sample.git克隆远程仓库到本地 git clone git192.168.0.100:/home/git/sample.git已有项目&#xff0c;绑定远程仓库 # 查看远程仓库绑定 git remote -v# 解除…...

如何让营销更生动,更有效!

作为专业的营销人员&#xff0c;我们深知在当今竞争激烈的市场环境中&#xff0c;如何让自己的产品或服务脱颖而出&#xff0c;吸引更多的潜在客户&#xff0c;是企业成功的关键。而中昱维信视频短信平台&#xff0c;正是您实现这一目标的得力助手。 一、视频短信&#xff0c;…...

RestTemplate请求参数需要转义 处理

项目需求 iam的token鉴权 需要带转义的回调http路径 用以下处理参数 接口仍然返回异常&#xff1a; public String authBack(String backUrl){ // backUrl http://192.168.1.156:sdm/String state URLEncoder.encode(state, "UTF-8"); }查了一下&#xff0c;Rest…...

使用Kaptcha实现的验证码功能

目录 一.需求 二.验证码功能实现步骤 验证码 引入kaptcha依赖 完成application.yml配置文件 浏览器显示验证码 前端页面 登录页面 验证成功页面 后端 此验证码功能是以SpringBoot框架下基于kaptcha插件来实现的。 一.需求 1.页面生成验证码 2.输入验证码&#xff…...

【无标题】CTF之SQLMAP

拿这一题来说 抓个包 复制报文 启动我们的sqlmap kali里边 sqlmap -r 文件路径 --dump --dbs 数据库 --tables 表...

【Qt之Quick模块】1. 概述及Quick应用程序创建流程

概述 Qt的Quick模块是用于创建现代化、动态和响应式用户界面的工具集。它是基于QML&#xff08;Qt Meta-Object Language&#xff09;和JavaScript的。 QML是一种声明性的语言&#xff0c;用于描述用户界面的结构和行为。它使用层叠样式表&#xff08;CSS&#xff09;的语法来…...

C语言-数组指针笔试题讲解(1)-干货满满!!!

文章目录 ▶️1.sizeof和strlen的对比&#x1f4af;➡️1.1 sizeof是什么&#xff1f;&#x1f4af;➡️1.2sizeof用法举例&#x1f4af;▶️1.3strlen是什么&#xff1f;&#x1f4af;▶️1.4 strlen函数用法举例&#xff1a;&#x1f4af;▶️1.5 strlen和sizeof的对比&#…...

springboot整合vue,将vue项目整合到springboot项目中

将vue项目打包后&#xff0c;与springboot项目整合。 第一步&#xff0c;使用springboot中的thymeleaf模板引擎 导入依赖 <!-- thymeleaf 模板 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-t…...

C++ 二叉搜索树(BST)的实现(非递归版本与递归版本)与应用

C 二叉搜索树的实现与应用 一.二叉搜索树的特点二.我们要实现的大致框架三.Insert四.InOrder和Find1.InOrder2.Find 五.Erase六.Find,Insert,Erase的递归版本1.FindR2.InsertR3.EraseR 七.析构,拷贝构造,赋值运算符重载1.析构2.拷贝构造3.赋值运算重载 八.Key模型完整代码九.二…...

分类预测 | Matlab实现AOA-SVM算术优化支持向量机的数据分类预测【23年新算法】

分类预测 | Matlab实现AOA-SVM算术优化支持向量机的数据分类预测【23年新算法】 目录 分类预测 | Matlab实现AOA-SVM算术优化支持向量机的数据分类预测【23年新算法】分类效果基本描述程序设计参考资料 分类效果 基本描述 1.Matlab实现AOA-SVM算术优化支持向量机的数据分类预测…...

代码随想录算法训练营第七天 | 454.四数相加II、383. 赎金信、15. 三数之和 、18. 四数之和

454.四数相加II 题目链接&#xff1a;454.四数相加II 给你四个整数数组 nums1、nums2、nums3 和 nums4 &#xff0c;数组长度都是 n &#xff0c;请你计算有多少个元组 (i, j, k, l) 能满足&#xff1a; 0 < i, j, k, l < nnums1[i] nums2[j] nums3[k] nums4[l] 0…...

SpringBoot 3.2.0 版本 mysql 依赖下载错误

最近想尝试一下最新的 SpringBoot 项目&#xff0c;于是将自己的开源项目进行了一些升级。 JDK 版本从 JDK8 升级至 JDK17。SpringBoot 版本从 SpringBoot 2.7.3 升级到 SpringBoot 3.2.0 其中 JDK 的升级比较顺利&#xff0c;毕竟 JDK 的旧版本兼容性一直非常好。 但是在升级…...

内网穿透的应用-如何结合Cpolar内网穿透工具实现在IDEA中远程访问家里或者公司的数据库

文章目录 1. 本地连接测试2. Windows安装Cpolar3. 配置Mysql公网地址4. IDEA远程连接Mysql小结 5. 固定连接公网地址6. 固定地址连接测试 IDEA作为Java开发最主力的工具&#xff0c;在开发过程中需要经常用到数据库&#xff0c;如Mysql数据库&#xff0c;但是在IDEA中只能连接本…...

ElasticSearch单机或集群未授权访问漏洞

漏洞处理方法&#xff1a; 1、可以使用系统防火墙 来做限制只允许ES集群和Server节点的IP来访问漏洞节点的9200端口&#xff0c;其他的全部拒绝。 2、在ES节点上设置用户密码 漏洞现象&#xff1a;直接访问9200端口不需要密码验证 修复过程 2.1 生成认证文件 必须要生成…...

【华为OD题库-097】最大岛屿体积-java

题目 题目描述 给你一个由大于0的数&#xff08;陆地)和0(水)组成的的二维网格&#xff0c;请你计算网格中最大岛屿的体积。陆地的数表示所在岛屿的体积。岛屿总是被水包围&#xff0c;并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。 此外&#xff0c;你可以假…...