Skip to content

Commit 0752286

Browse files
authoredJul 17, 2020
BAEL-4135 - When are static variables initialized? (eugenp#9607)
* Hexagonal Architecture in Java A quick and practical example of Hexagonal Architecture in Java * Fixed code formatting issues * When are static variables initialized Sample class and test class. * Revert "When are static variables initialized" This reverts commit c781923. * Revert "Revert "When are static variables initialized"" This reverts commit 2bffdf4. * New java module for static variable Created a new core-java-lang-3 module for static variables. * PR review changes: Added more scenarios to the static variable test Covered the following cases: 1. Variable initialization in a static block 2. Variable initialization in a static nested class
1 parent 03c7d92 commit 0752286

File tree

16 files changed

+449
-1
lines changed

16 files changed

+449
-1
lines changed
 

‎core-java-modules/core-java-lang-3/pom.xml

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,4 +39,4 @@
3939
<assertj.version>3.12.2</assertj.version>
4040
</properties>
4141

42-
</project>
42+
</project>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
package com.baeldung.staticvariables;
2+
3+
public class StaticVariableDemo {
4+
public static int i;
5+
public static int j = 20;
6+
public static int z;
7+
8+
static {
9+
z = 30;
10+
a = 40;
11+
}
12+
13+
public static int a = 50;
14+
15+
public static final int b = 100;
16+
17+
public StaticVariableDemo() {
18+
}
19+
20+
static class Nested {
21+
public static String nestedClassStaticVariable = "test";
22+
}
23+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,113 @@
1+
package com.baeldung.staticvariables;
2+
3+
import static org.assertj.core.api.Assertions.assertThat;
4+
5+
import java.lang.reflect.Field;
6+
7+
import org.junit.jupiter.api.Test;
8+
9+
public class StaticVariableUnitTest {
10+
11+
@Test
12+
public void initializeStaticVariable_checkAssignedValues() {
13+
14+
try {
15+
Class<?> staticVariableDemo = this.getClass()
16+
.getClassLoader()
17+
.loadClass("com.baeldung.staticvariables.StaticVariableDemo");
18+
19+
Field field1 = staticVariableDemo.getField("i");
20+
21+
assertThat(field1.getInt(staticVariableDemo)).isEqualTo(0);
22+
23+
Field field2 = staticVariableDemo.getField("j");
24+
25+
assertThat(field2.getInt(staticVariableDemo)).isEqualTo(20);
26+
27+
} catch (ClassNotFoundException | NoSuchFieldException | SecurityException e) {
28+
e.printStackTrace();
29+
} catch (IllegalArgumentException e) {
30+
e.printStackTrace();
31+
} catch (IllegalAccessException e) {
32+
e.printStackTrace();
33+
}
34+
35+
}
36+
37+
@Test
38+
public void initializeStaticVariable_checkStaticBlock() {
39+
40+
try {
41+
Class<?> staticVariableDemo = this.getClass()
42+
.getClassLoader()
43+
.loadClass("com.baeldung.staticvariables.StaticVariableDemo");
44+
45+
Field field1 = staticVariableDemo.getField("z");
46+
47+
assertThat(field1.getInt(staticVariableDemo)).isEqualTo(30);
48+
49+
Field field2 = staticVariableDemo.getField("a");
50+
51+
assertThat(field2.getInt(staticVariableDemo)).isEqualTo(50);
52+
53+
} catch (ClassNotFoundException | NoSuchFieldException | SecurityException e) {
54+
e.printStackTrace();
55+
} catch (IllegalArgumentException e) {
56+
e.printStackTrace();
57+
} catch (IllegalAccessException e) {
58+
e.printStackTrace();
59+
}
60+
61+
}
62+
63+
@Test
64+
public void initializeStaticVariable_checkFinalValues() {
65+
66+
try {
67+
Class<?> staticVariableDemo = this.getClass()
68+
.getClassLoader()
69+
.loadClass("com.baeldung.staticvariables.StaticVariableDemo");
70+
71+
Field field1 = staticVariableDemo.getField("b");
72+
73+
assertThat(field1.getInt(staticVariableDemo)).isEqualTo(100);
74+
75+
} catch (ClassNotFoundException | NoSuchFieldException | SecurityException e) {
76+
e.printStackTrace();
77+
} catch (IllegalArgumentException e) {
78+
e.printStackTrace();
79+
} catch (IllegalAccessException e) {
80+
e.printStackTrace();
81+
}
82+
83+
}
84+
85+
@Test
86+
public void initializeStaticVariable_checkInnerClassValues() {
87+
88+
try {
89+
Class<?> staticVariableDemo = this.getClass()
90+
.getClassLoader()
91+
.loadClass("com.baeldung.staticvariables.StaticVariableDemo");
92+
93+
Class<?>[] nestedClasses = staticVariableDemo.getClasses();
94+
95+
for (Class<?> nestedClass : nestedClasses) {
96+
if (nestedClass.getName()
97+
.equals("Nested")) {
98+
99+
Field field1 = nestedClass.getField("nestedClassStaticVariable");
100+
assertThat(field1.get(nestedClass)).isEqualTo("test");
101+
}
102+
}
103+
104+
} catch (ClassNotFoundException | NoSuchFieldException | SecurityException e) {
105+
e.printStackTrace();
106+
} catch (IllegalArgumentException e) {
107+
e.printStackTrace();
108+
} catch (IllegalAccessException e) {
109+
e.printStackTrace();
110+
}
111+
112+
}
113+
}
+59
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
<?xml version="1.0" encoding="UTF-8"?>
2+
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
3+
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
4+
<modelVersion>4.0.0</modelVersion>
5+
<parent>
6+
<groupId>org.springframework.boot</groupId>
7+
<artifactId>spring-boot-starter-parent</artifactId>
8+
<version>2.3.0.RELEASE</version>
9+
<relativePath/> <!-- lookup parent from repository -->
10+
</parent>
11+
<groupId>com.baeldung</groupId>
12+
<artifactId>hexagonal-architecture</artifactId>
13+
<version>1.0</version>
14+
<name>hexagonal-architecture</name>
15+
<description>Project for hexagonal architecture in java</description>
16+
17+
<properties>
18+
<java.version>1.8</java.version>
19+
</properties>
20+
21+
<dependencies>
22+
<dependency>
23+
<groupId>org.springframework.boot</groupId>
24+
<artifactId>spring-boot-starter-web</artifactId>
25+
</dependency>
26+
27+
<dependency>
28+
<groupId>org.springframework.boot</groupId>
29+
<artifactId>spring-boot-starter-data-mongodb</artifactId>
30+
</dependency>
31+
32+
<dependency>
33+
<groupId>org.springframework.boot</groupId>
34+
<artifactId>spring-boot-starter-test</artifactId>
35+
<scope>test</scope>
36+
<exclusions>
37+
<exclusion>
38+
<groupId>org.junit.vintage</groupId>
39+
<artifactId>junit-vintage-engine</artifactId>
40+
</exclusion>
41+
</exclusions>
42+
</dependency>
43+
<dependency>
44+
<groupId>org.mockito</groupId>
45+
<artifactId>mockito-core</artifactId>
46+
<scope>test</scope>
47+
</dependency>
48+
</dependencies>
49+
50+
<build>
51+
<plugins>
52+
<plugin>
53+
<groupId>org.springframework.boot</groupId>
54+
<artifactId>spring-boot-maven-plugin</artifactId>
55+
</plugin>
56+
</plugins>
57+
</build>
58+
59+
</project>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
package com.baeldung.pattern.hexagonal;
2+
3+
import org.springframework.boot.SpringApplication;
4+
import org.springframework.boot.autoconfigure.SpringBootApplication;
5+
6+
@SpringBootApplication
7+
public class HexArchApplicationDemo {
8+
9+
public static void main(String[] args) {
10+
SpringApplication.run(HexArchApplicationDemo.class, args);
11+
}
12+
13+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
package com.baeldung.pattern.hexagonal.config;
2+
3+
import com.baeldung.pattern.hexagonal.domain.services.EmployeeService;
4+
import com.baeldung.pattern.hexagonal.domain.services.EmployeeServiceImpl;
5+
import com.baeldung.pattern.hexagonal.persistence.EmployeeRepository;
6+
import org.springframework.context.annotation.Bean;
7+
import org.springframework.context.annotation.Configuration;
8+
9+
@Configuration
10+
public class AppConfig {
11+
@Bean
12+
public EmployeeService getEmployeeService(EmployeeRepository employeeRepository) {
13+
return new EmployeeServiceImpl(employeeRepository);
14+
}
15+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
package com.baeldung.pattern.hexagonal.config;
2+
3+
import com.baeldung.pattern.hexagonal.persistence.MongoRepoEx;
4+
import org.springframework.context.annotation.Configuration;
5+
import org.springframework.data.mongodb.repository.config.EnableMongoRepositories;
6+
7+
@Configuration
8+
@EnableMongoRepositories(basePackageClasses = MongoRepoEx.class)
9+
public class MongoConfig {
10+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
package com.baeldung.pattern.hexagonal.controller;
2+
3+
import com.baeldung.pattern.hexagonal.domain.model.Employee;
4+
import com.baeldung.pattern.hexagonal.domain.services.EmployeeService;
5+
import org.springframework.beans.factory.annotation.Autowired;
6+
import org.springframework.http.MediaType;
7+
import org.springframework.web.bind.annotation.*;
8+
9+
@RestController
10+
@RequestMapping("/employees")
11+
public class EmployeeController {
12+
@Autowired
13+
EmployeeService employeeService;
14+
15+
@PostMapping(produces = MediaType.APPLICATION_JSON_VALUE, consumes = MediaType.APPLICATION_JSON_VALUE)
16+
@ResponseBody
17+
public Employee addEmployee(@RequestBody Employee employee) {
18+
return employeeService.addEmployee(employee);
19+
}
20+
21+
@GetMapping(path = "/{employeeId}")
22+
public Employee getEmployee(@PathVariable("employeeId") String employeeId) {
23+
return employeeService.getEmployee(employeeId);
24+
}
25+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
package com.baeldung.pattern.hexagonal.domain.model;
2+
3+
import org.springframework.data.annotation.Id;
4+
5+
import java.util.Objects;
6+
7+
public class Employee {
8+
@Id
9+
private String empId;
10+
private String empName;
11+
private String empJobTitle;
12+
13+
public String getEmpId() {
14+
return empId;
15+
}
16+
17+
public void setEmpId(String empId) {
18+
this.empId = empId;
19+
}
20+
21+
public String getEmpName() {
22+
return empName;
23+
}
24+
25+
public void setEmpName(String empName) {
26+
this.empName = empName;
27+
}
28+
29+
public String getEmpJobTitle() {
30+
return empJobTitle;
31+
}
32+
33+
public void setEmpJobTitle(String empJobTitle) {
34+
this.empJobTitle = empJobTitle;
35+
}
36+
37+
@Override
38+
public boolean equals(Object o) {
39+
if (this == o)
40+
return true;
41+
if (o == null || getClass() != o.getClass())
42+
return false;
43+
Employee employee = (Employee) o;
44+
return empId.equals(employee.empId);
45+
}
46+
47+
@Override
48+
public int hashCode() {
49+
return Objects.hash(empId);
50+
}
51+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
package com.baeldung.pattern.hexagonal.domain.services;
2+
3+
import com.baeldung.pattern.hexagonal.domain.model.Employee;
4+
5+
public interface EmployeeService {
6+
7+
Employee addEmployee(Employee employee);
8+
9+
Employee getEmployee(String employeeId);
10+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
package com.baeldung.pattern.hexagonal.domain.services;
2+
3+
import com.baeldung.pattern.hexagonal.domain.model.Employee;
4+
import com.baeldung.pattern.hexagonal.persistence.EmployeeRepository;
5+
import org.springframework.beans.factory.annotation.Autowired;
6+
7+
import java.util.Optional;
8+
9+
public class EmployeeServiceImpl implements EmployeeService {
10+
11+
private EmployeeRepository employeeRepository;
12+
13+
@Autowired
14+
public EmployeeServiceImpl(EmployeeRepository employeeRepository) {
15+
this.employeeRepository = employeeRepository;
16+
}
17+
18+
@Override
19+
public Employee addEmployee(Employee employee) {
20+
return employeeRepository.add(employee);
21+
}
22+
23+
@Override
24+
public Employee getEmployee(String employeeId) {
25+
Optional<Employee> employee = employeeRepository.findById(employeeId);
26+
27+
if (employee.isPresent()) {
28+
return employee.get();
29+
} else {
30+
// throw
31+
}
32+
return null;
33+
}
34+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
package com.baeldung.pattern.hexagonal.persistence;
2+
3+
import com.baeldung.pattern.hexagonal.domain.model.Employee;
4+
import org.springframework.stereotype.Repository;
5+
6+
import java.util.Optional;
7+
8+
@Repository
9+
public interface EmployeeRepository {
10+
11+
Employee add(Employee employee);
12+
13+
Optional<Employee> findById(String id);
14+
15+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
package com.baeldung.pattern.hexagonal.persistence;
2+
3+
import com.baeldung.pattern.hexagonal.domain.model.Employee;
4+
import org.springframework.beans.factory.annotation.Autowired;
5+
import org.springframework.stereotype.Repository;
6+
7+
import java.util.Optional;
8+
9+
@Repository
10+
public class MongoDBRepository implements EmployeeRepository {
11+
12+
@Autowired
13+
MongoRepoEx mongoRepository;
14+
15+
@Override
16+
public Employee add(Employee employee) {
17+
return mongoRepository.insert(employee);
18+
}
19+
20+
@Override
21+
public Optional<Employee> findById(String id) {
22+
return mongoRepository.findById(id);
23+
}
24+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
package com.baeldung.pattern.hexagonal.persistence;
2+
3+
import com.baeldung.pattern.hexagonal.domain.model.Employee;
4+
import org.springframework.data.mongodb.repository.MongoRepository;
5+
import org.springframework.stereotype.Repository;
6+
7+
@Repository
8+
public interface MongoRepoEx extends MongoRepository<Employee, String> {
9+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
package com.baeldung.pattern.hexagonal.domain.services;
2+
3+
import com.baeldung.pattern.hexagonal.domain.model.Employee;
4+
import com.baeldung.pattern.hexagonal.persistence.EmployeeRepository;
5+
import org.junit.jupiter.api.BeforeEach;
6+
import org.junit.jupiter.api.Test;
7+
import java.util.Optional;
8+
9+
import static org.junit.jupiter.api.Assertions.*;
10+
import static org.mockito.ArgumentMatchers.any;
11+
import static org.mockito.Mockito.mock;
12+
import static org.mockito.Mockito.when;
13+
14+
class EmployeeServiceImplTest {
15+
16+
private EmployeeRepository employeeRepository;
17+
private EmployeeService testService;
18+
private Employee testModel;
19+
20+
@BeforeEach
21+
void setUp() {
22+
employeeRepository = mock(EmployeeRepository.class);
23+
24+
testService = new EmployeeServiceImpl(employeeRepository);
25+
testModel = new Employee();
26+
testModel.setEmpId("2000");
27+
testModel.setEmpName("Test user 1");
28+
testModel.setEmpJobTitle("Software engineer");
29+
}
30+
31+
@Test
32+
void addEmployee() {
33+
when(employeeRepository.add(any(Employee.class))).thenReturn(testModel);
34+
35+
Employee testResponse = testService.addEmployee(testModel);
36+
assertEquals(testModel, testResponse);
37+
}
38+
39+
@Test
40+
void getEmployee() {
41+
when(employeeRepository.findById("2000")).thenReturn(Optional.of(testModel));
42+
43+
Employee testResponse = testService.getEmployee("2000");
44+
assertEquals(testModel, testResponse);
45+
}
46+
}

0 commit comments

Comments
 (0)
Please sign in to comment.