See real examples of how you can use AI in your testing. Right now. Learn More >>
JUnit Tutorial: Setting Up, Writing, and Running Java Unit Tests
Understand the basics and scale your unit testing practice like a pro with this tutorial.
 
					
				Understand the basics and scale your unit testing practice like a pro with this tutorial.
Do you want to skip the basics and see how to automate unit test generation enhanced with AI to go from 0 to 60%+ code coverage in <5 minutes? Check out Jtest >>
Let’s first talk a little bit about what unit testing is and why it matters in general.
Unit testing is a form of white box testing in which test cases are based on internal structure. The tester chooses inputs to exercise particular paths through the code and configures assertions that validate the output. The purpose of unit testing is to examine the individual components or pieces of methods/classes to verify functionality, ensuring the behavior is as expected.
The exact scope of a "unit" is often left to interpretation, but a nice rule of thumb is to define a unit as the least amount of code needed to perform a standalone task, for example, a single method or class.
There’s a good reason why we limit scope when unit testing: if we construct a test that validates a broad portion of a project, then we shift focus from the functionality of a single method to interactions between different portions of the code. If the test fails, we don’t know why it failed. We’re left wondering whether the point of failure was within the method we were interested in or within the dependencies associated with that method.
JUnit Basics: Getting Started With JUnit and Using Automation to Create Unit Tests
JUnit is the most popular Java unit testing framework. An open-source framework, it’s used to write and run repeatable automated tests.
As with anything else, the JUnit testing framework has evolved over time. JUnit 4.0 was released in 2006, and 5.0 was released in 2017. At the time of this writing, the latest version is 5.12.1. JUnit 5.x has addressed many of the earlier limitations of JUnit, and it has become the most robust Java unit testing framework.
This blog post will cover the usage of both JUnit 4 and 5.
So let’s dive right in. Here are the steps to set up JUnit.
The more common IDEs, such as Eclipse and IntelliJ, will already have JUnit testing integration installed by default. If you’re not using an IDE and perhaps relying solely on a build system such as Maven or Gradle, the installation of JUnit 4/5 is handled via the pom.xml or build.gradle file, respectively.
It’s important to note that JUnit 5 was split into 3 modules, one of those being a vintage module that supports annotation/syntax of JUnit 4 and 3. JUnit 3 usage is very low at this point and is usually seen only in much older projects.
Because of the modular fashion of JUnit 5, a Bill of Materials POM is used to import all JUnit modules and dependencies. If only specific modules are needed, individual groups or artifacts can be specified instead.
To add JUnit 5 to Maven, add the following to pom.xml:
<dependency>
<groupId>org.junit</groupId>
<artifactId>junit-bom</artifactId>
<version>5.12.1</version>
<scope>test</scope>
</dependency>
For Gradle, add the following to the build.gradle file:
apply plugin: 'java'
dependencies {
implementation 'org.junit:junit-bom:5.12.1'
}
To add JUnit 4 to Maven, add the following to pom.xml.
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.13.2</version>
<scope>test</scope>
</dependency>
For Gradle, add the following to the build.gradle:
apply plugin: 'java'
dependencies {
testCompile 'junit:junit:4.13.2'
}
If you need to manually add JUnit to the classpath for JUnit testing, you need to reference the raw jar file(s) directly, although this isn’t usually required. Both JUnit 4 and 5 have the jar available to download directly. For JUnit 5, you’ll need to download a fat jar (aka uber jar) as described here.
Improve Unit Testing for Java With Automation: Best Practices for Java Developers
Now that we’ve talked a little about JUnit setup, let’s move on to the actual construction and execution of these tests. To best illustrate the creation of JUnits, we’ll start with something basic. In the JUnit test example below, we have a simple method (left) that converts Fahrenheit to Celsius and a JUnit test (right) written to test our method. I’ve numbered the key parts of the JUnit test and will discuss each part in detail below.

These are imports for the JUnit classes and packages used by the unit test. The imports can be specified as individual classes but are commonly specified as entire packages using asterisks. Either way works—the level of granularity of the imports is a matter of preference.
This defines the start of our test class. The important thing to take note of here is the naming convention used for the class, which is ClassNameTest. This naming convention is not required, but it’s the most common way to name JUnit classes because the name succinctly describes the purpose of the unit test class and which class it is testing.
Here, we see our first JUnit-specific syntax: the @Test annotation. Annotations are extremely important when creating JUnits. This is how the JUnit framework identifies the important parts of the unit test. In our example, the @Test annotation tells JUnit that the public void method to which it is attached can be run as a test case.
There are many other annotations, but some of the most common are the following.
Again, note the naming convention testMethodName, where methodName is the name of the method being tested in the class under test.
In the Given section of the test, we construct a new instance of the class under test and initialize it as appropriate. This is necessary since the test method needs to call the method under test to test it. In our example, no other initialization is needed beyond instantiating the class, but in many cases, additional setup may need to happen, such as initializing objects to pass into the constructor or calling methods that configure the state of the class under test.
The When section of the test includes initializing variables that need to be passed when calling the method being tested and then calling the test method (part 8). The variables should be given meaningful values that cause the test to exercise the parts of the test method that we care about. Note that if a variable is an object, it can be instantiated or mocked.
If the method under test returns a value, it should be captured in a variable so that its value can be asserted on.
Unit tests are only valuable if they include assertions that validate that the method being tested returns the right value and/or adjusts the state of other objects as expected. Without assertions, you have no verification, and your test is at best a smoke test that gives feedback only when an exception is thrown.
The JUnit assertion methods, which are included in the org.junit.jupiter.api.Assertions class in JUnit 5 and the org.junit.Assert class in JUnit 4, are commonly used to determine the pass/fail status of test cases. Only failed assertions are reported by the JUnit framework. Like with annotations, there are many assertion options.
In our example JUnit above, we use the assertEquals (expected, actual, delta) method. The first argument is:
Choose your own adventure! Here, we will look at three ways to run JUnits: straight from the command line, from the IDE (Eclipse and IntelliJ), and using build systems (Maven and Gradle).
To run a JUnit directly from the command line, you need a few things:
The command is as follows. This example is for JUnit 4.
java -cp /path/to/junit.jar org.junit.runner.JUnitCore <test class name>
Note: Running a test from the command line is most commonly done from a CI/CD process running in a build system like Jenkins or Azure DevOps.
Within the Package Explorer, locate your JUnit test. Right-click and select Run As > JUnit Test. This will execute your test and report results within the JUnit Eclipse view.

Running a test in IntelliJ is similar to Eclipse. From the Project window, locate your test, right-click and select Run ‘testName’. Like Eclipse, a JUnit window will open with the results of the test.

Maven has made running unit tests simple. Ensure you are in the proper location from your command line, and the project pom.xml is properly configured. Then you can run the following to execute your JUnits.
To run the entire test suite:
mvn test
To run a single/specific test(s):
mvn -Dtest=TestName test
Gradle, like maven, has made running tests simple.
To run the entire test suite:
 gradlew test
To run a single/specific test(s):
 gradlew -Dtest.single=testName test
Note: Maven and Gradle are their own beasts. What is shown here is minimal to cover the basics. Check out their documentation if you want to learn more.
Our example ran through a simple unit test, and of course, this is just the start of unit testing. More complex methods that need to be tested may call methods in dependent classes, or connect to external systems like a database. In these kinds of cases, it may be desirable to isolate the code through mocking.
Mocking helps to isolate units of code so that our unit tests can focus only on the specific class/method being tested. The most common framework used for mocking in JUnit tests is Mockito.
To learn more about mocking, read my colleague’s post: How to automate a Java unit test, including mocking and assertions.
If unit testing is so important, why doesn’t everyone do it consistently? Despite its importance, unit testing isn’t always easy to implement or maintain. It requires deep development knowledge and ongoing effort to keep test suites current. As a result, unit testing often gets deprioritized—until regressions happen.
But it doesn’t have to be that way.
This is where Parasoft Jtest comes in. Its AI-powered Unit Test Assistant was designed to remove the friction from writing and maintaining unit tests. With just a few clicks, teams starting with 0% code coverage can automatically generate robust test suites that cover 60% or more of their Java code. As one financial services firm noted, "Since we implemented Parasoft Jtest, we have successfully reduced the amount of time it takes to create and maintain unit tests by more than 50%."
Here’s how your team can scale your testing practice.
If writing and maintaining JUnit tests has felt like a chore, it’s time to modernize your approach. AI-enhanced testing solutions like this can turn test creation from a bottleneck into a strategic advantage—giving you more time to focus on building great software.
Make unit testing easier and faster with AI-enhanced Parasoft Jtest.
Try it yourself with a free, full-access trial for 14 days.