SpringBoot Reference
Concepts
Loose Coupling
Loose coupling refers to designing a system where components are minimally dependent on each other.
- This allows for easier modification, testing, and maintenance because changes in one component have little to no impact on others.
- By Autowiring, we achieve loose coupling.
By using
new
keyword (instantiating an obejct), we “tightly couple” the dependency which is not good
-
Tightly Coupled:
public class UserService { private final UserRepository userRepository = new UserRepository(); // Directly creating a dependency // Methods using userRepository }
-
Loosely Coupled (using Dependency Injection):
public class UserService { private final UserRepository userRepository; // Constructor Injection public UserService(UserRepository userRepository) { this.userRepository = userRepository; } // Methods using userRepository }
Here, UserService
is loosely coupled with UserRepository
because it doesn’t
create the dependency itself.
Instead, UserRepository
is injected into UserService
, making it easier to
swap out UserRepository
implementations.
Inversion of Control
Instead of the Class taking responsibility of creating the object, the framework manages it for you
- the control of object creation and dependency management is * *inverted** from the application code to a framework.
- The framework manages the lifecycle and interactions of objects.
Dependency Injection
Dependency Injection (DI) is a design pattern used to implement IoC ( Inversion of Control).
- In DI, the framework (like Spring) handles the creation and injection of dependencies or bean instantiation beans and wiring dependencies, rather than the classes managing their own dependencies.
Stereotype Annotations
Basic philosophy of Spring Boot : Conventions over configurations
@Component
- Purpose: Marks a class as a Spring component or bean.
- Commonly Used in: Utility classes, business logic classes, and other non-specialized components.
@Controller
- Purpose: Marks a class as a Spring MVC controller.
- Use Case: Used for classes that handle HTTP requests in a Spring MVC web application.
- Commonly Used in: Classes that define request mappings, handle user input, and return views or data to the client.
@Service
- Purpose: Marks a class as a service or business logic component.
- Use Case: Typically used for classes that contain the business logic of the application.
- Commonly Used in: Service layer classes that encapsulate business rules, data processing, and interactions with repositories or other services.
@Repository
- Purpose: Marks a class as a Spring Data repository.
- Use Case: Used for classes that interact with a database or external data source.
- Commonly Used in: Data access objects (DAOs) that perform CRUD (Create, Read, Update, Delete) operations on entities.
Config
Set a desired Port
server.port=8089
Application Yaml settings
Good Practice : Design application configuration using
@ConfigurationProperties
to ensure Type Safety
myConfig:
flag: true
message: From YAML
number: 100
Type Safety can be ensured with this
@Component
@ConfigurationProperties("myConfig")
public class MyConfiguration {
private boolean flag;
private String message;
private int number;
Banner
For Ascii banner, put the ASCII Art in banner.txt in and it will be taken Sample file
to turn off the banner
spring:
main:
banner-mode: "off"
For image banner, put the logo.png file and
spring:
banner:
image:
location: logo.png
Springboot Startup process
https://nitinkc.github.io/spring/microservices/springboot-startup/
Initial Data Setup
keep the sql script in the resources folder by the name data.sql
Scans
Component Scan
By default, the package containing the main method is scanned.
-
@ComponentScan
is used to specify the packages that Spring should scan to discover Spring-managed components like beans, controllers, services, etc.
@ComponentScan(basePackages = {"com.spring5.concepts",
"com.spring5.services"
"com.test.animals",
"com.flowers"
})
Entity Scan
@EntityScan
is specific to Spring Data JPA.
- It’s used to specify the packages where JPA entities are located.
- This is important because Spring Data JPA needs to know where the entity classes are in order to create repositories and perform CRUD operations.
@EntityScan(basePackages = {"com.learningJPA.dSpringDataRepository"
,"com.learningJPA.eTest.model"
//,"com.learningJPA.hibernate.*"
})
SpringBootApplication ScanBasePackages
@SpringBootApplication
is a meta-annotation that combines several annotations,
including @ComponentScan
.
-
scanBasePackages
within@SpringBootApplication
allows you to specify the base packages to scan for Spring components. - used in the main application class.
- It also scans the default package where the main application class is located.
- exclude argument is used to exclude specific auto-configurations, which means that Spring Boot won’t automatically configure the classes mentioned.
@SpringBootApplication(
scanBasePackages = {
"com.test.package1",
"com.test.service.security"
},
exclude = { JmxAutoConfiguration.class })
@Autowired & Dependency Injection
https://nitinkc.github.io/spring/microservices/dependency-injection-concepts/
Autowiring
- byType
- byName
- constructor - similar to byType, but through constuctor
https://nitinkc.github.io/spring/microservices/dependency-injection-concepts/#Autowiring
Eliminates the need of creating a new object and hence the need of constructors
from the components
StudentService studentService = new StudentService();
@Autowired
StudentService studentService;//Free to use studentService object within the class anywhere
@Autowired
used for automatic dependency injection.
- Spring should find the matching bean and wire the dependency
Dependency injection is a design pattern in which objects are provided with their dependencies (i.e., the objects they need to collaborate with) rather than creating those dependencies themselves.
Automatic Injection: When you annotate a field, setter method, or
constructor with @Autowired
, Spring will
automatically inject the required dependency (another Spring bean) at runtime.
Constructor vs Setter Injection
- Constructor Injection for Mandatory Dependencies
- Setter Injection for Optional Dependencies
Lombok
To use the @RequiredArgsConstructor
annotation. Mark the fields you want to
include in the constructor as final
or annotate them with @NonNull
.
@RequiredArgsConstructor
public class UserService {
private final UserRepository userRepository;
Sequence of execution
Postman/browser/client -> Controller -> Service -> Repository -> Service -> Controller
REST APIs
- Retrieve all Users - GET
/users
- Create a User - POST
/users
- Retrieve one User - GET
/user/{id}
->/user/1
- Delete a User - DELETE
/user/{id}
->/user/1
- Retrieve all posts for a User - GET
/user/{id}/posts
- Create a posts for a User - POST
/user/{id}/posts
- Retrieve details of a post - GET
/user/{id}/posts/{post_id}
Best Practises
- Use Plurals
https://nitinkc.github.io/microservices/Idempotence-HTTP-methods/#designing-restful-uris
Controller Vs RestController
@Controller
on a Controller class needs @ResponseBody
with method name
@Controller
@RequestMapping(method= RequestMethod.GET, path = "/health",
produces = { "application/json", MediaType.APPLICATION_XML_VALUE})
public class HealthCheckController {
@RequestMapping(path = "/check", method = RequestMethod.GET)
public @ResponseBody String hello(){// ResponseBody Annotation is compulsory
return "Health is Ok";
}
}
GET Request
https://nitinkc.github.io/spring/microservices/GET-rest-calls/
@RequestMapping(method = RequestMethod.GET,
path = "/student/{studentId}")
OR
@GetMapping(value = "/student/{studentId}",
path = "/student/{studentId}",
produces = { "application/json", MediaType.APPLICATION_XML_VALUE})
Between value
and path
attribute, value is commonly used to describe the
path
Shortened GetMapping
@GetMapping(path="/getMapping")
GetMapping returning a bean (JSON Response)
@GetMapping(path="/getBean")
public HelloWorldReturnBean helloWorldReturnBean() {
return new HelloWorldReturnBean("Hello World - From HelloWorldReturnBean");
}
Path Variable vs Request Param
Validation
Request Validation
- At Class level, add
@Validated
annotation and at the@Valid
at the parameter level - Check the
required
anddefaultValue
arguments of RequestParam Annotation
@RestController
@RequestMapping("/test")
@Validated
public class ValidationController {
@GetMapping("/email")
public String testEmail(@Valid @Email(message = "Please provide a valid email address")
@RequestParam(value = "email") String email ,
@RequestParam(value = "greet", required = false, defaultValue = "No Val from Request")
String greet,
@RequestParam(value = "count", required = false, defaultValue = "-1") Integer count) {
StringBuilder sb= new StringBuilder();
sb.append(email).append(" email OK").append("\nCount is ").append(count).append("\n").append(greet);
return sb.toString();
}
}
Response Validation
In the DTO Class (Using Lombok)
@Data
public class StudentRequestBody {
private int count;
@JsonProperty("studentIds")//If the name in the request body differs from variable name
private List<String> studentIdList;
@Email(message = "Incorrect EmailID received from DB")
private String emailId;
}
POST Request
@RequestMapping(method = RequestMethod.POST)
//OR
@PostMapping("/students")
With Map as Request Body
if request body is like below, a map can be used
Curl Request
curl --location 'localhost:8090/student/db/studentIdsByMap' \
--header 'Content-Type: application/json' \
--data '{
"values": ["10","12.5","50","100"]
}'
Read the Request Body using @RequestBody
in the method parameter into either a
Map, for simple structures or a class for complex
@PostMapping(path = "/studentIdsByMap",
consumes = {MediaType.APPLICATION_JSON_VALUE},
produces = {"application/json"})
public List<StudentDto> getStudentByIdsByMap(@RequestBody Map<String,List<Integer>> mapStudentIds){
...
}
With Class as Request Body
Curl Request
curl --location 'localhost:8090/student/db/studentIdsByClassName' \
--header 'Content-Type: application/json' \
--data '{
"greeting":"Hi from postman.",
"count":5,
"studentIds": ["1","2","3","4","5"]
}'
Corresponding Java class to catch the request Body
public class StudentRequestBody {
private int count;
@JsonProperty("studentIds")//If the name in the request body differs from variable name
private List<String> studentIdList;
private String greeting;
}
Controller with @RequestBody
@PostMapping(path = "/studentIdsByClassName",
consumes = {MediaType.APPLICATION_JSON_VALUE},
produces = {"application/json"})
public StudentDtoClass getStudentByIdsRequestBody(@RequestBody StudentRequestBody studentRequestBody){
...
}
Service
For a single student Id, JPA’s findById method can be utilized. It returns an Optional, so if in case the return is a null Optional Class findById can be utilized.
Return an Object
Optional<Student> studentById = studentRepository.findById(studentId);//Method from JPA Repo, returns Optional
Student student = studentById.orElseGet(Student::new);//Return empty constructor if no data/Null
The supplier in orElseGet can be written in whichever way feels intuitive.
Student student = studentById.orElseGet(Student::new);//Return empty constructor if no data/Null
//student = studentById.orElseGet(() -> new Student());
//student = studentById.orElseGet(() -> Student.builder().build());
Simple Mapper
If the class structure of DAO Class is different from the DTO Class, then separate mappers or convertors can be written.
StudentDto studentDto = studentMapper.convert(student);//Convertor/Mapper/Transformer
The convert method takes in a DAO Object and returns a DTO object
@Component
public class StudentMapper {
public StudentDto convert(Student studentById) {
return StudentDto.builder()
.fullName(studentById.getFirstName() + " " + studentById.getLastName())
.city(studentById.getCityOfBirth())
.sex(studentById.getGender())
.university(studentById.getUniversity())
.emailId(studentById.getEmail())//TODO: The email validation.
.build();
}
}
Over all the service class with method to return a single student object
@Service
public class StudentServiceWithDb {
@Autowired StudentRepository studentRepository;
@Autowired StudentMapper studentMapper;
public StudentDto getStudentById(int studentId) {
Optional<Student> studentById = studentRepository.findById(studentId);//Method from JPA Repo, returns Optional
Student student = studentById.orElseGet(Student::new);//Return empty constructor if no data/Null
//student = studentById.orElseGet(() -> new Student());
//student = studentById.orElseGet(() -> Student.builder().build());
StudentDto studentDto = studentMapper.convert(student);//Convertor/Mapper/Transformer
return studentDto;
}
}
Return a List of Object
The method that returns a List of objects, based on the multiple student id’s
passed can be written using the
findAllByIds
method of JpaRepository Interface.
List<Student> studentDetailsList = studentRepository.findAllById(studentIdList);
In order to convert the list of DAO objects to a list of DTO objects, the intuition could be of for loop
//Intuitive way
List<StudentDto> studentDtoList = new ArrayList<>();//Initialize the return array
for(Student s:studentDetailsList){
StudentDto singleStudentDto = studentMapper.convert(s);
studentDtoList.add(singleStudentDto);
}
But a better way of achieving this is the use of functional style of programming
//Java 8
List<StudentDto> studentDtoList = studentDetailsList.stream()
//.filter(Objects::nonNull)//Remove any null rows if needed
.map(studentMapper::convert)
.collect(Collectors.toList());
Finally, the overall method would be
public List<StudentDto> getStudentByIds(List<Integer> studentIdList) {
List<Student> studentDetailsList = studentRepository.findAllById(studentIdList);
List<StudentDto> studentDtoList = studentDetailsList.stream()
.map(studentMapper::convert)
.collect(Collectors.toList());
return studentDtoList;
}
Mapping for DTO
Jackson Mapper
Map Struct
The simple one is Jackson mapper, with lot of control
@JsonProperty("studentIds")//If the name in the request body differs from variable name
private List<String> studentIdList;
Control other aspects of the DTO
- if empty values in the array is not needed
@JsonInclude(JsonInclude.Include.NON_NULL)
@Data
public class Filters {
@JsonProperty("treatmentDay")
public String treatmentDay;
}
Repository
Spring Data JPA is an implementation of Java Persistence API
Custom Exceptions
Use @ControllerAdvice
or @RestControllerAdvice
for Global exception
handling.
Difference between @ControllerAdvice
& @RestControllerAdvice
The primary difference is in the type of responses they handle.
- @RestControllerAdvice is geared toward RESTful services, where responses are typically data-centric (e.g., JSON or XML),
- while @ControllerAdvice is used in traditional web applications, where responses often include both views and data.
HikariCP
HikariCP, often referred to simply as Hikari, is a popular and high-performance connection pool library for Java applications.
Connection pooling is a technique used to efficiently manage and reuse database connections in applications that interact with a relational
spring.jpa.properties.hibernate.default_schema="chinookMusic"
spring.datasource.url = jdbc:postgresql://localhost:5432/mydb
# or combined
spring.datasource.url=jdbc:postgresql://postgres:5432/mydb?currentSchema=test
Read Environment properties
Use the Environment dependency
@Autowired
private Environment environment;
int portNumber = Integer.parseInt(environment.getProperty("server.port"));
Another method by @Value Annotation
@Value("${custom.value}")
private String customVal;
Rest Template
Refer the following page for details Rest Template
CommandLineRunner
https://nitinkc.github.io/spring/microservices/CommandLineRunner/
@Component
@Slf4j
@RequiredArgsConstructor
@Order(value = 1)
@ConditionalOnExpression("${dateExpiration:false}")
public class RutWhileBooting implements CommandLineRunner {
private final LicenseService licenseService;
@Override
public void run(String... args) throws Exception {
log.info("Starting Runner : ExpiryDate");
//Do processing
licenseService.runJob();
}
}
Scheduling a Job
https://nitinkc.github.io/spring/microservices/spring-scheduler/
use @EnableScheduling
on the application main class
@Component
@AllArgsConstructor
public class DailyTaskScheduler {
private final MyService myService;
@Scheduled(cron = "0 0 0 * * *") // Executes at midnight every day
//@Scheduled(fixedRate = 5000) // Executes every 5 seconds (5000 milliseconds) for testing
public void runDailyTask() {
myService.runJob();
}
Spring Boot Actuator
Monitoring
- /env, /metrics, /trace, /dump
- /beans, / autoconfig, /configprops, /mappings
Metric logging - Prometheus and micrometer
Design Patterns in Spring
- Front Controller - Dispatcher Servlet
- Prototype - Beans
- Dependency Injection
- Factory Pattern - Bean Factory & Application Context
- Template Method - org.springframework.web.servlet.mvc.AbstractController
Aspect Oriented Programming - AOP
https://nitinkc.github.io/spring/microservices/spring-aop/
Bean Scope
https://nitinkc.github.io/microservices/spring-beans/#bean-scope