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

如何让自动化测试更加灵活简洁?

简化的架构对于自动化测试和主代码一样重要。冗余和不灵活性可能会导致一些问题:比如 UI 中的任何更改都需要更新多个文件,测试可能在功能上相互重复,并且支持新功能可能会变成一项耗时且有挑战性的工作来适应现有测试。

页面对象模式如何理顺代码

在为应用程序编写测试时,我们需要在运行各种检查或操作时引用应用程序的视图元素。如果我们总是在编写的每个测试中明确说明元素 ID,这将使我们的代码容易受到 UI 更改的影响:我们必须在使用这些元素的每个测试中更新所有已更改的 ID。

页面对象模式有助于避免这种情况。页面对象模式的理念是将页面(应用程序屏幕)作为一个对象(测试抽象)来呈现,该对象会公布和初始化页面上的所有图形元素,并设置与它们的交互。有关该模式的详细信息可以在此处了解(https://kasperskylab.github.io/Kaspresso/en/Wiki/Page_object_in_Kaspresso/)。

本文中的所有示例都使用我们的开源测试自动化框架 Kaspresso。(https://github.com/KasperskyLab/Kaspresso)为什么不使用Espresso?

首先,Kaspresso使用声明式方法编写测试,这种方法依赖于Kakao,它是Espresso的Kotlin DSL封装器。下面是一个例子:

Espresso

 
  1. @Test

  2. fun testFirstFeature() {

  3. onView(withId(R.id.toFirstFeature))

  4. .check(ViewAssertions.matches(

  5. ViewMatchers.withEffectiveVisibility(

  6. ViewMatchers.Visibility.VISIBLE)))

  7. onView(withId(R.id.toFirstFeature)).perform(click())

  8. }

Kaspresso

 
  1. @Test

  2. fun testFirstFeature() {

  3. MainScreen {

  4. toFirstFeatureButton {

  5. isVisible()

  6. click()

  7. }

  8. }

  9. }

其次,在拦截器的帮助下,Kaspresso 避免了测试的不稳定性,从而提高了稳定性。这些拦截器在我们处理异步图形元素或列表时特别有用。

第三,Kaspresso集成了KAutomator,这是一个方便的Kotlin DSL封装器,可用于UI Automator,从而加快UI测试的速度。下面是标准版(右)和加速版(左)UI Automator之间的区别:

图片

除此之外,Kaspresso 允许将测试分解为步骤,类似于手动测试用例的完成方式,并记录每个步骤。如果测试崩溃,日志将帮助你立即查看哪些步骤成功完成,哪些步骤失败。除了日志之外,你还可以访问图形元素的层次结构以及视频、屏幕截图等。Kaspresso 内置的 Android 调试桥 (adb) 支持将帮助你直接使用 Android。Allure集成可清晰显示测试结果。

那么,让我们开始讨论正题。你可以通过下载项目源代码并运行它来重现下面描述的所有步骤。我们将描述 MainActivity 页面并自动化 LoginActivity 测试。结果以及测试可在TECH-tutorial-results分支中找到,因此你可以随时前往那里查看完成的代码。

MainActivity 看起来像这样:

图片

我们创建一个继承自 KScreen 的 MainScreen 对象:

 
  1. object MainScreen : KScreen<MainScreen>() {

  2. override val layoutId: Int? = null

  3. override val viewClass: Class<*>? = null

  4. }

KScreen 实现了页面对象模式,它描述了与测试交互的所有视图元素。

Kaspresso 中的页面对象实现以 layoutId 和 viewClass 变量而闻名,它们可以帮助开发人员立即识别哪个布局文件用于相关页面以及哪个类提供其功能。但手头的任务是讨论页面对象概念本身,因此我们现在将它们设置为 null。

我们使用 Android Studio 中的 UI Automator Viewer 或 Layout Inspector 来查找登录活动按钮的 ID。页面上其余视图元素的标识符可以类似地找到。

图片

主屏幕元素的描述如下所示:

 
  1. object MainScreen : KScreen<MainScreen>() {

  2. override val layoutId: Int? = null

  3. override val viewClass: Class<*>? = null

  4. val titleTextView = KTextView { withId(R.id.title) }

  5. val simpleActivityButton = KButton { withId(R.id.simple_activity_btn) }

  6. val wifiActivityButton = KButton { withId(R.id.wifi_activity_btn) }

  7. val notificationActivityButton = KButton { withId(R.id.notification_activity_btn) }

  8. val loginActivityButton = KButton { withId(R.id.login_activity_btn) }

  9. val makeCallActivityButton = KButton { withId(R.id.make_call_activity_btn) }

  10. val flakyActivityButton = KButton { withId(R.id.flaky_activity_btn) }

  11. val listActivityButton = KButton { withId(R.id.list_activity_btn) }

  12. }

现在,我们可以从我们创建的任何测试中引用 MainScreen 对象,并使用此页面的视图元素。

让我们编写第一个测试,它将检查页面上是否有“登录活动”按钮并单击它。

为此,我们创建一个继承自 TestCase 的 LoginActivityTest 类:

 
  1. class LoginActivityTest : TestCase() {

  2. /**

  3. * activityScenarioRule is used to invoke MainActivity before running the test.

  4. * More details on activityScenarioRule are available here:

  5. * https://developer.android.com/reference/androidx/test/ext/junit/rules/ActivityScenarioRule

  6. */

  7. @get:Rule

  8. val activityRule = activityScenarioRule<MainActivity>()

  9. @Test

  10. fun test() {

  11. MainScreen {

  12. loginActivityButton {

  13. isVisible()

  14. click()

  15. }

  16. }

  17. }

  18. }

图片

...并创建登录屏幕:

 
  1. object LoginScreen : KScreen<LoginScreen>() {

  2. override val layoutId: Int? = null

  3. override val viewClass: Class<*>? = null

  4. val usernameEditText = KEditText { withId(R.id.input_username) }

  5. val passwordEditText = KEditText { withId(R.id.input_password) }

  6. val loginButton = KButton { withId(R.id.login_btn) }

  7. }

让我们修改 LoginActivityTest 并尝试使用登录名“123456”和密码“123456”获得授权:

 
  1. class LoginActivityTest : TestCase() {

  2. @get:Rule

  3. val activityRule = activityScenarioRule<MainActivity>()

  4. @Test

  5. fun test() {

  6. val username = "123456"

  7. val password = "123456"

  8. MainScreen {

  9. loginActivityButton {

  10. isVisible()

  11. click()

  12. }

  13. }

  14. LoginScreen {

  15. usernameEditText { replaceText(username) }

  16. passwordEditText { replaceText(password) }

  17. loginButton { click() }

  18. }

  19. }

  20. }

授权后,我们会看到最后一个页面 AfterLoginActivity。

图片

Kaspresso 可以使用Device类从测试内部检查正在显示的活动。我们通过检查授权后设备屏幕上是否出现 AfterLoginActivity 来结束第一个测试:

 
  1. class LoginActivityTest : TestCase() {

  2. @get:Rule

  3. val activityRule = activityScenarioRule<MainActivity>()

  4. @Test

  5. fun test() {

  6. val username = "123456"

  7. val password = "123456"

  8. MainScreen {

  9. loginActivityButton {

  10. isVisible()

  11. click()

  12. }

  13. }

  14. LoginScreen {

  15. usernameEditText { replaceText(username) }

  16. passwordEditText { replaceText(password) }

  17. loginButton { click() }

  18. }

  19. device.activities.isCurrent(AfterLoginActivity::class.java)

  20. }

  21. }

这种方法使得动态了解哪些测试字符串与哪些页面交互变得更加困难。添加新的检查和操作可能会使代码难以辨认。因此,我们建议使用页面对象来创建高质量的可扩展测试。

将测试分为几个步骤

任何测试,无论是自动测试还是手动测试,都要遵循一个测试用例--也就是说,测试人员要检查一连串的步骤,以确定页面是否功能齐全。在 step() 函数的帮助下,Kaspresso 将代码分解成多个步骤。步骤还有助于整理测试日志。

要使用步骤,需要在测试中调用 run{} 方法,并在大括号中列出测试中要运行的所有步骤。每个步骤都应在 step() 函数中调用。

让我们试一下:​​​​​​​

 
  1. class LoginActivityTest : TestCase() {

  2. @get:Rule

  3. val activityRule = activityScenarioRule<MainActivity>()

  4. @Test

  5. fun test() {

  6. run {

  7. val username = "123456"

  8. val password = "123456"

  9. step("Open login screen") {

  10. MainScreen {

  11. loginActivityButton {

  12. isVisible()

  13. click()

  14. }

  15. }

  16. }

  17. step("Try to login") {

  18. LoginScreen {

  19. usernameEditText { replaceText(username) }

  20. passwordEditText { replaceText(password) }

  21. loginButton { click() }

  22. }

  23. }

  24. step("Check current screen") {

  25. device.activities.isCurrent(AfterLoginActivity::class.java)

  26. }

  27. }

  28. }

  29. }

通过这些步骤,标记为“KASPRESSO”的信息级日志如下所示:

图片

如果你对步骤仍有疑问,建议你阅读这些(https://kasperskylab.github.io/Kaspresso/en/Tutorial/Steps_and_sections/)。它还提供了你可能在日志中注意到的之前/之后部分的详细信息。

现在,让我们尝试实施负面测试用例,例如用户输入的登录名或密码少于最小字符数(6 个)。

在创建一组自动测试时,应遵循的规则是为每个测试用例设置一个单独的测试方法。换句话说,我们不会在同一个方法中测试输入无效登录名或密码时的行为,而是在 LoginActivityTest 类中创建单独的方法:​​​​​​​

 
  1. @Test

  2. fun loginUnsuccessfulIfUsernameIncorrect() {

  3. run {

  4. val username = "12"

  5. val password = "123456"

  6. step("Open login screen") {

  7. MainScreen {

  8. loginActivityButton {

  9. isVisible()

  10. click()

  11. }

  12. }

  13. }

  14. step("Try to login") {

  15. LoginScreen {

  16. usernameEditText { replaceText(username) }

  17. passwordEditText { replaceText(password) }

  18. loginButton { click() }

  19. }

  20. }

  21. step("Check current screen") {

  22. device.activities.isCurrent(LoginActivity::class.java)

  23. }

  24. }

  25. }

另一个测试,使用有效的登录名和无效的密码:​​​​​​​

 
  1. @Test

  2. fun loginUnsuccessfulIfPasswordIncorrect() {

  3. run {

  4. val username = "123456"

  5. val password = "1234"

  6. step("Open login screen") {

  7. MainScreen {

  8. loginActivityButton {

  9. isVisible()

  10. click()

  11. }

  12. }

  13. }

  14. step("Try to login") {

  15. LoginScreen {

  16. usernameEditText { replaceText(username) }

  17. passwordEditText { replaceText(password) }

  18. loginButton { click() }

  19. }

  20. }

  21. step("Check current screen") {

  22. device.activities.isCurrent(LoginActivity::class.java)

  23. }

  24. }

  25. }

我建议你在执行第一个测试时重命名它,以便其名称显示我们仅检查是否成功授权。​​​​​​​

 
  1. @Test

  2. fun test()

我们将其更改为:​​​​​​​

 
  1. @Test

  2. fun loginSuccessfulIfUsernameAndPasswordCorrect()

你可能已经注意到,在上面的自动化测试中,用于导航到 LoginActivity 页面并输入登录凭据的字符串会重复。如果能重复使用这些步骤就好了。

使用Scenario

Kaspresso 包含一个名为 Scenario 的工具,它允许将多个步骤组合成有序的操作序列。这在编写重复步骤的测试时非常有用。

让我们创建一个继承自 Scenario 的 LoginScenario 类。为了使其工作,我们需要重写 steps 属性以列出Scenario中的所有步骤。​​​​​​​

 
  1. class LoginScenario : Scenario() {

  2. override val steps: TestContext<Unit>.() -> Unit = {

  3. step("Open login screen") {

  4. MainScreen {

  5. loginActivityButton {

  6. isVisible()

  7. click()

  8. }

  9. }

  10. }

  11. step("Try to login") {

  12. LoginScreen {

  13. usernameEditText { replaceText(username) }

  14. passwordEditText { replaceText(password) }

  15. loginButton { click() }

  16. }

  17. }

  18. }

  19. }

class LoginScenario : Scenario()

更改为:​​​​​​​

 
  1. class LoginScenario(

  2. private val username: String,

  3. private val password: String

  4. ) : Scenario()

这是生成的Scenario代码:​​​​​​​

 
  1. class LoginScenario(

  2. private val username: String,

  3. private val password: String

  4. ) : Scenario() {

  5. override val steps: TestContext<Unit>.() -> Unit = {

  6. step("Open login screen") {

  7. MainScreen {

  8. loginActivityButton {

  9. isVisible()

  10. click()

  11. }

  12. }

  13. }

  14. step("Try to login") {

  15. LoginScreen {

  16. usernameEditText { replaceText(username) }

  17. passwordEditText { replaceText(password) }

  18. loginButton { click() }

  19. }

  20. }

  21. }

  22. }​​​​​​​

让我们在 LoginActivityTest 测试中使用此Scenario:​​​​​​​

 
  1. class LoginActivityTest : TestCase() {

  2. @get:Rule

  3. val activityRule = activityScenarioRule<MainActivity>()

  4. @Test

  5. fun loginSuccessfulIfUsernameAndPasswordCorrect() {

  6. run {

  7. step("Try to login with correct username and password") {

  8. scenario(

  9. LoginScenario(

  10. username = "123456",

  11. password = "123456",

  12. )

  13. )

  14. }

  15. step("Check current screen") {

  16. device.activities.isCurrent(AfterLoginActivity::class.java)

  17. }

  18. }

  19. }

  20. @Test

  21. fun loginUnsuccessfulIfUsernameIncorrect() {

  22. run {

  23. step("Try to login with incorrect username") {

  24. scenario(

  25. LoginScenario(

  26. username = "12",

  27. password = "123456",

  28. )

  29. )

  30. }

  31. step("Check current screen") {

  32. device.activities.isCurrent(LoginActivity::class.java)

  33. }

  34. }

  35. }

  36. @Test

  37. fun loginUnsuccessfulIfPasswordIncorrect() {

  38. run {

  39. step("Try to login with incorrect password") {

  40. scenario(

  41. LoginScenario(

  42. username = "123456",

  43. password = "1234",

  44. )

  45. )

  46. }

  47. step("Check current screen") {

  48. device.activities.isCurrent(LoginActivity::class.java)

  49. }

  50. }

  51. }

  52. }​​​​​​​

我们研究了一种有利于使用Scenario的案例——在同一页面的不同测试中重复使用相同的步骤。然而,这并不是Scenario的唯一目的。

一个应用程序可以有多个页面,你只能以授权用户的身份访问这些页面。然后,你需要重新描述每个页面的授权步骤。但是,如果你使用Scenario,这将变得非常简单。

目前,AfterLoginActivity 页面在我们登录后打开。让我们为该屏幕编写一个测试。

首先我们创建一个页面对象:​​​​​​​

 
  1. object AfterLoginScreen : KScreen<AfterLoginScreen>() {

  2. override val layoutId: Int? = null

  3. override val viewClass: Class<*>? = null

  4. val title = KTextView { withId(R.id.title) }

  5. }

然后我们添加测试:​​​​​​​

 
  1. class AfterLoginActivityTest : TestCase() {

  2. @get:Rule

  3. val activityRule = activityScenarioRule<MainActivity>()

  4. @Test

  5. fun test() {

  6. }

  7. }

我们需要获得授权才能访问该页面。如果没有Scenario,我们将不得不再次重新运行所有步骤:打开主页,单击按钮,输入登录名和密码,然后再次单击按钮。整个过程现在简化为使用 LoginScenario:​​​​​​​

 
  1. class AfterLoginActivityTest : TestCase() {

  2. @get:Rule

  3. val activityRule = activityScenarioRule<MainActivity>()

  4. @Test

  5. fun test() {

  6. run {

  7. step("Open AfterLogin screen") {

  8. scenario(

  9. LoginScenario(

  10. username = "123456",

  11. password = "123456"

  12. )

  13. )

  14. }

  15. step("Check title") {

  16. AfterLoginScreen {

  17. title {

  18. isVisible()

  19. }

  20. }

  21. }

  22. }

  23. }

  24. }

总而言之,使用Scenario使代码干净、清晰且可重用。如果你想要测试仅授权用户可以访问的页面,则无需再重复大量相同的步骤。重要的是,我们还实现了适当的测试可扩展性。如果 LoginActivity 页面上的 UI 元素的标识符发生更改,则不需要更新测试代码。要使测试再次正常工作,你所需要做的就是修复 LoginScreen。

作为对比,这里是没有以上最佳实践的测试代码。我希望你能像一场噩梦一样忘记这种写作风格。​​​​​​​

 
  1. class LoginActivityTest : TestCase() {

  2. @get:Rule

  3. val activityRule = activityScenarioRule<MainActivity>()

  4. @Test

  5. fun test() {

  6. val loginActivityButton = KButton { withId(R.id.login_activity_btn) }

  7. loginActivityButton {

  8. isVisible()

  9. click()

  10. }

  11. val usernameEditText = KEditText { withId(R.id.input_username) }

  12. val passwordEditText = KEditText { withId(R.id.input_password) }

  13. val loginButton = KButton { withId(R.id.login_btn) }

  14. usernameEditText { replaceText("123456") }

  15. passwordEditText { replaceText("123456") }

  16. loginButton { click() }

  17. device.activities.isCurrent(AfterLoginActivity::class.java)

  18. pressBack()

  19. usernameEditText { replaceText("123456") }

  20. passwordEditText { replaceText("1234") }

  21. loginButton { click() }

  22. device.activities.isCurrent(LoginActivity::class.java)

  23. usernameEditText { replaceText("12") }

  24. passwordEditText { replaceText("123456") }

  25. loginButton { click() }

  26. device.activities.isCurrent(LoginActivity::class.java)

  27. }

  28. }​​​​​​​

Kaspresso 框架相关链接:

https://github.com/KasperskyLab/Kaspresso

https://kasperskylab.github.io/Kaspresso/en

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

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!有需要的小伙伴可以点击下方小卡片领取 

 

相关文章:

如何让自动化测试更加灵活简洁?

简化的架构对于自动化测试和主代码一样重要。冗余和不灵活性可能会导致一些问题&#xff1a;比如 UI 中的任何更改都需要更新多个文件&#xff0c;测试可能在功能上相互重复&#xff0c;并且支持新功能可能会变成一项耗时且有挑战性的工作来适应现有测试。 页面对象模式如何理…...

linux 下载依赖慢和访问github代码慢

1 pip install 下载依赖慢&#xff0c;添加清华镜像源 pip config set global.index-url https://pypi.tuna.tsinghua.edu.cn/simple 2 git 出现错误 Could not resolve host: github.com 原来是因为github.com没有被主机给解析&#xff0c; 第一步 先 ping 看一下主机地址 …...

奥比中光astra_pro相机使用记录

一、信息获取 1、官网 用于了解产品信息 http://www.orbbec.com.cn/sys/37.html 2、开发者社区 咨询问题下载开发部https://developer.orbbec.com.cn/ 二 、windowvs19 1、相机型号 orbbec_astro_pro 根据对应的型号找到需要的包工具 踩坑1&#xff0c;因为这个相机型号…...

【MindSpore学习打卡】应用实践-计算机视觉-深入解析 Vision Transformer(ViT):从原理到实践

在近年来的深度学习领域&#xff0c;Transformer模型凭借其在自然语言处理&#xff08;NLP&#xff09;中的卓越表现&#xff0c;迅速成为研究热点。尤其是基于自注意力&#xff08;Self-Attention&#xff09;机制的模型&#xff0c;更是推动了NLP的飞速发展。然而&#xff0c…...

Debezium系列之:支持在一个数据库connector采集中过滤某些表的删除事件

Debezium系列之:支持在一个数据库connector采集中过滤某些表的删除事件 一、需求二、相关技术三、参数设置四、消费数据一、需求 在一个数据库的connector中采集了多张表,部分表存在数据归档的业务场景,会定期从表中删除历史数据,希望能过滤掉存在数据归档这些表的删除事件…...

SQL Server端口配置指南:最佳实践与技巧

1. 引言 SQL Server通常使用默认端口1433进行通信。为了提高安全性和性能&#xff0c;正确配置SQL Server的端口非常重要。本指南将帮助您了解如何配置和优化SQL Server的端口设置&#xff0c;以满足不同环境和需求。 2. 端口配置基础 2.1 默认端口 SQL Server的默认端口是…...

FastGPT 报错:undefined 该令牌无权使用模型:gpt-3.5-turbo (request id: xxx)

目录 一、FastGPT 报错 二、解决方法 一、FastGPT 报错 进行对话时 FastGPT 报错如下所示。 [Error] 2024-07-01 09:25:23 sse error: undefined 该令牌无权使用模型:gpt-3.5-turbo (request id: xxxxx) {message: 403 该令牌无权使用模型:gpt-3.5-turbo (request id: x…...

springboot系列八: springboot静态资源访问,Rest风格请求处理, 接收参数相关注解

文章目录 WEB开发-静态资源访问官方文档基本介绍快速入门注意事项和细节 Rest风格请求处理基本介绍应用实例注意事项和细节思考题 接收参数相关注解基本介绍应用实例PathVariableRequestHeaderRequestParamCookieValueRequestBodyRequestAttributeSessionAttribute ⬅️ 上一篇…...

# 职场生活之道:善于团结

在职场这个大舞台上&#xff0c;每个人都是演员&#xff0c;也是观众。要想在这个舞台上站稳脚跟&#xff0c;除了专业技能&#xff0c;更要学会如何与人相处&#xff0c;如何团结他人。团结&#xff0c;是职场生存的重要法则之一。 1. 主动团结&#xff1a;多一个朋友&#x…...

go sync包(五) WaitGroup

WaitGroup sync.WaitGroup 可以等待一组 Goroutine 的返回&#xff0c;一个比较常见的使用场景是批量发出 RPC 或者 HTTP 请求&#xff1a; requests : []*Request{...} wg : &sync.WaitGroup{} wg.Add(len(requests))for _, request : range requests {go func(r *Reque…...

基于深度学习的相机内参标定

基于深度学习的相机内参标定 相机内参标定&#xff08;Camera Intrinsic Calibration&#xff09;是计算机视觉中的关键步骤&#xff0c;用于确定相机的内部参数&#xff08;如焦距、主点位置、畸变系数等&#xff09;。传统的标定方法依赖于已知尺寸的标定板&#xff0c;通常…...

适合金融行业的国产传输软件应该是怎样的?

对于金融行业来说&#xff0c;正常业务开展离不开文件传输场景&#xff0c;一般来说&#xff0c;金融行业常用的文件传输工具有IM通讯、邮件、自建文件传输系统、FTP应用、U盘等&#xff0c;这些传输工具可以基础实现金融机构的文件传输需求&#xff0c;但也存在如下问题&#…...

昇思25天学习打卡营第9天|MindSpore使用静态图加速(基于context的开启方式)

在Graph模式下&#xff0c;Python代码并不是由Python解释器去执行&#xff0c;而是将代码编译成静态计算图&#xff0c;然后执行静态计算图。 在静态图模式下&#xff0c;MindSpore通过源码转换的方式&#xff0c;将Python的源码转换成中间表达IR&#xff08;Intermediate Repr…...

class类和style内联样式的绑定

这里的绑定其实就是v-bind的绑定&#xff0c;如代码所示&#xff0c;div后面的引号就是v-bind绑定&#xff0c;然后大括号将整个对象括起来&#xff0c;对象内先是属性&#xff0c;属性后接的是变量&#xff0c;这个变量是定义在script中的&#xff0c;后通过这个变量&#xff…...

3033.力扣每日一题7/5 Java

博客主页&#xff1a;音符犹如代码系列专栏&#xff1a;算法练习关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ 目录 思路 解题方法 时间复杂度 空间复杂度 Code 思路 首先创建一个与…...

GPT-5:下一代AI如何彻底改变我们的未来

GPT-5 发布前瞻&#xff1a;技术突破与未来展望 随着科技的飞速发展&#xff0c;人工智能领域不断迎来新的突破。根据最新消息&#xff0c;OpenAI 的首席技术官米拉穆拉蒂在一次采访中确认&#xff0c;GPT-5 将在一年半后发布&#xff0c;并描述了其从 GPT-4 到 GPT-5 的飞跃如…...

重载一元运算符

自增运算符 #include<iostream> using namespace std; class CGirl { public:string name;int ranking;CGirl() { name "zhongge"; ranking 5; }void show() const{ cout << "name : "<<name << " , ranking : " <…...

10元 DIY 一个柔性灯丝氛围灯

之前TikTok上特别火的线性氛围灯Augelight刚出来的时候一度卖到80多美金&#xff0c;国内1688也能到400多人民币。 随着各路国内厂商和DIY创客的跟进&#xff0c;功能变多的同时价格一路下滑&#xff0c;虽然有的质感的确感人&#xff0c;但是便宜啊。 甚至关注的up有把成本搞到…...

表单自定义组件 - 可选择卡片SelectCard

import React from react; import styles from ./index.module.less;type OptionsType {/*** 每个item渲染一行&#xff0c;第0项为标题*/labels?: any[];/*** 自定义渲染内容*/label?: string | React.ReactNode;value: any; }; interface IProps {value?: any;onChange?…...

Ubuntu / Debian安装FTP服务

本章教程,记录在Ubuntu中安装FTP服务的具体步骤。FTP默认端口:21 1、安装 pure-ftpd sudo apt-get install pure-ftpd2、修改默认配置 # 与 centos 不同,这里需要在 /etc/pure-ftpd/conf 文件夹下执行下列命令,增加对应配置文件: # 创建 /etc/pure-ftpd/conf/PureDB 文件…...

若依 Vue 前端分离 3.8.8 版中生成的前端代码中关于下拉框只有下拉箭头的问题

生成代码修改前 <el-form-item label"课程学科" prop"subject"><el-select v-model"queryParams.subject" placeholder"请选择课程学科" clearable><el-optionv-for"dict in course_subject":key"dict…...

C++把一个类封装成动态链接库

一、步骤 1. 创建类头文件 首先&#xff0c;定义你要封装的类。例如&#xff0c;创建一个名为MyClass的类&#xff1a; // MyClass.h #pragma once#ifdef MYCLASS_EXPORTS #define MYCLASS_API __declspec(dllexport) #else #define MYCLASS_API __declspec(dllimport) #end…...

每天一个项目管理概念之项目章程

项目管理中&#xff0c;项目章程扮演着至关重要的角色。它是项目正式启动的标志&#xff0c;为项目的执行提供法律和组织上的认可。项目章程是项目管理知识体系&#xff08;PMBOK&#xff09;中定义的关键文档之一&#xff0c;对于确保项目的顺利进行具有决定性的影响。 定义与…...

c++11新特性-4-返回类型后置

文章目录 返回类型后置1.基本语法 返回类型后置 1.基本语法 auto func(参数1&#xff0c;参数2&#xff0c;参数3&#xff0c;...)->decltype(参数表达式) {...... }例如&#xff1a; template<typename T,typename U> auto add(T t,U u)->decltype(t u) {retu…...

Linux-C语言实现一个进度条小项目

如何在linux中用C语言写一个项目来实现进度条&#xff1f;&#xff08;如下图所示&#xff09; 我们知道\r是回车&#xff0c;\n是换行&#xff08;且会刷新&#xff09; 我们可以用 \r 将光标移回行首&#xff0c;重新打印一样格式的内容&#xff0c;覆盖旧的内容&#xff0c;…...

vue使用glide.js实现轮播图(可直接复制使用)

效果图 可以实现自动轮播&#xff0c;3种切换方式&#xff1a;直接滑动图片、点击两侧按钮、点击底部按钮 体验链接:http://website.livequeen.top 实现 一、引入依赖 1、控制台引入依赖 npm install glidejs/glide 2、在css中引用 <style scoped> import glidejs/g…...

TK养号工具开发会用上的源代码科普!

在当今数字化时代&#xff0c;社交媒体平台的崛起使得网络账号的维护与管理变得日益重要&#xff0c;其中&#xff0c;TK作为一款备受欢迎的社交媒体平台&#xff0c;吸引了大量用户。 在TK上进行账号养护&#xff0c;即通过各种方式提升账号权重、增加曝光量&#xff0c;已成…...

信创-办公软件应用工程师认证

随着国家对信息技术自主创新的战略重视程度不断提升&#xff0c;信创产业迎来前所未有的发展机遇。未来几年内&#xff0c;信创产业将呈现市场规模扩大、技术创新加速、产业链完善和国产化替代加速的趋势。信创人才培养对于推动产业发展具有重要意义。应加强高校教育、建立人才…...

数组操作forEach和map

forEach和map的相同点 1、都是循环遍历数组中的每一项 2、入参匿名函数都支持三个参数&#xff0c;当前项item&#xff0c;当前项索引index&#xff0c;原始数组arr&#xff1b;匿名函数中的this都指向window 3、都可以通过return跳过本次循环 4、都无法通过使用 break 语句来中…...

流式处理应用场景与流式计算处理框架选择建议

文章目录 前言使用场景如何选择流式处理框架 前言 在之前的文章中我们介绍了如何进行流式处理——从一般性的概念和模式说起&#xff0c;并列举了一些Streams的例子&#xff1a; 流式处理相关概念总结说明流式处理设计模式总结说明Kafka Streams 架构概览 接下来的文章将介绍…...

2024年软件测试岗必问的100+个面试题【含答案】

一、基础理论 1、开场介绍 介绍要领&#xff1a;个人基本信息、工作经历、之前所做过的工作及个人专长或者技能优势。扬长避短&#xff0c;一定要口语化&#xff0c;语速适中。沟通好的就多说几句&#xff0c;沟通不好的话就尽量少说两句。举例如下&#xff1a; 面试官你好&…...

A4-C四驱高防轮式巡检机器人

在当今数字化和智能化迅速发展的时代&#xff0c;旗晟智能带来了一款革命性的创新产品——A4-C四驱高防轮式巡检机器人。这款机器人以其卓越的性能和多功能性&#xff0c;为工业巡检领域带来了全新的解决方案。 一、产品亮点 1、四驱动力与高防护设计 四驱高防轮式巡检机器人…...

Https网站如何申请免费的SSL证书及操作使用指南

前言 在当今互联网环境下&#xff0c;HTTPS已成为网站安全的标配&#xff0c;它通过SSL/TLS协议为网站数据传输提供加密&#xff0c;保障用户信息的安全。申请并部署免费SSL证书&#xff0c;不仅能够提升网站的专业形象&#xff0c;还能增强用户信任。本文将详细介绍如何在知名…...

实现资产优化管理:智慧校园资产分类功能解析

在构建智慧校园的过程中&#xff0c;细致入微的资产管理是确保教育资源高效运作的关键一环&#xff0c;而资产分类功能则扮演着举足轻重的角色。系统通过精心设计的分类体系&#xff0c;将校园内的各类资产&#xff0c;从昂贵的教学设备到日常使用的办公物资&#xff0c;乃至无…...

大厂开发必知必会:Devops、CI/CD、流水线和Paas的关系解析说明

为什么作为程序开发人员需要了解ci/cd流程和原理&#xff1f; 作为程序开发人员&#xff0c;了解CI/CD&#xff08;持续集成/持续交付&#xff09;的流程和原理具有以下几个重要的理由&#xff1a; 1. 提高代码质量和稳定性 自动化测试&#xff1a;CI/CD流程中集成了自动化测…...

Qt学习:Qt窗口组件以及窗口类型

一、Qt的窗口组件有哪些 Qt是一个跨平台的C应用程序开发框架&#xff0c;它的窗口组件&#xff0c;也称为用户界面元素&#xff0c;为开发者提供了丰富的可视化界面设计选项。在Qt中&#xff0c;窗口组件主要包括&#xff1a; 窗口&#xff08;Window&#xff09;&#xff1a;…...

基于AGX ORIN与FPGA K7实现PCIE高速数据通信/Orin与FPGA高速数据传输/XDMA在linux系统使用教程

因最近想学习AGX orin和FPGA实现数据高速通信&#xff0c;借此机会和大家一起学习AGX orin和FPGA 制作不易&#xff0c;记得三连哦&#xff0c;给我动力&#xff0c;持续更新&#xff01;&#xff01;&#xff01; 完整工程文件下载&#xff1a;AGX orin与FPGA实现PCIE完整…...

Vue3:全局播放背景音乐

说明&#xff1a;一个全局播放的背景音乐&#xff0c;首页无音乐无音乐图标&#xff0c;在首页互动跳转页面并开始播放音乐&#xff0c;切换页面不需暂停音乐也不会重置音乐&#xff0c;可以通过音乐图标控制暂停或播放。 MusicPlay.vue&#xff08;音乐组件&#xff09; <…...

2024年07月03日 Redis部署方式和持久化

Redis持久化方式&#xff1a;RDB和AOF&#xff0c;和混合式 RDB&#xff1a;周期备份模式&#xff0c;每隔一段时间备份一份快照文件&#xff0c;从主线程Fork一个备份线程出来备份&#xff0c;缺点是会造成数据的丢失。 AOF&#xff1a;日志模式&#xff0c;每条命令都以操作…...

成都仅需浏览器即可快速查看的数据采集监控平台!

LP-SCADA数据采集监控平台无需额外客户端&#xff0c;只需要一个标准的Web浏览器&#xff0c;用户可以迅速访问系统并开始使用&#xff0c;同时支持跨平台访问。一个用户可监控多个过程&#xff0c;多个用户可以监控同一过程&#xff0c;真正实现了数据的开放性及过程信号的透明…...

LLM - 神经网络的训练过程

1. 对于回归问题&#xff0c;用损失函数来计算预测值和真实值的差异&#xff0c;一种常用的公式是如下图所示(Mean Square Error)&#xff0c;如果损失函数的值越小说明神经网络学习越准确&#xff0c;所以神经网络训练目标是减小损失函数的值&#xff0c; 2. 对于分类问题&…...

【全网最全ABC三题完整版】2024年APMCM第十四届亚太地区大学生数学建模竞赛(中文赛项)完整思路解析+代码+论文

我是Tina表姐&#xff0c;毕业于中国人民大学&#xff0c;对数学建模的热爱让我在这一领域深耕多年。我的建模思路已经帮助了百余位学习者和参赛者在数学建模的道路上取得了显著的进步和成就。现在&#xff0c;我将这份宝贵的经验和知识凝练成一份全面的解题思路与代码论文集合…...

Python | Leetcode Python题解之第213题打家劫舍II

题目&#xff1a; 题解&#xff1a; class Solution:def rob(self, nums: List[int]) -> int:def robRange(start: int, end: int) -> int:first nums[start]second max(nums[start], nums[start 1])for i in range(start 2, end 1):first, second second, max(fi…...

揭秘数据之美:【Seaborn】在现代【数学建模】中的革命性应用

目录 已知数据集 tips 生成数据集并保存为CSV文件 数据预览&#xff1a; 导入和预览数据 步骤1&#xff1a;绘制散点图&#xff08;Scatter Plot&#xff09; 步骤2&#xff1a;添加回归线&#xff08;Regression Analysis&#xff09; 步骤3&#xff1a;分类变量分析&…...

【宠粉赠书】UML 2.5基础、建模与设计实践

为了回馈粉丝们的厚爱&#xff0c;今天小智给大家送上一套系统建模学习的必备书籍——《UML 2.5基础、建模与设计实践》。下面我会详细给大家介绍这本书&#xff0c;文末留有领取方式。 图书介绍 《UML 2.5基础、建模与设计实践》以实战为主旨&#xff0c;结合draw.io免费软件…...

Python中几个重要的集合

Python中几个重要的集合&#xff08;Collection&#xff09;类型&#xff0c;包括列表&#xff08;List&#xff09;、元组&#xff08;Tuple&#xff09;、集合&#xff08;Set&#xff09;和字典&#xff08;Dictionary&#xff09; 1. 列表&#xff08;List&#xff09; 说…...

【JS】纯web端使用ffmpeg实现的视频编辑器-视频合并

纯前端实现的视频合并 接上篇ffmpeg文章 【JS】纯web端使用ffmpeg实现的视频编辑器 这次主要添加了一个函数&#xff0c;实现了视频合并的操作。 static mergeArgs(timelineList) {const cmd []console.log(时间轴数据,timelineList)console.log("文件1",this.readD…...

解决Python用xpath爬取不到数据的一个思路

前言 最近在学习Python爬虫的知识&#xff0c;既然眼睛会了难免忍不住要实践一把。 不废话直接上主题 代码不复杂&#xff0c;简单的例子奉上&#xff1a; import requests from lxml import etreecookie 浏览器F12网络请求标头里有 user_agent 浏览器F12网络请求标头里有…...

C#面:如何把一个array复制到arrayist里

要将一个数组复制到ArrayList中&#xff0c;可以使用ArrayList的AddRange方法。以下是一个示例代码&#xff1a; int[] array { 1, 2, 3, 4, 5 }; ArrayList arrayList new ArrayList(); arrayList.AddRange(array); 在上面的代码中&#xff0c;我们首先创建了一个整数类型…...

解决前后端同一个端口跨域问题

前端起了一个代理 如果url是api开头的自动代理访问8080端口&#xff08;解决前后端端口不一致要么是前端代理&#xff0c;要么是后端加过滤器&#xff09; proxy:{/api:{target:http://localhost:8080,changeOrigin : true,// 替换去掉路径上的api// rewrite:(path)>path.r…...