头歌JUnit单元测试相关实验入门

一、入门实验

1.1第一个Junit测试程序

任务描述

请学员写一个名为testSub()的测试函数,来测试给定的减法函数是否正确。

相关知识
Junit编写原则

1、简化测试的编写,这种简化包括测试框架的学习和实际测试单元的编写。 2、测试单元保持持久性。 3、利用既有的测试来编写相关的测试。

Junit特征

1、使用断言方法判断期望值和实际值差异,返回Boolean值

2、测试驱动设备使用共同的初始化变量或者实例。

3、测试包结构便于组织和集成运行。

4、支持图形交互模式和文本交互模式。

Junit框架的组成

1、测试用例(TestCase):对测试目标进行测试的方法与过程的集合

2、测试包(TestSuite):测试用例的集合,可容纳多个测试用例(TestCase)。

3、测试结果(TestResult):测试结果的描述与记录。

4、测试监听(TestListener):测试过程中事件的监听者。

5、测试失败元素(TestFailure):每一个测试方法所发生的与预期不一致状况的描述。

6、测试框架出错异常(AssertionFailedError):junit执行测试时所抛出的异常。

Junit作用介绍

  通常我们写完代码想要测试这段代码的正确性,那么必须新建一个类,然后创建一个 main() 方法,之后再编写测试代码。如果需要测试的代码很多呢?那么要么就会建很多main() 方法来测试,要么将其全部写在一个main()方法里面。这也会大大的增加测试的复杂度,降低程序员的测试积极性。而 Junit 能很好的解决这个问题,简化单元测试,写一点测一点,在之后的代码改动中如果发现问题可以较快的追踪到问题的原因,减小回归错误的纠错难度。

如何编写Junit测试

首先,我们将介绍一个测试类:Calculate.java

//Calculate.java
package com.trustie.junitest;
public class Calculate {
    public int sum(int var1, int var2) {
        return var1 + var2;
    }
}

  在上面的代码中,我们可以看到,Calculate类有一个公共的方法sum(), 它接收输入两个整数,将它们相加并返回结果。在这里,我们将测试这个方法。为了这个目的,我们将创建另一个类及其方法,将测试之前的类(在此情况下,我们只有一个方法进行测试)中的方法,这是使用的最常见的方式。当然,如果一个方法非常复杂且要扩展,我们可以用多个试验方法来对其进行测试。创建测试用例的详细信息将显示在下面的部分。下面,有一个类是:CalculateTest.java,它具有我们的测试类的角色的代码:

// CalculateTest.java
package com.trustie.test;
import org.junit.Test;
import static org.junit.Assert.assertEquals;
import com.trustie.junitest.Calculate;

public class CalculateTest {
  // 创建一个Calculate对象
  Calculate calculation = new Calculate();
  // 调用sum方法计算2和5的和,并将结果赋值给变量sum
  int sum = calculation.sum(2, 5);
  // 定义一个预期的和值
  int testSum = 7;

  // 使用@Test注解标记测试方法
  @Test
  public void testSum() {
      // 使用assertEquals方法比较实际计算结果和预期结果是否相等
      assertEquals(sum, testSum);
  }
}
    先来解释一下上面的代码。首先,我们可以看到,有一个@Test的注解testSum()方法的上方。 这个注释指示该方法它所附着的代码可以做为一个测试用例。因此,testSum()方法将用于测试公开方法 sum() 。 我们再观察一个方法 assertEquals(sum, testsum)    assertEquals ([String message], object expected, object actual) 方法持有两个对象作为输入,并断言这两个对象相等。

然后在Bash执行:

// 编译Calculate.java文件,生成字节码文件
javac -d . Calculate.java

// 编译CalculateTest.java文件,生成字节码文件
javac -d . CalculateTest.java

// 运行CalculateTest测试类,使用JUnit框架进行单元测试
java org.junit.runner.JUnitCore com.trustie.test.CalculateTest
就可以看到:
JUnit version 4.12
.
Time: 0.003
OK (1 test)
这里首先打印出了JUnit版本号,然后输出了耗时和测试结果。在这里,我们的测试结果是OK,证明测试通过,原函数功能正确。
编程要求

本关的编程任务是在JunitSubTest.java中的补全测试函数testSub(),具体要求:用subtestSub作为参数,来验证JunitSub.javasub函数是否正确的是否正确。

本关涉及的代码文件JunitSub.java的代码如下:

package step1;
public class JunitSub {
    public int sub(int var1, int var2) {
        return var1 - var2;
    }
}
本关涉及的代码文件JunitSubTest.java的代码如下:
package step1;
import org.junit.Test;
import static org.junit.Assert.assertEquals;
import step1.JunitSub;
public class JunitSubTest {
     //引入JunitSub对象
     JunitSub js = new JunitSub();
     int sub = js.sub(5,2);
     int testSub = 3;
     /*
     请在下面的Begin/End内写一个测试函数,
     来验证JunitSub中的sub函数编写是否正确
     */
     /***********************Begin**************************/
     /************************End***************************/
}
评测说明

本关卡的测试文件是TestRunner.java,该文件进行了函数封装且学员不可见,用于验证学员的Junit测试代码是否正确。

具体测试过程如下:

1.平台自动编译生成TestRunner.exe;

2.平台运行TestRunner.exe

3.获取TestRunner.exe输出,并将其输出与预期输出对比:如果一致则测试通过,否则测试失败。

预期输入: 预期输出:true

友情提示

1.请不要直接println最终输出,否则平台发现此类情况后,将一律扣掉本关经验值,并且追加处罚措施。

2.学员答题时请尽量手敲代码,请勿从实训讲解代码片段中复制代码段粘贴到答题区域作答,复制的内容会保留一些格式和字符,导致编译失败。

开始你的任务吧,祝你成功!

package step1;

import org.junit.Test;
import static org.junit.Assert.assertEquals;
import step1.JunitSub;

public class JunitSubTest {
     //引入JunitSub对象
     JunitSub js = new JunitSub();
     int sub = js.sub(5,2);
     int testSub = 3;

     /*
     请在下面的Begin/End内写一个测试函数,
     来验证JunitSub中的sub函数编写是否正确
     */
     /***********************Begin**************************/

     @Test
     public void testSubFunction() {
          assertEquals(testSub, sub);
     }
     /************************End***************************/

}

 1.2Junit注解

相关知识
Junit注解

Java注解((Annotation)的使用方法是"@ + 注解名" 。借助注解,我们可以在编程中通过简单的注解来实现一些功能。在junit中常用的注解有 @Test、@Ignore、@BeforeClass、@AfterClass、@Before、@After 下表列出了这些注释的概括:

具体解释如下:

1、@Test,表明此方法为测试方法。

2、@Before,用此注解修饰的方法在每个test方法运行前执行

3、@BeforeClass,用此注解修饰的方法将在所有方法运行前被执行,是一个static方法,只执行一次。

4、@After,用此注解修饰的方法在每个test方法运行后执行

5、@AfterClass,用此注解修饰的方法将在所有方法运行后被执行,也是一个static方法,只执行一次。

6、@Ignore,用此注解修饰的方法会被Junit忽略。

代码示例

这里新建一个JunitAnnotation.java,把上面所讲的注解全部加到某个测试函数之前,这些注解的作用一目了然:

// 导入JUnit相关的类和静态方法
import static org.junit.Assert.*;
import java.util.*;
import org.junit.*;

// 定义一个名为AnnotationsTest的公共类,用于进行单元测试
public class AnnotationsTest {
    // 定义一个私有的ArrayList变量testList,用于存储测试数据
    private ArrayList testList;

    // 使用@BeforeClass注解标记的方法,在所有的测试方法执行前只执行一次
    @BeforeClass
    public static void onceExecutedBeforeAll() {
        // 输出提示信息
        System.out.println("@BeforeClass: onceExecutedBeforeAll");
    }

    // 使用@Before注解标记的方法,在每个测试方法执行前都会执行一次
    @Before
    public void executedBeforeEach() {
        // 初始化testList为一个新的ArrayList对象
        testList = new ArrayList();
        // 输出提示信息
        System.out.println("@Before: executedBeforeEach");
    }

    // 使用@AfterClass注解标记的方法,在所有的测试方法执行后只执行一次
    @AfterClass
    public static void onceExecutedAfterAll() {
        // 输出提示信息
        System.out.println("@AfterClass: onceExecutedAfterAll");
    }

    // 使用@After注解标记的方法,在每个测试方法执行后都会执行一次
    @After
    public void executedAfterEach() {
        // 清空testList中的所有元素
        testList.clear();
        // 输出提示信息
        System.out.println("@After: executedAfterEach");
    }

    // 使用@Test注解标记的方法,表示这是一个测试方法
    @Test
    public void EmptyCollection() {
        // 断言testList为空,如果为空则测试通过,否则测试失败
        assertTrue(testList.isEmpty());
        // 输出提示信息
        System.out.println("@Test: EmptyArrayList");
    }

    // 使用@Test注解标记的方法,表示这是一个测试方法
    @Test
    public void OneItemCollection() {
        // 向testList中添加一个元素
        testList.add("oneItem");
        // 断言testList的大小为1,如果大小为1则测试通过,否则测试失败
        assertEquals(1, testList.size());
        // 输出提示信息
        System.out.println("@Test: OneItemArrayList");
    }

    // 使用@Ignore注解标记的方法,表示这个测试方法被忽略,不会被执行
    @Ignore
    public void executionIgnored() {
        // 输出提示信息
        System.out.println("@Ignore: This execution is ignored");
    }
}

如果我们运行上面的测试,控制台输出将是下面:

@BeforeClass: onceExecutedBeforeAll
@Before: executedBeforeEach
@Test: EmptyArrayList
@After: executedAfterEach
@Before: executedBeforeEach
@Test: OneItemArrayList
@After: executedAfterEach
@AfterClass: onceExecutedAfterAll
// 在所有测试方法执行前只执行一次的方法
@BeforeClass
public void onceExecutedBeforeAll() {
    // ...
}

// 在每个测试方法执行前都会执行的方法
@Before
public void executedBeforeEach() {
    // ...
}

// 测试方法1:空的ArrayList
@Test
public void EmptyArrayList() {
    // ...
}

// 在每个测试方法执行后都会执行的方法
@After
public void executedAfterEach() {
    // ...
}

// 在每个测试方法执行前都会执行的方法(重复)
@Before
public void executedBeforeEach() {
    // ...
}

// 测试方法2:只有一个元素的ArrayList
@Test
public void OneItemArrayList() {
    // ...
}

// 在所有测试方法执行后只执行一次的方法
@AfterClass
public void onceExecutedAfterAll() {
    // ...
}
编程要求

本关的编程任务是在JunitAnnotation.java中修改测试函数对应的注解,使得原代码输出结果变为逆序。

本关涉及的代码文件JunitAnnotation.java的代码如下:

见下答案版

评测说明

本关卡的测试文件是TestRunner.java,该文件进行了函数封装且学员不可见,用于验证学员的Junit测试代码是否正确。

具体测试过程如下:

1.平台自动编译生成TestRunner.exe; 2.平台运行TestRunner.exe; 3.获取TestRunner.exe输出,并将其输出与预期输出对比:如果一致则测试通过,否则测试失败。

预期输入: 预期输出:

in after class
in after
in test
in before
in before class
true
package step2;

import org.junit.After;
import org.junit.AfterClass;
import org.junit.Before;
import org.junit.BeforeClass;
import org.junit.Ignore;
import org.junit.Test;

public class JunitAnnotation {
    /*
     *以下Junit测试程序的输出结果为:
     *in before class
     *in before
     *in test
     *in after
     *in after class
     *请修改下面Begin/End内各个测试函数的注解,使输出结果逆序
     */
     /***********************Begin**************************/

  
   @BeforeClass
   public static void afterClass() {
      System.out.println("in after class");
   }

   //execute after test
   
   @Before
   public void after() {
      System.out.println("in after");
   }

   //execute before test
   @After
   public void before() {
      System.out.println("in before");
   }

   //execute after class
   @AfterClass
   public static void beforeClass() {
      System.out.println("in before class");
   }

   //test case
   @Test
   public void test() {
      System.out.println("in test");
   }

   //execute before class
   
    /************************End***************************/
}

 1.3Junit断言

任务描述

给定一个断言测试类AssertionsTest.java,请按要求补全代码,写出相应的断言测试。

相关知识
Junit断言

Junit的断言方法允许检查测试方法的期望结果值和真实返回值。Junit的org.junit.Assert类提供了各种断言方法来写junit测试,这些方法被用来检查方法的真实结果值和期望值。

代码示例

创建一个文件名为 TestAssertions.java 的类,如下:

package com.trustie.junittest;
import org.junit.Test;
import static org.junit.Assert.*;

public class TestAssertions {
   @Test
   public void testAssertions() {
      //测试数据
      String str1 = new String ("abc");
      String str2 = new String ("abc");
      String str3 = null;
      String str4 = "abc";
      String str5 = "abc";
      int val1 = 5;
      int val2 = 6;
      String[] expectedArray = {"one", "two", "three"};
      String[] resultArray =  {"one", "two", "three"};

      //检查两个对象是否相等
      assertEquals(str1, str2);
      
      //检查条件是否为真
      assertTrue (val1 < val2);
      //检查条件是否为假
      assertFalse(val1 > val2);
      //检查对象不为空
      assertNotNull(str1);
      //检查对象为空
      assertNull(str3);
      //检查两个对象引用是否指向同一个对象
      assertSame(str4,str5);
      //检查两个对象引用是否不指向同一个对象
      assertNotSame(str1,str3);
      //检查两个数组是否相等
      assertArrayEquals(expectedArray, resultArray);
   }
}

在以上类中我们可以看到,这些断言方法是可以工作的。

  • assertEquals() 如果比较的两个对象是相等的,此方法将正常返回;否则失败显示在JUnit的窗口测试将中止。

  • assertSame() 和 assertNotSame() 方法测试两个对象引用指向完全相同的对象。

  • assertNull() 和 assertNotNull() 方法测试一个变量是否为空或不为空(null)。

  • assertTrue() 和 assertFalse() 方法测试if条件或变量是true还是false。

  • assertArrayEquals() 将比较两个数组,如果它们相等,则该方法将继续进行不会发出错误。否则失败将显示在JUnit窗口和中止测试。

编程要求

本关的编程任务是给定一个断言测试类AssertionsTest.java,请按要求补全代码,写出相应的断言测试。

具体要求如下:

1.断言obj1和obj2相等;

2.断言obj3和obj4指向完全相同的对象;

3.断言obj2和obj4指向不同的对象;

4.断言obj1对象不为空;

5.断言obj5对象为空;

6.断言var1小于var2;

7.断言arithmetic1和arithmetic2两个数组相等。

本关涉及的代码文件AssertionsTest.java的代码如下:

见下

评测说明

本关卡的测试文件是TestRunner.java,该文件进行了函数封装且学员不可见,用于验证学员的Junit测试代码是否正确。

具体测试过程如下:

1.平台自动编译生成TestRunner.exe; 2.平台运行TestRunner.exe; 3.获取TestRunner.exe输出,并将其输出与预期输出对比:如果一致则测试通过,否则测试失败。

预期输入: 预期输出:true

package step3;

import static org.junit.Assert.*;
import org.junit.Test;

public class AssertionsTest {
    String obj1 = "junit";
    String obj2 = "junit";
    String obj3 = "test";
    String obj4 = "test";
    String obj5 = null;
    int var1 = 1;
    int var2 = 2;
    int[] arithmetic1 = { 1, 2, 3 };
    int[] arithmetic2 = { 1, 2, 3 };

    @Test
    public void test() {
        //请在下面的Begin/End内写添加断言测试的代码,不要改动其他地方的代码
        /***********************Begin**************************/
       
        assertEquals(obj1,obj2); // 判断两个字符串是否相等
        assertSame(obj3,obj4); // 判断两个对象引用是否指向同一个对象
        assertNotSame(obj2,obj4); // 判断两个对象引用是否不指向同一个对象
        assertNotNull(obj1); // 判断对象引用是否不为空
        assertNull(obj5); // 判断对象引用是否为空
        assertTrue (var1 < var2); // 判断一个整数是否小于另一个整数
        assertArrayEquals(arithmetic1,arithmetic2); // 判断两个数组是否相等
		/************************End***************************/
    }
}

1.4Junit时间测试

任务描述

要求学员实现一个Junit时间测试:超过1000毫秒未执行结束就判定测试未通过。

相关知识
Junit时间测试

Junit 提供了一个暂停的方便选项。如果一个测试用例比起指定的毫秒数花费了更多的时间,那么Junit 将自动将它标记为失败。timeout 参数和@Test注释一起使用。现在让我们看看活动中的 @Test(timeout)

代码示例
import org.junit.*;

/**
 * JUnit TimeOut Test
 */
public class JunitTest {
    @Test(timeout = 50000)  //单位是毫秒
    public void infinity() {  
        while (true);  // 无限循环,导致测试超时
    }  
}

在上面的例子中,infinity() 方法将不会返回,因此JUnit引擎将其标记为失败,并抛出一个异常。

java.lang.Exception:test timed out after 50000 milliseconds

编程要求

本关的编程任务是在TestTimeOut.java中实现一个Junit时间测试,超过1000毫秒未执行结束就判定测试未通过。

本关涉及的代码文件TestTimeOut.java的代码如下:

见下述答案

评测说明

本关卡的测试文件是TestRunner.java,该文件进行了函数封装且学员不可见,用于验证学员的Junit测试代码是否正确。

具体测试过程如下:

1.平台自动编译生成TestRunner.exe; 2.平台运行TestRunner.exe; 3.获取TestRunner.exe输出,并将其输出与预期输出对比:如果一致则测试通过,否则测试失败。

预期输入: 预期输出:

test(step4.TestTimeOut): test timed out after 1000 milliseconds
false

package step4;


import org.junit.Test;


public class TestTimeOut {

     //请在下面的Begin/End内补全test()超时测试函数,要求如果超过1000毫秒执行未结束,就判定测试未通过
    /***********************Begin**************************/
    @Test(timeout = 1000)
    public void test() {
        while(true){}
    }
    /************************End***************************/
}

 

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

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

相关文章

【Python】Python给工作减负-读Excel文件生成xml文件

目录 ​前言 正文 1.Python基础学习 2.Python读取Excel表格 2.1安装xlrd模块 2.2使用介绍 2.2.1常用单元格中的数据类型 2.2.2 导入模块 2.2.3打开Excel文件读取数据 2.2.4常用函数 2.2.5代码测试 2.2.6 Python操作Excel官方网址 3.Python创建xml文件 3.1 xml语法…

计算机组成原理,硬件组成,存储器,控制器,控制器的任务, 运算器,中央处理器CPU,主存

计算机组成原理 课程需求 前导课程&#xff1a; 后继课程 汇编 操作系统 数逻 组成 系统结构 数电 微机原理 课程结构 计算机特性 1 从外部角度来看计算机的特性 快速 通用 准确 逻辑 2从外部特性与内部特性的关系 计算机组成 一 硬件组成 运算器 主要功能是进行算术…

强化学习(一)——基本概念及DQN

1 基本概念 智能体 agent &#xff0c;做动作的主体&#xff0c;&#xff08;大模型中的AI agent&#xff09; 环境 environment&#xff1a;与智能体交互的对象 状态 state &#xff1b;当前所处状态&#xff0c;如围棋棋局 动作 action&#xff1a;执行的动作&#xff0c;…

CRM系统是怎样帮助销售流程自动化的?

销售业绩是衡量企业经营的重要指标&#xff0c;也是销售人员一直要达成的目标。销售业绩能否提高取决于销售人员的能力、客户服务水平&#xff0c;还需要借助有效的工具。CRM系统就是这样的一款软件。企业如何提高销售业绩&#xff1f;不妨试试CRM销售流程自动化。 CRM如何实现…

【从删库到跑路 | MySQL总结篇】事务详细介绍

个人主页&#xff1a;兜里有颗棉花糖 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 兜里有颗棉花糖 原创 收录于专栏【MySQL学习专栏】&#x1f388; 本专栏旨在分享学习MySQL的一点学习心得&#xff0c;欢迎大家在评论区讨论&#x1f48c; 目录 一、事务…

JavaScript 数据结构

JavaScript 数据结构 目录 JavaScript 数据结构 一、标识符 二、关键字 三、常量 四、变量 每一种计算机编程语言都有自己的数据结构&#xff0c;JavaScript脚本语言的数据结构包括&#xff1a;标识符、常量、变量、保留字等。 一、标识符 标识符&#xff0c;说白了&…

使用gcloud SDK 管理和部署 Cloud run service

查看cloud run 上的service 列表&#xff1a; gcloud run services list > gcloud run services listSERVICE REGION URL LAST DEPLOYED BY LAST DEPL…

【QT】Windows环境下,cmake引入QML

这里使用的QT库为5.7版本。 1、添加环境变量 QT库根目录环境变量 QTDIR QT库平台插件环境变量 QT_PLUGIN_PATH QML支持环境变量 QML2_IMPORT_PATH &#xff08;该环境变量仅在需要使用QML时添加&#xff09; QT库动态库环境变量&#xff0c;bin目录下包含了QT程序运行所需的dl…

常见的攻击防护

只做模拟机器使用&#xff0c;不使用真实机器 目录 一、 DHCP饿死和防护应对措施.................................. 1 1&#xff0c; 实验拓扑&#xff1a;...................................................... 2 2&#xff0c; 实验配置............................…

AD23等间距拉线、布线的方法

U M 键进行多根走线&#xff0c; 多根走线想保持10个mil 我可以直接按table键,弹出Multi-Routing ponent&#xff0c;项的Bus Spadng输入框中填充10个mil&#xff0c;新走线产生10个mil的等间距 保持最小的一个规则&#xff0c;可以去到6mil线距。 在拉线操作过程中&#…

详解Spring中BeanPostProcessor在Spring工厂和Aop发挥的作用

&#x1f609;&#x1f609; 学习交流群&#xff1a; ✅✅1&#xff1a;这是孙哥suns给大家的福利&#xff01; ✨✨2&#xff1a;我们免费分享Netty、Dubbo、k8s、Mybatis、Spring...应用和源码级别的视频资料 &#x1f96d;&#x1f96d;3&#xff1a;QQ群&#xff1a;583783…

TCP连接为什么是三次握手,而不是两次和四次

答案 阻止重复的历史连接同步初始序列号避免资源浪费 原因 阻止重复的历史连接&#xff08;首要原因&#xff09; 考虑这样一种情况&#xff1a; 客户端现在要给服务端建立连接&#xff0c;向服务端发送了一个SYN报文段&#xff08;第一次握手&#xff09;&#xff0c;以表示请…

Mininet学习记录(常用命令+创建网络拓扑+OpenDaylight显示拓扑结构)

目录 1.Mininet简介2.Mininet常用命令2.1创建网络拓扑常用参数2.2常用的内部交换命令 3.创建网络拓扑的三种方式3.1通过命令行创建3.2通过miniedit可视化界面创建3.3通过python脚本创建 4.问题总结 1.Mininet简介 Mininet 是由一些虚拟的终端节点 (end-hosts) 、交换机、路由器…

【STM32】TIM定时器

第一部分&#xff1a;定时器基本定时的功能&#xff1b; 第二部分&#xff1a;定时器的输出比较功能&#xff1b; 第三部分&#xff1a;定时器输入捕获的功能&#xff1b; 第四部分&#xff1a;定时器的编码接口。 1 TIM简介 TIM&#xff08;Timer&#xff09;定时器&#…

【数据库】数据库基于封锁机制的调度器,使冲突可串行化,保障事务和调度一致性

封锁使可串行化 ​专栏内容&#xff1a; 手写数据库toadb 本专栏主要介绍如何从零开发&#xff0c;开发的步骤&#xff0c;以及开发过程中的涉及的原理&#xff0c;遇到的问题等&#xff0c;让大家能跟上并且可以一起开发&#xff0c;让每个需要的人成为参与者。 本专栏会定期更…

《地理信息系统原理》笔记/期末复习资料(8. 数字高程模型)

目录 8. 数字高程模型 8.1 概述 8.1.1 数字高程模型概念 8.1.2 数字高程模型特点 8.2 DEM数据分布特征 8.2.1 格网状数据 8.2.2 离散数据 8.3 DEM的表示方法 8.3.1 数学方法 8.3.2 图形方法 8.3.3 DEM三维表达方法 8.4 TIN的生成方法 8.4.1 人工方法 8.4.2 程序自…

Android drawable layer-list右上角红点,xml布局实现,Kotlin

Android drawable layer-list右上角红点&#xff0c;xml布局实现&#xff0c;Kotlin <?xml version"1.0" encoding"utf-8"?> <layer-list xmlns:android"http://schemas.android.com/apk/res/android"><itemandroid:id"id…

Vue3获取阴历/农历日期

安装插件 pnpm add chinese-lunar-calendar引入阳历/阴历切换函数 import {getLunar} from chinese-lunar-calendarexport function lunarDate(pDate){const year pDate.getFullYear()const month pDate.getMonth() 1const day pDate.getDate()const result getLunar(yea…

VMware安装Debian12.2作为服务器(无桌面)

[TOC]VMware安装Debian12.2作为服务器&#xff08;无桌面&#xff09; 下载Debian系统 官方网站&#xff1a;https://www.debian.org/index.zh-cn.html 创建新的虚拟机 打开VMware Workstation&#xff0c;点击创建新的虚拟机 向导虚拟机类型选择 一般我会选择典型&…

记录华为云服务器(Linux 可视化 宝塔面板)-- 防火墙篇

文章目录 前言安装防火墙防火墙设置防火墙操作1.设置开机启动防火墙2.查看防火墙开放哪些端口3.重载防火墙配置&#xff08;修改配置后重新启动才生效&#xff09;4.查看防火墙状态5.开启防火墙6.关闭防火墙 若遇到无法开启查询已开放的端口查询端口是否开放&#xff08;80&…