单元测试通常是指对一个函数或方法测试。单元测试的目的是验证每个单元的行为是否符合预期,并且在修改代码时能够快速检测到任何潜在的问题。通过编写测试用例,我们可以验证这些模块在特定输入下是否产生正确的输出。单元测试的目的是确保每个模块在各种情况下都能正常运行。
可以带来以下几个好处:
写单元测试是一种良好的软件开发实践,可以提高代码质量、可维护性和可靠性,同时也可以提高开发效率和支持持续集成和持续交付。
上手单元测试,通常同时从静态测试(Static Test)开始,因为它简单,好理解,静态测试(Static Test)是指在编写测试用例时,我们提前定义好所有的测试方法和测试数据。这些测试方法和数据在编译时就已经确定,不会在运行时发生变化。Junit 中的静态测试通常的常规注解,如 @Test、@Before、@After 等。先来看看一组简单的静态测试示例。
首先,确保你的 pom.xml 文件包含 JUnit 的依赖:
<dependencies>
<dependency>
<groupId>org.junit.jupiter</groupId>
<artifactId>junit-jupiter-api</artifactId>
<version>5.8.0</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.junit.jupiter</groupId>
<artifactId>junit-jupiter-engine</artifactId>
<version>5.8.0</version>
<scope>test</scope>
</dependency>
</dependencies>
然后,创建一个简单的计算器类,通常这里替换为你实际要测试的业务类:
public class SimpleCalculator {
public int add(int a, int b) {
return a + b;
}
public int subtract(int a, int b) {
return a - b;
}
}
然后在 /test 的相同目录下创建对应的测试类
import org.junit.jupiter.api.*;
import static org.junit.jupiter.api.Assertions.assertEquals;
public class SimpleCalculatorTest {
// 在所有测试方法执行前,仅执行一次。这个方法需要是静态的。
@BeforeAll
static void setup() {
System.out.println("BeforeAll - 初始化共享资源,例如数据库连接");
}
// 在所有测试方法执行后,仅执行一次。这个方法需要是静态的。
@AfterAll
static void tearDown() {
System.out.println("AfterAll - 清理共享资源,例如关闭数据库连接");
}
// 在每个测试方法执行前,都会执行一次。用于设置测试方法所需的初始状态。
@BeforeEach
void init() {
System.out.println("BeforeEach - 初始化测试实例所需的数据");
}
// 在每个测试方法执行后,都会执行一次。用于清理测试方法使用的资源。
@AfterEach
void cleanup() {
System.out.println("AfterEach - 清理测试实例所用到的资源");
}
// 标注一个测试方法,用于测试某个功能。
@Test
void testAddition() {
System.out.println("Test - 测试加法功能");
SimpleCalculator calculator = new SimpleCalculator();
assertEquals(5, calculator.add(2, 3), "2 + 3 应该等于 5");
}
// 再添加一个测试方法
@Test
void testSubtraction() {
System.out.println("Test - 测试减法功能");
SimpleCalculator calculator = new SimpleCalculator();
assertEquals(1, calculator.subtract(3, 2), "3 - 2 应该等于 1");
}
}
以上程序,可以看到 Junit 常用注解使用说明:
如果是 maven 项目,可以在目录下执行命令执行测试:
mvn test
输出结果:
[INFO] -------------------------------------------------------
[INFO] T E S T S
[INFO] -------------------------------------------------------
[INFO] Running SimpleCalculatorTest
BeforeAll - 初始化共享资源,例如数据库连接
BeforeEach - 初始化测试实例所需的数据
Test - 测试加法功能
AfterEach - 清理测试实例所用到的资源
BeforeEach - 初始化测试实例所需的数据
Test - 测试减法功能
AfterEach - 清理测试实例所用到的资源
AfterAll - 清理共享资源,例如关闭数据库连接
[INFO] Tests run: 2, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.058 s - in SimpleCalculatorTest
[INFO]
[INFO] Results:
[INFO]
[INFO] Tests run: 2, Failures: 0, Errors: 0, Skipped: 0
[INFO]
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
或者可以直接在 IDEA 中执行测试,如下:

以上就是静态测试的简单示例
动态测试(Dynamic Test):动态测试是指在编写测试用例时,我们可以在运行时生成测试方法和测试数据。这些测试方法和数据在编译时不确定,而是在运行时根据特定条件或数据源动态生成。因为在静态单元测试中,由于测试样本数据有限,通常很难覆盖所有情况,覆盖率到了临界值就很难提高。JUnit 5 中引入动态测试,相比静态测试更复杂,当然也更灵活,也更适合复杂的场景。接下来通过一个简单的示例来展示动态测试和静态测试的区别,我们创建 MyStringUtil 类,它有一个方法 reverse() 用于反转字符串,如下:
public class MyStringUtil {
public String reverse(String input) {
if (input == null) {
return null;
}
return new StringBuilder(input).reverse().toString();
}
}
在静态测试类中,我们使用 @Test 定义 3 个方法来尝试覆盖 reverse() 可能得多种情况:
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.assertEquals;
public class MyStringUtilStaticTest {
private MyStringUtil stringUtil = new MyStringUtil();
@Test
void reverseString() {
// 反转字符串 'hello'
assertEquals("olleh", stringUtil.reverse("hello"));
}
@Test
void reverseEmptyString() {
// 反转空字符串
assertEquals("", stringUtil.reverse(""));
}
@Test
void handleNullString() {
// 处理 null 字符串
assertEquals(null, stringUtil.reverse(null));
}
}
然后用动态测试来实现同样的测试用例:
import org.junit.jupiter.api.DynamicTest;
import org.junit.jupiter.api.TestFactory;
import java.util.Arrays;
import java.util.Collection;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.DynamicTest.dynamicTest;
public class MyStringUtilDynamicTest {
private MyStringUtil stringUtil = new MyStringUtil();
// 使用 @TestFactory 注解定义了一个动态测试工厂方法 reverseStringDynamicTests()
// 工厂方法返回一个 Collection<DynamicTest>
@TestFactory
Collection<DynamicTest> reverseStringDynamicTests() {
// 包含了 3 个动态测试用例,每个测试用例使用 dynamicTest() 方法创建
return Arrays.asList(
dynamicTest("动态测试:反转字符串 'hello'", () -> assertEquals("olleh", stringUtil.reverse("hello"))),
dynamicTest("动态测试:反转空字符串", () -> assertEquals("", stringUtil.reverse(""))),
dynamicTest("动态测试:处理 null 字符串", () -> assertEquals(null, stringUtil.reverse(null)))
);
}
}
在动态测试类中逻辑如下:
@TestFactory 注解定义了一个动态测试工厂方法 reverseStringDynamicTests()。Collection<DynamicTest>,其中包含了 3 个动态测试用例。dynamicTest() 方法创建。以上就是基本的单元测试使用方法,关于 Junit 5 的具体使用并不打算在这里详解,有兴趣可以去参考 Junit 5 的官方文档
编写单元测试需要尽可能的遵循 契约式设计 (Design By Contract, DbC) 代码风格,关于契约式设计可以参考以下的描述:
契约式设计 (Design By Contract, DbC) 是一种软件开发方法,它强调在软件开发中对于每个模块或者函数,应该明确定义其输入和输出的约定(契约)。这些契约可以包括前置条件(preconditions)和后置条件(postconditions),以及可能发生的异常情况。在代码实现时,必须满足这些约定,否则就会引发错误或者异常。
这样说可能比较抽象,可以通过以下的示例代码来理解,如何使用断言来实现契约式设计:
public class BankAccount {
private double balance;
public BankAccount(double balance) {
this.balance = balance;
}
public void withdraw(double amount) {
assert amount > 0 : "Amount must be positive";
assert amount <= balance : "Insufficient balance";
balance -= amount;
assert balance >= 0 : "Balance can't be negative";
}
public double getBalance() {
return balance;
}
}
在这个示例中,我们使用了 Java 中的断言(assertion)来实现契约式设计。具体来说:
assert amount > 0 : "Amount must be positive"; 表示取款金额 amount 必须大于 0assert amount <= balance : "Insufficient balance"; 表示取款金额 amount 必须小于等于账户余额 balanceassert balance >= 0 : "Balance can't be negative"; 表示取款完成后,账户余额 balance 的值应该为非负数可以通过使用 JVM 的 -ea 参数来开启断言功能,不过因为启用 Java 本地断言很麻烦,Guava 团队添加一个始终启用的用来替换断言的 Verify 类。他们建议静态导入 Verify 方法。用法和断言差不多,这里就不过多赘述了。
测试驱动开发(TDD)是一种软件开发方法,也是我个人非常推崇的一种软件开发方法,就是在编写代码之前编写单元测试。TDD 的核心思想是在编写代码之前,先编写测试用例。开发人员在编写代码前先思考预期结果,以便能够编写测试用例。接着开发人员编写足够简单的代码来通过测试用例,再对代码进行重构以提高质量和可维护性。
如图:
作为 TDD 的长期实践者,我总结 TDD 能带来的好处如下:
充足的日志可以帮助开发人员更好地了解程序的运行情况。通过查看日志,可以了解程序中发生了什么事情,以及在哪里发生了问题。这可以帮助开发人员更快地找到和解决问题,从而提高程序的稳定性和可靠性。此外,日志还可以用于跟踪程序的性能和行为,以便进行优化和改进。
通过以下是打印简单日志的示例:
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
<version>1.7.30</version>
</dependency>
<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>1.2.3</version>
</dependency>
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class HelloWorld {
private static final Logger logger = LoggerFactory.getLogger(HelloWorld.class);
public static void main(String[] args) {
logger.info("Hello World");
}
}
这将使用SLF4J打印一条信息,其中包含“Hello World”字符串。你可以在控制台或日志文件中查看此信息。
主要是为了帮助开发人员更好地控制和管理日志输出。SLF4J 定义了多个日志级别:
| 日志级别 | 内容 |
|---|---|
| TRACE | 用于跟踪程序的细节信息,通常用于调试。 |
| DEBUG | 用于调试程序,输出程序中的详细信息,例如变量的值、方法的调用等。 |
| INFO | 用于输出程序的运行状态信息,例如程序启动、关闭、连接数据库等。 |
| WARN | 用于输出警告信息,表示程序可能存在潜在的问题,但不会影响程序的正常运行。 |
| ERROR | 用于输出错误信息,表示程序发生了错误,包括致命错误。 |
不同的日志级别用于记录不同的信息。这样做的目的不仅可以减少不必要的日志输出和文件大小,还可以提供快速定位的能力,例如开发环境通常使用 TRACE、DEBUG 日志,生产环境通常使用 INFO,WARN 日志等。这些信息都可以在 logback.xml 日志配置文件里面配置。
以下是一个基本的 logback 配置文件示例,该配置文件将日志输出到控制台和文件中:
<configuration>
<appender name="CONSOLE" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
</encoder>
</appender>
<appender name="FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
<file>/var/log/myapp.log</file>
<rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
<fileNamePattern>/var/log/myapp.%d{yyyy-MM-dd}.log</fileNamePattern>
<maxHistory>7</maxHistory>
</rollingPolicy>
<encoder>
<pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
</encoder>
</appender>
<root level="INFO">
<appender-ref ref="CONSOLE" />
<appender-ref ref="FILE" />
</root>
</configuration>
在此配置文件中,定义了两个 appender:
控制台的日志格式使用了 pattern 格式化方式,而文件的日志使用了 RollingFileAppender 实现每日轮换,并定义了最多保存 7 天的日志历史。同时,定义了一个根(root)级别为 INFO 的 logger,它会将日志输出到 CONSOLE 和 FILE 两个 appender 中,其他日志级别(TRACE、DEBUG、WARN、ERROR)则按照默认配置输出到根 logger 中。
在 Java 静态扫描工具可以帮助开发人员在开发过程中及时发现和修复代码中的问题和错误,从而提高代码质量和安全性。这些静态扫描工具还可以约束代码风格,在团队协助开发中,统一的风格,可以增强团队协作和沟通,可以增加代码的可读性,可维护性,还减少不必要的讨论和争议,有利于后续的 CodeReview 进展。下面是一些常用的 Java 静态扫描工具:
| 工具名称 | Github 地址 |
|---|---|
| FindBugs | https://github.com/findbugsproject/findbugs |
| PMD | https://github.com/pmd/pmd |
| Checkstyle | https://github.com/checkstyle/checkstyle |
| SonarQube | https://github.com/SonarSource/sonarqube |
| IntelliJ IDEA | https://github.com/JetBrains/intellij-community/ |
访问它们的 Github 地址也提供了更多的信息和支持,可以帮助开发人员更好地理解和使用这些工具。另外,建议在开发过程中,将这些工具集成到持续集成和持续交付的流程中,以便自动化地进行代码检查和修复。
人工的 CodeReview 通常是开发流程的最后一步,为什么前面做了那么多测试和检查工具,到最后还需要人工检查呢 ?
因为静态扫描工具通常只能检查一些简单的问题和错误,相比人工检查它存在以下局限性:
相比机器扫描,人工 Code Review 可以提供以下不可替代的优势:
综上所述,虽然静态扫描工具可以帮助开发人员自动化地发现代码中的问题和错误,但 Code Review 仍然是一种必要的软件开发实践,可以提高代码的质量、可读性和可维护性,同时也可以促进团队协作和学习。因此,建议在开发过程中,将人工 Code Review 和静态扫描工具结合起来,以便更全面和深入地审核和审查代码。
在现代软件开发中,单元测试、TDD、日志、静态检查扫描和人工 Code Review 都是必要的实践,可以帮助开发人员确保软件质量、提高代码可读性和可维护性,并促进团队协作和学习。
首先,单元测试是一种测试方法,用于测试代码的基本单元,例如函数、方法等。单元测试可以帮助开发人员及早发现和解决代码中的问题和错误,从而提高代码质量和可靠性。同时,单元测试还可以提高代码的可读性和可维护性,使代码更易于理解和修改。
其次,TDD(Test-Driven Development,测试驱动开发)是一种开发方法,要求在编写代码之前先编写测试用例。通过使用 TDD,开发人员可以更好地理解代码需求和规范,避免代码中的错误和问题,并提高代码的可读性和可维护性。
第三,日志是一种记录程序运行时状态和信息的方法。日志可以帮助开发人员调试程序,发现潜在的错误和问题,并提供更好的错误处理和处理方案。同时,日志还可以记录程序运行时的性能和状态,从而帮助开发人员分析和优化程序性能。
第四,静态检查扫描工具是一种自动化的代码审核和审查工具,可以帮助开发人员及早发现和解决代码中的问题和错误。通过使用静态检查扫描工具,开发人员可以更全面地检查代码中的问题和错误,并提高代码质量和可读性。
最后,人工 Code Review 是一种手动审核和审查代码的方法,可以更深入地检查代码中的问题和错误,并提供更好的解决方案和建议。人工 Code Review 可以促进团队协作和学习,提高代码质量和可读性,同时还可以遵循特定的编码规范和标准。
综上所述,单元测试、TDD、日志、静态检查扫描和人工 Code Review 都是必要的软件开发实践,可以提高代码质量、可读性和可维护性,并促进团队协作和学习。在进行软件开发时,应该尽可能地遵循这些实践,并使用相应的工具和技术进行代码审核和测试。
很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm
我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI
这似乎非常适得其反,因为太多的gem会在window上破裂。我一直在处理很多mysql和ruby-mysqlgem问题(gem本身发生段错误,一个名为UnixSocket的类显然在Windows机器上不能正常工作,等等)。我只是在浪费时间吗?我应该转向不同的脚本语言吗? 最佳答案 我在Windows上使用Ruby的经验很少,但是当我开始使用Ruby时,我是在Windows上,我的总体印象是它不是Windows原生系统。因此,在主要使用Windows多年之后,开始使用Ruby促使我切换回原来的系统Unix,这次是Linux。Rub
我正在玩HTML5视频并且在ERB中有以下片段:mp4视频从在我的开发环境中运行的服务器很好地流式传输到chrome。然而firefox显示带有海报图像的视频播放器,但带有一个大X。问题似乎是mongrel不确定ogv扩展的mime类型,并且只返回text/plain,如curl所示:$curl-Ihttp://0.0.0.0:3000/pr6.ogvHTTP/1.1200OKConnection:closeDate:Mon,19Apr201012:33:50GMTLast-Modified:Sun,18Apr201012:46:07GMTContent-Type:text/plain
无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD
前言作为一名程序员,自己的本质工作就是做程序开发,那么程序开发的时候最直接的体现就是代码,检验一个程序员技术水平的一个核心环节就是开发时候的代码能力。众所周知,程序开发的水平提升是一个循序渐进的过程,每一位程序员都是从“菜鸟”变成“大神”的,所以程序员在程序开发过程中的代码能力也是根据平时开发中的业务实践来积累和提升的。提高代码能力核心要素程序员要想提高自身代码能力,尤其是新晋程序员的代码能力有很大的提升空间的时候,需要针对性的去提高自己的代码能力。提高代码能力其实有几个比较关键的点,只要把握住这些方面,就能很好的、快速的提高自己的一部分代码能力。1、多去阅读开源项目,如有机会可以亲自参与开源
导读:随着叮咚买菜业务的发展,不同的业务场景对数据分析提出了不同的需求,他们希望引入一款实时OLAP数据库,构建一个灵活的多维实时查询和分析的平台,统一数据的接入和查询方案,解决各业务线对数据高效实时查询和精细化运营的需求。经过调研选型,最终引入ApacheDoris作为最终的OLAP分析引擎,Doris作为核心的OLAP引擎支持复杂地分析操作、提供多维的数据视图,在叮咚买菜数十个业务场景中广泛应用。作者|叮咚买菜资深数据工程师韩青叮咚买菜创立于2017年5月,是一家专注美好食物的创业公司。叮咚买菜专注吃的事业,为满足更多人“想吃什么”而努力,通过美好食材的供应、美好滋味的开发以及美食品牌的孵
在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList()Obt