JUnit 5学习笔记

JUnit 5 学习笔记

    • 1.JUnit5的改变
    • 2.JUnit5常用注解及测试
      • 2.1 @DisplayName/@Disabled/@BeforeEach/@AfterEach/@BeforeAll/@AfterAll
      • 2.2 @Timeout
      • 2.3 @RepeatedTest
    • 3.断言
      • 3.1 简单断言
      • 3.2 数组断言
      • 3.3 组合断言
      • 3.4 异常断言
      • 3.5 超时断言
      • 3.6 快速失败
    • 4.前置条件
    • 5.嵌套测试
    • 6.参数化测试
      • 6.1 @ValueSource
      • 6.2. @EnumSource
      • 6.3 @CsvSource 和 @CsvFileSource
      • 6.4. @MethodSource
    • 7.JUnit4 → Junit5

1.JUnit5的改变

JUnit 5 = JUnit Platform + JUnit Jupiter + JUnit Vintage

JUnit Platform: Junit
Platform是在JVM上启动测试框架的基础,不仅支持Junit自制的测试引擎,其他测试引擎也都可以接入。

JUnit Jupiter: JUnit Jupiter提供了JUnit5的新的编程模型,是JUnit5新特性的核心。内部包含了一个测试引擎,用于在Junit Platform上运行。

JUnit Vintage: 由于JUint已经发展多年,为了照顾老的项目,JUnit Vintage提供了兼容JUnit4.x,Junit3.x的测试引擎。

2.JUnit5常用注解及测试

JUnit5的注解与JUnit4的注解有所变化

  • @Test :表示方法是测试方法。但是与JUnit4的@Test不同,他的职责非常单一不能声明任何属性,拓展的测试将会由Jupiter提供额外测试
  • @ParameterizedTest :表示方法是参数化测试,下方会有详细介绍
  • @RepeatedTest :表示方法可重复执行,下方会有详细介绍
  • @DisplayName :为测试类或者测试方法设置展示名称
  • @BeforeEach :表示在每个单元测试之前执行
  • @AfterEach :表示在每个单元测试之后执行 @BeforeAll :表示在所有单元测试之前执行
  • @AfterAll :表示在所有单元测试之后执行
  • @Tag :表示单元测试类别,类似于JUnit4中的@Categories
  • @Disabled :表示测试类或测试方法不执行,类似于JUnit4中的@Ignore
  • @Timeout :表示测试方法运行如果超过了指定时间将会返回错误
  • @ExtendWith :为测试类或测试方法提供扩展类引用

2.1 @DisplayName/@Disabled/@BeforeEach/@AfterEach/@BeforeAll/@AfterAll

@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {//----------------------------常用注解----------------------------
//@DisplayName("测试DisplayName注解---方法1")@Testpublic void testDisplayName1() {System.out.println(123);}@Disabled@DisplayName("测试DisplayName注解---方法2")@Testpublic void testDisplayName2() {System.out.println(456);}@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}
}

在这里插入图片描述

我们再把 testDisplayName2 方法上的@Disabled注解打开。可以看到,这个测试方法已经被禁用了。

在这里插入图片描述

2.2 @Timeout

package com.szh.boot;import org.junit.jupiter.api.*;import java.time.Duration;
import java.util.concurrent.TimeUnit;import static org.junit.jupiter.api.Assertions.*;@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}@DisplayName("测试Timeout注解")@Timeout(value = 5, unit = TimeUnit.MILLISECONDS)@Testpublic void testTimeout() throws InterruptedException {Thread.sleep(1000);}
}

在这里插入图片描述

2.3 @RepeatedTest

package com.szh.boot;import org.junit.jupiter.api.*;import java.time.Duration;
import java.util.concurrent.TimeUnit;import static org.junit.jupiter.api.Assertions.*;@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}@RepeatedTest(5)@Testpublic void testRepeatedTest() {System.out.println(1);}
}

在这里插入图片描述

3.断言

断言(assertions)是测试方法中的核心部分,用来对测试需要满足的条件进行验证。这些断言方法都是org.junit.jupiter.api.Assertions 的静态方法。JUnit 5 内置的断言可以分成如下几个类别:

3.1 简单断言

用来对单个值进行简单的验证。如:

方法说明
assertEquals判断两个对象或两个原始类型是否相等
assertNotEquals判断两个对象或两个原始类型是否不相等
assertSame判断两个对象引用是否指向同一个对象
assertNotSame判断两个对象引用是否指向不同的对象
assertTrue判断给定的布尔值是否为 true
assertFalse判断给定的布尔值是否为 false
assertNull判断给定的对象引用是否为 null
assertNotNull判断给定的对象引用是否不为 null
package com.szh.boot;import org.junit.jupiter.api.*;import java.time.Duration;
import java.util.concurrent.TimeUnit;import static org.junit.jupiter.api.Assertions.*;@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}//----------------------------断言----------------------------  /**    * 如果前面的断言失败,则后面的代码不会再执行    * * */   @DisplayName("测试简单断言")@Testpublic void testSimpleAssertions() {int result = cal(1, 2);assertEquals(5, result, "业务逻辑计算结果出错....");Object obj1 = new Object();Object obj2 = new Object();assertSame(obj1, obj2);}public int cal(int x, int y) {return x + y;}
}

assertEquals方法判断出运算结果不同,也即此时第一个断言失败了,那么后面的assertSame不会再执行。
当我们修改assertEquals方法使其断言成功,那么后续的assertSame将会继续执行。

在这里插入图片描述

在这里插入图片描述

3.2 数组断言

通过 assertArrayEquals 方法来判断两个对象或原始类型的数组是否相等。

package com.szh.boot;import org.junit.jupiter.api.*;import java.time.Duration;
import java.util.concurrent.TimeUnit;import static org.junit.jupiter.api.Assertions.*;@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}     //----------------------------断言----------------------------    @DisplayName("测试数组断言1")@Testpublic void testArrayAssertions1() {assertArrayEquals(new int[]{1, 2}, new int[]{1, 2});}@DisplayName("测试数组断言2")@Testpublic void testArrayAssertions2() {assertArrayEquals(new int[]{3, 4}, new int[]{1, 2}, "数组内容不相等....");}
}

在这里插入图片描述

在这里插入图片描述

3.3 组合断言

assertAll 方法接受多个 org.junit.jupiter.api.Executable 函数式接口的实例作为要验证的断言,可以通过 lambda 表达式很容易的提供这些断言。
组合断言中,当所有断言都成功时,代码才会顺利向下执行。assertTrue虽然执行成功了,但是assertEquals失败了,所有下面的sout不会执行。

package com.szh.boot;import org.junit.jupiter.api.*;import java.time.Duration;
import java.util.concurrent.TimeUnit;import static org.junit.jupiter.api.Assertions.*;@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}     //----------------------------断言---------------------------- @DisplayName(("测试组合断言"))@Testpublic void testAllAssertions() {assertAll("testAll", () -> assertTrue(true && true),() -> assertEquals(1, 2));//当以上两个断言全部成功时,才会打印下面的内容 System.out.println("success!!!");}
}

在这里插入图片描述

3.4 异常断言

在JUnit4时期,想要测试方法的异常情况时,需要用@Rule注解的ExpectedException变量还是比较麻烦的。而JUnit5提供了一种新的断言方式Assertions.assertThrows() ,配合函数式编程就可以进行使用。

package com.szh.boot;import org.junit.jupiter.api.*;import java.time.Duration;
import java.util.concurrent.TimeUnit;import static org.junit.jupiter.api.Assertions.*;@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}     //----------------------------断言----------------------------    @DisplayName("测试异常断言")@Testpublic void testExceptionAssertions() {assertThrows(ArithmeticException.class, () -> {int i = 10 / 2;//此时断言会执行            // int j = 10 / 0; // 此时断言不会执行       }, "业务逻辑居然正常运行???");}
}

如果代码执行 int i = 10 / 2 运行正确,不会出现 ArithmeticException 异常,所以此时assertThrows断言会执行。反之则不会执行。

在这里插入图片描述

3.5 超时断言

Junit5还提供了Assertions.assertTimeout() 为测试方法设置了超时时间。

package com.szh.boot;import org.junit.jupiter.api.*;import java.time.Duration;
import java.util.concurrent.TimeUnit;import static org.junit.jupiter.api.Assertions.*;@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}@DisplayName("测试超时断言")@Testpublic void testTimeoutAssertions() {assertTimeout(Duration.ofMillis(1000), () -> Thread.sleep(2000));}
}

在这里插入图片描述

3.6 快速失败

通过 fail 方法直接使得测试失败。

package com.szh.boot;import org.junit.jupiter.api.*;import java.time.Duration;
import java.util.concurrent.TimeUnit;import static org.junit.jupiter.api.Assertions.*;@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}@DisplayName("快速失败")@Testpublic void testFailAssertions() {System.out.println(1);System.out.println(2);fail("直接走人....");System.out.println(3);System.out.println(4);}
}

在这里插入图片描述

4.前置条件

JUnit 5 中的前置条件(assumptions【假设】)类似于断言,不同之处在于不满足的断言会使得测试方法失败,而不满足的前置条件只会使得测试方法的执行终止。前置条件可以看成是测试方法执行的前提,当该前提不满足时,就没有继续执行的必要。

package com.szh.boot;import org.junit.jupiter.api.*;import java.time.Duration;
import java.util.concurrent.TimeUnit;import static org.junit.jupiter.api.Assertions.*;@SpringBootTest
@DisplayName("Junit5功能测试类")
public class TestJunit {@BeforeEachpublic void testBeforeEach() {System.out.println("测试就要开始了....");}@AfterEachpublic void testAfterEach() {System.out.println("测试已经结束了....");}@BeforeAllpublic static void testBeforeAll() {System.out.println("所有测试就要开始了....");}@AfterAllpublic static void testAfterAll() {System.out.println("所有测试已经结束了....");}     //----------------------------前置条件----------------------------    @DisplayName("测试前置条件")@Testpublic void testAssumptions() {Assumptions.assumeTrue(true, "结果不是true....");System.out.println(111);}
}

在这里插入图片描述

将 Assumptions.assumeTrue(false, “结果不是true…”); 中的 true 改为 false,此时因为 true != false,所以不满足该前置条件,则直接导致测试方法被终止执行。

在这里插入图片描述

5.嵌套测试

JUnit 5 可以通过 Java 中的内部类和@Nested 注解实现嵌套测试,从而可以更好的把相关的测试方法组织在一起。在内部类中可以使用@BeforeEach 和@AfterEach 注解,而且嵌套的层次没有限制。

package com.szh.boot;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 java.util.EmptyStackException;
import java.util.Stack;import static org.junit.jupiter.api.Assertions.*;/*** 嵌套测试中,外层的@Test不能驱动内层的@Before(After)Each/All之类的方法提前/之后运行          * 内层的@Test可以驱动外层的@Before(After)Each/All之类的方法提前/之后运行*/
@DisplayName("A stack")
public class TestingAStackDemo {Stack<Object> stack;@Test@DisplayName("is instantiated with new Stack()")void isInstantiatedWithNew() {new Stack<>();}@Nested@DisplayName("when new")class WhenNew {@BeforeEachvoid createNewStack() {stack = new Stack<>();}@Test@DisplayName("is empty")void isEmpty() {assertTrue(stack.isEmpty());}@Test@DisplayName("throws EmptyStackException when popped")void throwsExceptionWhenPopped() {assertThrows(EmptyStackException.class, stack::pop);}@Test@DisplayName("throws EmptyStackException when peeked")void throwsExceptionWhenPeeked() {assertThrows(EmptyStackException.class, stack::peek);}@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());}@Test@DisplayName("returns the element when popped and is empty")void returnElementWhenPopped() {assertEquals(anElement, stack.pop());assertTrue(stack.isEmpty());}@Test@DisplayName("returns the element when peeked but remains not empty")void returnElementWhenPeeked() {assertEquals(anElement, stack.peek());assertFalse(stack.isEmpty());}}}
}

6.参数化测试

参数化测试是JUnit5很重要的一个新特性,它使得用不同的参数多次运行测试成为了可能,也为我们的单元测试带来许多便利。

利用@ValueSource等注解,指定入参,我们将可以使用不同的参数进行多次单元测试,而不需要每新增一个参数就新增一个单元测试,省去了很多冗余代码。

  • @ValueSource: 为参数化测试指定入参来源,支持八大基础类以及String类型,Class类型

  • @NullSource: 表示为参数化测试提供一个null的入参

  • @EnumSource: 表示为参数化测试提供一个枚举入参

  • @CsvFileSource:表示读取指定CSV文件内容作为参数化测试入参

  • @MethodSource:表示读取指定方法的返回值作为参数化测试入参(注意方法返回需要是一个流)

6.1 @ValueSource

@ValueSource 是参数化测试的最简单形式。它允许你为测试方法提供一组简单的值,就像是一份精选的薯片口味列表。

应用场景 :适用于测试方法只有一个参数,而且这个参数的类型为简单类型(如 int、long、double、float、boolean、String)。

案例:

import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.ValueSource;class ValueSourceExampleTest {@ParameterizedTest@ValueSource(strings = {"apple", "banana", "orange"})void testFruitNames(String fruitName) {assertNotNull(fruitName);}
}

注意事项 :测试方法的参数列表中的参数类型必须与 @ValueSource 中的类型相匹配。例如,如果你使用 @ValueSource(ints = {…}),那么测试方法的参数类型必须是 int 或 Integer。

在这里插入图片描述

6.2. @EnumSource

@EnumSource 是针对枚举类型的参数化测试。它允许你为测试方法提供一个枚举类型的值集合,就像是为不同口味的薯片定义了一个标准分类。

应用场景 :适用于测试方法只有一个参数,而且这个参数的类型为枚举类型。

案例 :

package com.example.test;import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.EnumSource;import static org.junit.jupiter.api.Assertions.assertNotNull;public class EnumSourceExampleTest {enum Size {SMALL, MEDIUM, LARGE}@ParameterizedTest@EnumSource(Size.class)void testSizes(Size size) {assertNotNull(size);}}

在这里插入图片描述

注意事项 :测试方法的参数列表中的参数类型必须与 @EnumSource 中指定的枚举类型相匹配。

6.3 @CsvSource 和 @CsvFileSource

@CsvSource 和 @CsvFileSource 是基于 CSV 格式的参数化测试。它们允许你为测试方法提供一组 CSV 格式的值,就像是一张详细的薯片销售报表。

应用场景 :适用于测试方法有多个参数,需要以 CSV 格式提供输入数据。

案例:

package com.example.test;import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.CsvSource;import static org.junit.jupiter.api.Assertions.assertEquals;public class CsvSourceExampleTest {@ParameterizedTest@CsvSource({"1, 2, 3", "4, 5, 9", "6, 7, 13"})void testAddition(int a, int b, int expectedSum) {assertEquals(expectedSum, a + b);}
}

在这里插入图片描述
对于 @CsvFileSource,你需要将 CSV 数据存储在一个文件中,并通过 resources 属性指定文件路径。

package com.example.test;import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.CsvFileSource;import static org.junit.jupiter.api.Assertions.assertEquals;public class CsvFileSourceExampleTest {@ParameterizedTest@CsvFileSource(resources = "/addition.csv", numLinesToSkip=1)void testAddition(int a, int b, int expectedSum) {assertEquals(expectedSum, a + b);}
}

在这里插入图片描述

在这里插入图片描述

注意事项 :确保 CSV 数据与测试方法的参数类型和数量相匹配。对于 @CsvFileSource,文件应该放在项目的资源文件夹中。

6.4. @MethodSource

@MethodSource是基于方法返回值的参数化测试。它允许你为测试方法提供一个方法的返回值作为参数,就像是一个智能贩卖机,根据顾客的喜好自动调整薯片口味。

应用场景 :适用于测试方法有多个参数,需要从方法返回值中获取输入数据。

案例 :

package com.example.test;import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.MethodSource;import java.util.stream.Stream;import static org.junit.jupiter.api.Assertions.assertEquals;public class MethodSourceExampleTest {@ParameterizedTest@MethodSource("provideStrings")void testStringLength(String input, int expectedLength) {assertEquals(expectedLength, input.length());}static Stream<Arguments> provideStrings() {return Stream.of(Arguments.of("apple", 5),Arguments.of("banana", 6),Arguments.of("orange", 6));}
}

在这里插入图片描述

注意事项 :确保提供方法的返回值与测试方法的参数类型和数量相匹配。提供方法必须是静态方法,且无需参数。

7.JUnit4 → Junit5

在进行迁移的时候需要注意如下的变化:

  • 注解在 org.junit.jupiter.api 包中,断言在 org.junit.jupiter.api.Assertions 类中,前置条件在org.junit.jupiter.api.Assumptions 类中。
  • 把@Before 和@After替换成@BeforeEach 和@AfterEach。
  • 把@BeforeClass 和@AfterClass 替换成@BeforeAll和@AfterAll。
  • 把@Ignore 替换成@Disabled。 把@Category 替换成@Tag。
  • 把@RunWith、@Rule 和@ClassRule 替换成@ExtendWith。

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

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

相关文章

2.2 利用MyBatis实现CRUD操作

MyBatis 是一个半自动的持久层框架&#xff0c;它简化了数据库操作&#xff0c;允许开发者通过 XML 或注解的方式来配置 SQL 语句&#xff0c;实现数据的增删改查&#xff08;CRUD&#xff09;操作。 1. 环境搭建 引入依赖&#xff1a;在项目中添加 MyBatis 以及数据库驱动的…

【每日刷题】Day65

【每日刷题】Day65 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;每日刷题&#x1f34d; &#x1f33c;文章目录&#x1f33c; 1. LCR 175. 计算二叉树的深度 - 力扣&#xff08;LeetCode&#xff09; 2. 序列找数_牛客题霸_牛客网…

upload-labs第十二关教程

upload-labs第十二关教程 一、源代码分析代码审计 二、绕过分析截断绕过magic_quotes_gpc函数介绍关闭magic_quotes_gpc上传eval.png文件使用burpsuite进行抓包修改放包&#xff0c;查看是否上传成功使用中国蚁剑进行连接 一、源代码分析 代码审计 $is_upload false; $msg …

2024年6个恢复删除数据的方法,看这篇就够了~

在数字化飞速发展的今天&#xff0c;数据已成为我们生活中不可或缺的组成部分&#xff0c;它记录着我们的记忆、创意和辛勤付出。然而&#xff0c;生活总是充满意外&#xff0c;我们可能会遭遇数据意外删除或丢失的困境。在这种关键时刻&#xff0c;如何高效、准确地恢复数据就…

PG数据库插入数据,正则字段的反斜杠丢了

问题描述 将公司数据发现场&#xff0c;导入后发现正则的\"丢"了。其实&#xff0c;是当成转义符了。但是根据下面PG官网提示自PostgreSQL9.1开始默认值就是on&#xff0c;现场肯定是有人主动off了。太坑了~~~ standard_conforming_stringson,\失去了转义&#xff0…

【Kafka】Kafka Producer 分区-05

【Kafka】Kafka Producer 分区-05 1. 分区的好处2. 分区策略2.1 默认的分区器 DefaultPartitioner 3. 自定义分区器 1. 分区的好处 &#xff08;1&#xff09;便于合理使用存储资源&#xff0c;每个Partition在一个Broker上存储&#xff0c;可以把海量的数据按照分区切割成一块…

MyBatis的逆向工程详细步骤操作

1. MyBatis的逆向工程详细步骤操作 文章目录 1. MyBatis的逆向工程详细步骤操作2. 逆向工程配置与生成2.1 MyBatis3Simple&#xff1a;基础版&#xff0c;只有基本的增删改查2.1.1 第一步&#xff1a;在pom.xml 中添加逆向工程插件2.1.2 第二步&#xff1a;配置 generatorConfi…

Ubuntu 18.04下普通用户的一次提权过程

Ubuntu 18.04下普通用户的一次提权过程 一.背景介绍:二.主要调试过程:三.相关命令:1.设置BMC密码,获取BMC IP2.找一台ubuntu搭建TFTP服务,用来替换grub.cfg文件3.从调试服务器的/boot/grub/grub.cfg中提取出recovery mode的配置,简化并生成新的配置文件grub.cfg,放在tftp服务的…

万能破题方法包(3)暴力破解法

一、前言 暴力破解法是指通过尝试所有可能的密码组合来破解密码 1.1、概念 暴力破解法是一种通过尝试所有可能的密码组合来破解密码的方法。它基于暴力的方式&#xff0c;不依赖于任何密码漏洞或特殊技巧&#xff0c;而是通过穷举所有可能性来找到正确的密码。 1.2、解决步骤 …

Qt项目天气预报(2) - 重写事件函数

鼠标右键实现退出界面 知识点QMenu: QMenu 弹出对话框 --> 相对QMessageBox 更加轻量点 QMenu是Qt库中用于创建弹出式菜单的类&#xff0c;它通常出现在应用程序的顶部菜单栏、按钮的右键菜单或自定义上下文菜单中。以下是关于QMenu的详细介绍&#xff1a; 1. 类的基本特…

【WEB前端2024】3D智体编程:乔布斯3D纪念馆-第38课-密室逃脱-3D互动剧情

【WEB前端2024】3D智体编程&#xff1a;乔布斯3D纪念馆-第38课-密室逃脱 使用dtns.network德塔世界&#xff08;开源的智体世界引擎&#xff09;&#xff0c;策划和设计《乔布斯超大型的开源3D纪念馆》的系列教程。dtns.network是一款主要由JavaScript编写的智体世界引擎&…

热镀锌钢板耐液体性能测 彩钢板抗拉强度检测

钢板检测范围&#xff1a;钢板、彩钢板、不锈钢板、耐磨钢板、合金钢板、压型钢板、冷轧钢板、弹簧钢板、碳钢板、热轧钢板、厚钢板、热镀锌钢板、冲孔钢板、船用钢板、硅钢板、花纹钢板、压力容器钢板、耐候钢板、 钢板检测项目包括化学性能检测、性能检测、机械性能检测、老…

AI办公自动化:kimi批量搜索提取PDF文档中特定文本内容

工作任务&#xff1a;PDF文档中有资料来源这一行&#xff0c;比如&#xff1a; 资料来源&#xff1a;moomoo tech、The Information、Bloomberg、Reuters&#xff0c;浙商证券研究所 数据来源&#xff1a;CSDN、浙商证券研究所 数据来源&#xff1a;CSDN、arXiv、浙商证券研…

C++ 19 之 封装

c19封装.cpp #include <iostream> #include <string.h> using namespace std;// 封装&#xff1a;将行为和属性作为一个整体来表现生活中的事物// 人&#xff1a; 行为&#xff1a; 吃饭 属性&#xff1a; 姓名、年龄 struct person {char name[20];int age;…

掌握特劳特定位理论核心,明晰企业战略定位之重

在当今瞬息万变的市场环境中&#xff0c;企业战略定位的重要性日益凸显。它不仅是企业在激烈竞争中保持优势的关键&#xff0c;更是企业实现长期可持续发展的基石。 哈佛大学战略学教授迈克尔波特&#xff08;Michael Porter&#xff09;指出战略就是形成一套独具的运营活动&a…

为什么微信输入法是比搜狗输入法更好的选择?

微信输入法官网&#xff1a;https://z.weixin.qq.com/ 最近使用搜狗输入法时&#xff0c;频繁弹出广告&#xff0c;实在令人烦恼&#xff0c;于是我干脆卸载了它。然而&#xff0c;电脑上没有输入法是不行的。经过在网上对比了许多输入法软件后&#xff0c;我发现了微信输入法。…

skywalking9.4 链路追踪

下载&#xff0c;很慢很慢很慢&#xff01;&#xff01;&#xff01;&#xff01; jdk 使用jdk17 skywalking-apm 9.4 java-agent 9.0 idea 本地开发配置 第1行配置按实际来&#xff1b; 第2行自定义&#xff0c;一般和微服务名称相同&#xff1b; 第3行ip写安装的机器ip,端…

装备名称检索与推荐

1、引言 在这个信息爆炸的时代&#xff0c;无论是军事爱好者、科研工作者&#xff0c;还是户外探险者&#xff0c;他们都需要快速准确地获取特定装备的信息。装备名称检索推荐系统正是为了应对这一挑战而生。它像一位经验丰富的向导&#xff0c;引领用户穿越复杂的装备海洋&am…

微信公众号打通与登录的实现

今天实现一下与微信公众号进行对接&#xff0c;通过扫描二维码的方式来进行注册与登录&#xff0c;获取用户的微信唯一标识作为用户的username&#xff0c;下面我们开始编写。 骨架建立&#xff1a; 建包&#xff1a; 第一步还是先将骨架建好&#xff0c;与网关骨架差不多&a…

玄机平台应急响应—MySQL应急

前言 这个是比较简单的&#xff0c;其实和MySQL没啥太大的关系&#xff0c;没涉及太多MySQL的知识。看一下它的flag要求吧。 flag1 它说黑客写入的shell&#xff0c;那我们就去它的网站目录去看看&#xff0c;果然有一个叫sh.php的文件。 flag1{ccfda79e-7aa1-4275-bc26-a61…