【Spring Boot】034-Spring Boot 整合 JUnit

【Spring Boot】034-Spring Boot 整合 JUnit

文章目录

  • 【Spring Boot】034-Spring Boot 整合 JUnit
  • 一、单元测试
    • 1、什么是单元
    • 2、什么是单元测试
    • 3、为什么要单元测试
  • 二、JUnit
    • 1、概述
      • 简介
      • 特点
    • 2、JUnit4
      • 概述
      • 基本用法
    • 3、JUnit5
      • 概述
      • 组成
    • 4、JUnit5 与 JUnit4 的常用注解对比
  • 三、Spring Boot 整合 JUnit
    • 1、Spring Boot 项目创建
    • 2、引入依赖
    • 3、外部库
    • 4、编写测试类
    • 5、运行结果截图
    • 6、`@SpringBootTest` 本身支持依赖注入
      • 编写组件
      • 依赖注入
    • 7、手动指定主启动类
      • 概述
      • 子包下可正常运行
      • 包外提示异常
      • 结构截图
      • 解决方案:显式声明主启动类的位置
  • 四、JUnit的断言机制
    • 1、概述
    • 2、基本使用
    • 3、组合条件断言
    • 4、异常抛出断言
    • 5、执行超时断言
    • 6、强制失败
  • 五、前置条件检查机制
    • 1、概述
    • 2、代码演示
    • 3、运行结果截图
  • 六、嵌套测试
    • 1、概述
    • 2、代码演示
    • 3、运行结果截图
    • 4、注意点
  • 七、带入参的单元测试
    • 1、概述
    • 2、参数化测试
    • 3、基本类型
      • 概述
      • 代码演示
      • 测试结果截图
    • 4、方法数据流返回
      • 概述
      • 代码演示
      • 测试结果截图

一、单元测试

1、什么是单元

单元:最终能分解的尽可能小的、独立的、可执行的元素。

对于 Java 程序:单元 = 类的方法。

2、什么是单元测试

对程序中的一个个单元进行测试,看看是否能够正常使用、是否存在问题等。

3、为什么要单元测试

通过最小测试范围确定出一个功能单元是否正常可用,通过单元测试的编写和执行,可以在尽可能早期筛查、发现出一些问题。

二、JUnit

1、概述

简介

JUnit 是针对 Java 语言的一个经典单元测试框架,它在测试驱动方面具有重大意义。JUnit 促进了“先测试后编码”的理论,它强调测试数据与程序代码的配合关系,使得开发者在程序开发中形成“编码一点,测试一点”的过程,这种编码习惯可以提高程序的正确性和稳定性,进而提高开发者的产出效率,减少后期排查错误的时间和精力。

特点

  • 开放的资源框架,用于编写和运行测试;
  • 提供注释来识别测试方法;
  • 提供断言来测试预期结果;
  • 提供测试运行来运行测试;
  • 允许编写代码更快,并能提高质量;
  • 测试代码编写优雅简洁,花费时间较少;
  • 测试代码可以自动运行并且检查自身结果并提供即时反馈,没有必要人工梳理测试结果的报告;
  • 测试代码可以被组织为测试套件,包含测试用例,甚至其他的测试套件;

2、JUnit4

概述

早期使用的 JUnit 版本为 4.x ,这个版本对 jdk 的最低限制是 jdk 1.5 ,整个 JUnit 4 的代码被整合到一个 jar 包中,使用时直接导入即可,主流的 IDE 都有对 JUnit 的原生支持。

基本用法

编写一个类 + 声明一个无参无返回值方法 + 加上@Test注解

使用 JUnit 4 的方式比较简单,只需要编写一个类,并声明一个无入参、无返回值的方法,并标注 @Test 注解即可被 IDE 识别且运行。

public class DemoTest {@Testpublic void test() {System.out.println("DemoTest test run ......");}
}

3、JUnit5

概述

一个单独 jar 包 => 一组 jar 包组合而成 + 支持用户定制!

2017 年 9 月,JUnit 5.0.0 正式发布,它最低支持的 Java 版本是 Java 8 ,而且它的构建不再由一个独立的 jar 包构成,而是以一组 jar 包共同组合而成。抛弃历史包袱,通过支持扩展(Extension),JUnit 5 给用户提供了定制特殊的测试需求与方式。

组成

总的来说,JUnit 5 由 3 个模块构成,分别是 JUnit Platform 、JUnit Jupiter 、JUnit Vintage 。

  • JUnit Platform :基于 JVM 上启动测试框架的基础,不仅支持 JUnit 的测试引擎,也可以兼容其他的测试引擎;
  • JUnit Jupiter :JUnit 5 的核心,提供 JUnit 5 的新的编程模型,内部包含一个测试引擎,该测试引擎会基于 JUnit Platform 运行;
  • JUnit Vintage :兼容 JUnit 4 、JUnit 3 支持的测试引擎。

使用 JUnit 5 的方式跟 JUnit 4 并无太大区别,同样是编写测试类,并声明方法,标注 @Test 注解即可,不再编写示例代码解释。

4、JUnit5 与 JUnit4 的常用注解对比

JUnit 5 相较于 JUnit 4 比较大的改动是注解的使用。下表展示了 JUnit 5 跟 JUnit 4 常用注解的对比。

注解意义JUnit 5JUnit 4
标注一个测试方法(无区别)@Test@Test
在每个测试方法前执行@BeforeEach@Before
在每个测试方法后执行@AfterEach@After
在当前类中的所有测试方法之前执行@BeforeAll@BeforeClass
在当前类中的所有测试方法之后执行@AfterAll@AfterClass
禁用测试方法/类@Disabled@Ignore
标记和过滤@Tag@Category
声明测试工厂进行动态测试(新增)@TestFactory/
嵌套测试(新增)@Nested/
注册自定义扩展(新增)@ExtendWith/

三、Spring Boot 整合 JUnit

1、Spring Boot 项目创建

image-20231113134810578

2、引入依赖

项目创建完成后已经自动引入了!spring-boot-starter-test

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.7.17</version><relativePath/> <!-- lookup parent from repository --></parent><groupId>com.zibo</groupId><artifactId>study-junit</artifactId><version>0.0.1-SNAPSHOT</version><name>study-junit</name><description>study-junit</description><properties><java.version>1.8</java.version></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter</artifactId></dependency><!-- 创建项目后,自动引入了该依赖! --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency></dependencies><build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId><configuration><image><builder>paketobuildpacks/builder-jammy-base:latest</builder></image></configuration></plugin></plugins></build></project>

3、外部库

image-20231113135348801

4、编写测试类

package com.zibo.studyjunit;import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
class StudyJunitApplicationTests {@Testvoid contextLoads() {System.out.println("Hello World!");}}

5、运行结果截图

image-20231113135706434

6、@SpringBootTest 本身支持依赖注入

编写组件

package com.zibo.studyjunit.component;import org.springframework.stereotype.Component;@Component
public class HelloComponent {public String sayHello() {return "Hello World!";}}

依赖注入

package com.zibo.studyjunit;import com.zibo.studyjunit.component.HelloComponent;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
class StudyJunitApplicationTests {@Autowiredprivate HelloComponent helloComponent;@Testvoid contextLoads() {helloComponent.sayHello();}}

7、手动指定主启动类

概述

我们编写 SpringBoot 测试类时,不可能把全部的测试类都放到与 SpringBoot 主启动类同包下,当测试类一多,整个 test 目录会非常混乱。为了方便寻找与管理,我们还是需要将单元测试类也分包管理。但是请各位注意,当 SpringBoot 测试类被放到其他包的时候,运行 SpringBoot 测试类是有区别的。

子包下可正常运行

package com.zibo.studyjunit.demo;import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
class Demo1Test {@Testvoid test1() {System.out.println("test1");}}

包外提示异常

package demo;import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
class Demo2Test {@Testvoid test2() {System.out.println("test2");}}

结构截图

image-20231113141817929

解决方案:显式声明主启动类的位置

代码

package demo;import com.zibo.studyjunit.StudyJunitApplication;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest(classes = StudyJunitApplication.class)
class Demo2Test {@Testvoid test2() {System.out.println("test2");}}

运行结果截图

image-20231113142023402

四、JUnit的断言机制

1、概述

下面讲解 JUnit 5 中的经典使用方式:断言。在 JUnit 4 中我们使用 Assert 类进行断言,而到了 JUnit 5 中使用的类是 Assertions ,类名变了,使用方式却大差不差,下面通过几个简单示例讲解 JUnit 5 的断言使用。

2、基本使用

Assertions 提供的最简单的断言方法,包含比对两个值是否相等两个对象是否是同一个对象是否为 null ,以及全场景通用的判断表达式的值为 true / false 。下面是一个简单的使用示例。

    @Testvoid testSimple() {// 一、两个值是否相等// 最简单的断言,断言计算值与预期值是否相等int num = 3 + 5;Assertions.assertEquals(num, 8);double result = 10.0 / 3;// 断言计算值是否在浮点数的指定范围内上下浮动Assertions.assertEquals(result, 3, 0.5);// 如果浮动空间不够,则会断言失败// 断言失败!// Assertions.assertEquals(result, 3, 0.2);// 传入message可以自定义错误提示信息// 断言失败!// Assertions.assertEquals(result, 3, 0.2, "计算数值偏差较大!");// 二、两个对象是否是同一个// 断言两个对象是否是同一个Object o1 = new Object();Object o2 = o1;Object o3 = new Object();Assertions.assertSame(o1, o2);// 断言失败!// Assertions.assertSame(o1, o3);// 断言两个数组的元素是否完全相同String[] arr1 = {"aa", "bb"};String[] arr2 = {"aa", "bb"};String[] arr3 = {"bb", "aa"};Assertions.assertArrayEquals(arr1, arr2);// 断言失败!// Assertions.assertArrayEquals(arr1, arr3);// 三、对象是否为 null// 断言对象是否为 nullObject o4 = null;Object o5 = new Object();Assertions.assertNull(o4);// 断言失败!// Assertions.assertNull(o5);// 四、判断表达式的值为 true / false// 断言表达式的值为 trueint a = 10;int b = 20;Assertions.assertTrue(a < b);// 断言表达式的值为 falseAssertions.assertFalse(a > b);}

3、组合条件断言

组合条件断言,实际上是要在一条断言中组合多个断言,要求这些断言同时、全部通过,则外部的组合断言才能通过。这种设计有点类似于父子断言。

    @Testvoid testCombination() {// 组合条件断言,只有当所有断言都成功时,才会断言成功int a = 10;int b = 20;int c = 30;// 断言 a < b 并且 b < cAssertions.assertAll("组合断言",() -> Assertions.assertTrue(a < b),() -> Assertions.assertTrue(b < c));}

4、异常抛出断言

异常抛出的断言,指的是被测试的内容最终运行时必定会抛出一个异常,如果没有抛出异常则断言失败

    @Testvoid testException() {// 断言抛出指定类型的异常Assertions.assertThrows(ArithmeticException.class, () -> {int i = 10 / 0;});}

5、执行超时断言

执行超时断言是,针对的是被测试代码的执行速度。

    @Testvoid testTimeout() {// 断言在指定时间内完成// 断言失败!org.opentest4j.AssertionFailedError: execution timed out after 100 msAssertions.assertTimeoutPreemptively(java.time.Duration.ofMillis(100), () -> {Thread.sleep(200);});}

6、强制失败

类似于最原始的抛出异常的方式,(当满足某些条件时)直接断言失败!

    @Testvoid testFail() {if (java.time.ZonedDateTime.now().getHour() > 12) {Assertions.fail();}}

五、前置条件检查机制

1、概述

前置条件的检查机制,同样应用在断言的场景中,它指的是:如果一个单元测试的前置条件不满足,则当前的测试会被跳过,后续的测试不会执行。使用前置条件检查机制,可以避免一些无谓的测试逻辑执行,从而提高单元测试的执行效率。

前置条件的检查使用的 API 是 Assumptions

2、代码演示

    @Testvoid testAssumptions() {// 假设条件为 true 时,才会执行后面的断言int num = 3 + 5;Assumptions.assumeTrue(num < 10);System.out.println("断言成功!");// 假设条件为 false 时,不会执行后面的断言Assumptions.assumeTrue(num > 10);// 断言失败!下面的代码不会执行System.out.println("断言失败!");}

3、运行结果截图

image-20231113145356361

六、嵌套测试

1、概述

嵌套测试是 JUnit 5 的一个高级特性,它支持我们在编写单元测试类时,以内部类的方式组织一些有关联的测试逻辑。有关嵌套测试的演示代码,在 JUnit 5 的官方文档中提供了一个非常全面的示例。

2、代码演示

package com.zibo.studyjunit.demo;import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertThrows;
import static org.junit.jupiter.api.Assertions.assertTrue;import java.util.EmptyStackException;
import java.util.Stack;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Nested;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
class TestingAStackDemo {Stack<Object> stack;// 测试用例:使用new关键字实例化堆栈@Testvoid isInstantiatedWithNew() {new Stack<>();}@Nestedclass WhenNew {@BeforeEachvoid createNewStack() {// 在每个测试方法之前创建一个新的堆栈实例stack = new Stack<>();}// 测试用例:堆栈在刚创建时是否为空@Test@DisplayName("is empty")void isEmpty() {assertTrue(stack.isEmpty());}// 测试用例:在空栈上执行pop操作是否抛出EmptyStackException异常@Test@DisplayName("throws EmptyStackException when popped")void throwsExceptionWhenPopped() {assertThrows(EmptyStackException.class, () -> stack.pop());}@Nested@DisplayName("after pushing an element")class AfterPushing {String anElement = "an element";@BeforeEachvoid pushAnElement() {// 在堆栈中推入一个元素stack.push(anElement);}// 测试用例:在推入元素后,堆栈是否不再为空@Test@DisplayName("it is no longer empty")void isNotEmpty() {assertFalse(stack.isEmpty());}// 测试用例:在推入元素后,执行pop操作是否返回推入的元素并使堆栈为空@Test@DisplayName("returns the element when popped and is empty")void returnElementWhenPopped() {assertEquals(anElement, stack.pop());assertTrue(stack.isEmpty());}}}}

3、运行结果截图

image-20231113150426000

4、注意点

官方提供的测试代码都是可以执行通过的,从这段测试代码中需要各位了解的几个关键特性:

  • 单元测试类可以通过编写内部类,并标注 @Nested 注解,表明内部类也是一个单元测试类
  • 内部的单元测试类可以直接使用外部的成员属性,且可以利用外部定义的 @BeforeEach@BeforeAll@AfterEach@AfterAll 等前后置逻辑注解标注的方法;
  • 外部的单元测试无法利用内部类定义的前后置逻辑注解。

七、带入参的单元测试

1、概述

JUnit 5 中的一个重要特性是支持单元测试方法的参数依赖注入,这也打破了我们已有的认知。通常我们编写的测试方法是不能有方法入参的,但是 JUnit 5 允许我们在编写单元测试方法中予以声明方法入参。默认情况下 JUnit 5 支持以下几个参数类型的依赖注入:

  • TestInfo :内部组装了当前单元测试所属的 Class 、Method ,以及对应的展示名(DisplayName)等;
  • RepetitionInfo :如果一个方法被标注了 @RepeatedTest ,或者该方法是一个 @BeforeEach / @AfterEach 方法,则可以拿到 RepetitionInfo 的信息,可以通过 RepetitionInfo 获取到当前重复信息以及相应的@RepeatedTest的重复总数;
  • TestReporter :注入 TestReporter 后可以获得数据发布能力,可以向测试结果中注册一些特殊的数据,这些数据可以被 TestExecutionListener 获取到。

2、参数化测试

参数化测试是 JUnit 5 中提高单元测试效率的重要手段,它通过给单元测试方法传入特定的参数,可以使得 JUnit 在执行单元测试时逐个参数来检验和测试,这样做的好处是更加规整和高效地执行单元测试

参数化测试支持我们使用如下的方式赋予参数:

  • 基本类型:8 种基本数据类型 + String + Class
  • 枚举类型:自定义的枚举;
  • CSV 文件:可传入一个 CSV 格式的表格文件,使用表格文件中的数据作为入参;
  • 方法的数据返回:可以通过一个方法返回需要测试入参的数据(流的形式返回)。

3、基本类型

概述

在使用参数化测试时,标注的注解不再是 @Test ,取而代之的是 @ParameterizedTest ;另外还需要声明需要传入的数据,对于简单的基本类型而言,使用 @ValueSource 注解即可指定。

代码演示

package com.zibo.studyjunit.demo;import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.ValueSource;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
class ParameterizedTestDemo {@ParameterizedTest@ValueSource(strings = {"aa", "bb", "cc"})void testSimpleParameterized(String value) {System.out.println(value);Assertions.assertTrue(value.length() < 3);}}

测试结果截图

image-20231113151330211

4、方法数据流返回

概述

参数化测试最吸引人的点是可以引用一个方法来实现测试数据的参数化,既然可以在一个方法中构造单元测试的入参数据,那么完全可以从数据库 / 缓存等任意位置加载数据,并构造为流的形式返回。以此法编写的参数化测试具有极大的灵活度和自由度。

代码演示

    @ParameterizedTest@MethodSource("dataProvider")// @MethodSource("getInteger")void testDataStreamParameterized(Integer value) {System.out.println(value);Assertions.assertTrue(value < 10);}private static Stream<Integer> dataProvider() {return Stream.of(1, 2, 3, 4, 5);}// 不能用这种方式,会报错// 返回值必须是Stream<T>或者Stream<Arguments>// private static Integer getInteger() {//     return 1;// }private static Stream<Arguments> getInteger() {return Stream.of(Arguments.of(1), Arguments.of(2), Arguments.of(3));}

测试结果截图

image-20231113152855091

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AR工业眼镜:智能化生产新时代的引领者!!

科技飞速发展&#xff0c;人工智能与增强现实&#xff08;AR&#xff09;技术结合正在改变生活工作方式。AR工业眼镜在生产领域应用广泛&#xff0c;具有实时信息展示、智能导航定位、远程协作培训、智能安全监测等功能&#xff0c;提高生产效率、降低操作风险&#xff0c;为企…

算法训练 第七周

一、最小栈 本题要求我们实现一个最小栈数据结构&#xff0c;要求它可以实现栈的基本功能&#xff0c;并且还能使用常数时间复杂度来获取栈中的最小值。 1.辅助栈 我们可以在普通栈的基础上再添加一个维护最小值的辅助栈来实现这个数据结构&#xff0c;我们先创建一个普通的栈…

docker搭建etcd集群

最近用到etcd&#xff0c;就打算用docker搭建一套&#xff0c;学习整理了一下。记录在此&#xff0c;抛砖引玉。 文中的配置、代码见于https://gitee.com/bbjg001/darcy_common/tree/master/docker_compose_etcd 搭建一个单节点 docker run -d --name etcdx \-p 2379:2379 \…

[html] 动态炫彩渐变背景

废话不多说&#xff0c;直接上源码 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>ZXW-NUDT: 动态炫…

【算法与数据结构】78、90、LeetCode子集I, II

文章目录 一、题目二、78.子集三、90.子集II三、完整代码 所有的LeetCode题解索引&#xff0c;可以看这篇文章——【算法和数据结构】LeetCode题解。 一、题目 二、78.子集 思路分析&#xff1a;【算法与数据结构】77、LeetCode组合。本题可以参考77题的组合问题代码&#xff0…

Flutter有状态组件StatefulWidget生命周期

StatefulWidget是Flutter中的一个有状态的组件&#xff0c;它的生命周期相对复杂一些。下面是StatefulWidget的生命周期方法及其调用顺序&#xff1a; 1. createState(): 当StatefulWidget被插入到Widget树中时&#xff0c;会调用createState()方法来创建与之关联的State对象。…

@Async注解的坑

问题描述 一个方法调用另一个方法(该方法使用Async注解)在同一个类文件中&#xff0c;该注解会失效&#xff01; 问题复现 TestAsyncController 类 import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.scheduling.annotation.Async; im…

git快速上传代码

① git init&#xff1b; 初始化git&#xff0c;之后在文件夹里有.git文件&#xff0c;这个需要 勾选才能查看。 ② git remote add test myFisrtTest: 测试专用 这里的test是自定义的&#xff0c;myFisrtTest: 测试专用 是远程仓库 ③ git branch -a 这里是查看分支 ④ …

react函数式组件props形式子向父传参

父组件中定义 子组件中触发回调传值 import { useState } from "react"; function Son(params) {const [count, setCount] useState(0);function handleClick() {console.log(params, paramsparamsparamsparamsparamsparams);params.onClick(111)setCount(count 1…

阿里云容器镜像服务的运维总结

一、背景 容器镜像服务&#xff0c;作为一个可选付费产品&#xff0c;主要作用是存储docker的镜像仓库&#xff0c;供k8s拉取到Pod节点里。 你可以自己搭建一个harbor镜像仓库&#xff0c;在公司的开发环境下&#xff0c;将image推送到仓库&#xff1b;然后在生产k8s从仓库拉取…

如果不用Baklib,哪一个帮助中心工具能够替代它?

在各行各业进入“留量时代”的当下&#xff0c;让用户获得良好的体验和留存老客户变得更为关键&#xff0c;这对于企业的客户服务提出了更高的要求。在使用各类互联网产品时&#xff0c;用户更倾向于通过自助方式寻找答案并解决问题&#xff0c;因此帮助中心的重要性也在不断提…

mysql之MHA

1、定义 全称是masterhigh avaliabulity。基于主库的高可用环境下可以实现主从复制及故障切换&#xff08;基于主从复制才能故障切换&#xff09; MHA最少要求一主两从&#xff0c;半同步复制模式 2、作用 解决mysql的单点故障问题。一旦主库崩溃&#xff0c;MHA可以在0-30…

μC/OS-II---时间管理(os_time.c)

目录 时间管理相关&#xff08;os_time.c&#xff09;Task延迟按时、分、秒、毫秒延时恢复被延时的Task返回系统当前的Tick计数值设置系统的Tick计数值 时间管理相关&#xff08;os_time.c&#xff09; Task延迟 void OSTimeDly (INT32U ticks) {INT8U y; #if OS_CRITI…

为什么UI自动化难做?—— 关于Selenium UI自动化的思考

在快速迭代的产品、团队中&#xff0c;UI自动化通常是一件看似美好&#xff0c;实际“鸡肋”&#xff08;甚至绝大部分连鸡肋都算不上&#xff09;的工具。原因不外乎以下几点&#xff1a; 1 效果有限 通常只是听说过&#xff0c;就想去搞UI自动化的团队&#xff0c;心里都认…

idea生成代码(一):实现java语言的增删改查功能(基于EasyCode插件)支持自定义模板【非常简单】

idea生成代码&#xff08;一&#xff09;&#xff1a;实现java语言的增删改查功能&#xff08;基于EasyCode插件&#xff09;支持自定义模板【非常简单】 idea生成代码&#xff08;二&#xff09;&#xff1a;实现java语言的增删改查功能&#xff08;基于mybatis-plus代码生成器…

矩阵置零00

题目链接 矩阵置零 题目描述 注意点 使用 原地 算法 解答思路 思路是需要存储每一行以及每一列是否有0&#xff0c;因为要尽可能使用更少的空间&#xff0c;且新设置为0的格子不能对后续的判断产生影响&#xff0c;所以要在原有矩阵上存储该信息先用两个参数存储第一行和第…

4、创建第一个鸿蒙应用

一、创建项目 此处以空模板为例来创建一个鸿蒙应用。在创建项目前请保持网页的畅通。 1、在欢迎页面点击“Create Project”。 2、左侧默认为“Application”&#xff0c;在“Template Market”中选择空模板&#xff08;Empty Ability&#xff09;&#xff0c;点击“Next” 3…

使用 Electron 来替代本地调试线上代理的场景

Cookie Samesite 问题 https://developers.google.com/search/blog/2020/01/get-ready-for-new-samesitenone-secure?hlzh-cnhttps://www.chromium.org/updates/same-site/https://github.com/GoogleChromeLabs/samesite-exampleshttps://releases.electronjs.org/releases/s…

【numpy】数据类型

1、Numpy Python 的扩展库&#xff08;数学函数库&#xff09;&#xff0c;主要用于数组计算。 N维数组对象ndarray广播功能函数整合C/C/Fortran代码的工具线性代数、傅里叶变换、随机数生成等功能 通常与SciPy&#xff08;开源算法库和数学工具包&#xff09;和Matplotlib&a…