Skip to content

Instantly share code, notes, and snippets.

@salmanwahed
Created July 4, 2024 08:42
Show Gist options
  • Save salmanwahed/28003438f24da0c737fda3c2ef3cf7e7 to your computer and use it in GitHub Desktop.
Save salmanwahed/28003438f24da0c737fda3c2ef3cf7e7 to your computer and use it in GitHub Desktop.
Spring Boot is a popular framework for building Java-based applications, especially web applications. Adhering to best practices helps ensure that your code is maintainable, scalable, and efficient. Here are some best practices for Spring Boot development, along with examples.

Spring Boot Best Practices

1. Use Spring Boot Starters

Best Practice: Leverage Spring Boot starters to simplify dependency management.

Example:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>

2. Follow Layered Architecture

Best Practice: Organize your code into layers such as Controller, Service, and Repository.

Example:

@RestController
@RequestMapping("/api/users")
public class UserController {

    @Autowired
    private UserService userService;

    @GetMapping("/{id}")
    public ResponseEntity<User> getUserById(@PathVariable Long id) {
        return ResponseEntity.ok(userService.getUserById(id));
    }
}

@Service
public class UserService {

    @Autowired
    private UserRepository userRepository;

    public User getUserById(Long id) {
        return userRepository.findById(id).orElseThrow(() -> new UserNotFoundException(id));
    }
}

@Repository
public interface UserRepository extends JpaRepository<User, Long> {
}

3. Use DTOs for Data Transfer

Best Practice: Use Data Transfer Objects (DTOs) to transfer data between layers.

Example:

public class UserDTO {
    private Long id;
    private String name;
    private String email;
    // getters and setters
}

public class UserMapper {
    public static UserDTO toDto(User user) {
        UserDTO dto = new UserDTO();
        dto.setId(user.getId());
        dto.setName(user.getName());
        dto.setEmail(user.getEmail());
        return dto;
    }

    public static User toEntity(UserDTO dto) {
        User user = new User();
        user.setId(dto.getId());
        user.setName(dto.getName());
        user.setEmail(dto.getEmail());
        return user;
    }
}

4. Handle Exceptions Properly

Best Practice: Use @ControllerAdvice to handle exceptions globally.

Example:

@ControllerAdvice
public class GlobalExceptionHandler {

    @ExceptionHandler(UserNotFoundException.class)
    public ResponseEntity<ErrorResponse> handleUserNotFoundException(UserNotFoundException ex) {
        ErrorResponse errorResponse = new ErrorResponse();
        errorResponse.setMessage(ex.getMessage());
        errorResponse.setTimestamp(LocalDateTime.now());
        return ResponseEntity.status(HttpStatus.NOT_FOUND).body(errorResponse);
    }
}

public class UserNotFoundException extends RuntimeException {
    public UserNotFoundException(Long id) {
        super("User not found with id: " + id);
    }
}

5. Use Profiles for Environment-Specific Configuration

Best Practice: Use Spring profiles to manage different configurations for different environments.

Example:

# application.yml
spring:
  profiles:
    active: dev

# application-dev.yml
spring:
  datasource:
    url: jdbc:mysql://localhost:3306/devdb
    username: devuser
    password: devpass

# application-prod.yml
spring:
  datasource:
    url: jdbc:mysql://prod-server:3306/proddb
    username: produser
    password: prodpass

6. Leverage Spring Boot Actuator

Best Practice: Use Spring Boot Actuator to monitor and manage your application.

Example:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
management:
  endpoints:
    web:
      exposure:
        include: health,info

7. Write Unit Tests

Best Practice: Write unit tests for your service and repository layers.

Example:

@SpringBootTest
@RunWith(SpringRunner.class)
public class UserServiceTest {

    @Autowired
    private UserService userService;

    @MockBean
    private UserRepository userRepository;

    @Test
    public void testGetUserById() {
        User user = new User();
        user.setId(1L);
        user.setName("John Doe");

        Mockito.when(userRepository.findById(1L)).thenReturn(Optional.of(user));

        UserDTO userDTO = userService.getUserById(1L);
        assertEquals("John Doe", userDTO.getName());
    }
}

8. Externalize Configuration

Best Practice: Externalize your configuration to make your application portable and easier to manage.

Example:

# application.yml
app:
  name: MySpringBootApp
  description: This is a sample Spring Boot application
@Component
@ConfigurationProperties(prefix = "app")
public class AppProperties {
    private String name;
    private String description;
    // getters and setters
}

9. Use Lombok for Boilerplate Code

Best Practice: Use Lombok to reduce boilerplate code for getters, setters, and constructors.

Example:

@Data
@NoArgsConstructor
@AllArgsConstructor
@Entity
public class User {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String name;
    private String email;
}

10. Secure Your Application

Best Practice: Use Spring Security to secure your application.

Example:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-security</artifactId>
</dependency>
@EnableWebSecurity
public class SecurityConfig extends WebSecurityConfigurerAdapter {

    @Override
    protected void configure(HttpSecurity http) throws Exception {
        http
            .authorizeRequests()
            .antMatchers("/api/public/**").permitAll()
            .anyRequest().authenticated()
            .and()
            .httpBasic();
    }
}

11. Implement Logging Properly

Best Practice: Use SLF4J with Logback for logging. Avoid using System.out.println() for logging.

Example:

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

@RestController
public class UserController {
    
    private static final Logger logger = LoggerFactory.getLogger(UserController.class);

    @GetMapping("/{id}")
    public ResponseEntity<User> getUserById(@PathVariable Long id) {
        logger.info("Fetching user with id: {}", id);
        User user = userService.getUserById(id);
        if (user == null) {
            logger.warn("User with id {} not found", id);
            return ResponseEntity.notFound().build();
        }
        return ResponseEntity.ok(user);
    }
}

12. Use Validation Annotations

Best Practice: Use Bean Validation (JSR-380) annotations to validate request parameters and payloads.

Example:

import javax.validation.constraints.NotNull;
import javax.validation.constraints.Size;

public class UserDTO {
    @NotNull
    private Long id;

    @NotNull
    @Size(min = 2, max = 30)
    private String name;

    @Email
    private String email;
    // getters and setters
}

@RestController
@RequestMapping("/api/users")
public class UserController {

    @PostMapping
    public ResponseEntity<User> createUser(@Valid @RequestBody UserDTO userDTO) {
        User user = userService.createUser(UserMapper.toEntity(userDTO));
        return ResponseEntity.status(HttpStatus.CREATED).body(user);
    }
}

13. Use Caching Where Appropriate

Best Practice: Use Spring Cache to improve performance by caching frequently accessed data.

Example:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-cache</artifactId>
</dependency>
@EnableCaching
@SpringBootApplication
public class MyApp {
    public static void main(String[] args) {
        SpringApplication.run(MyApp.class, args);
    }
}

@Service
public class UserService {

    @Cacheable("users")
    public User getUserById(Long id) {
        // Simulate a time-consuming database operation
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return userRepository.findById(id).orElse(null);
    }
}

14. Document Your API

Best Practice: Use Swagger to document your REST APIs.

Example:

<dependency>
    <groupId>io.springfox</groupId>
    <artifactId>springfox-boot-starter</artifactId>
    <version>3.0.0</version>
</dependency>
@Configuration
@EnableSwagger2
public class SwaggerConfig {
    @Bean
    public Docket api() {
        return new Docket(DocumentationType.SWAGGER_2)
                .select()
                .apis(RequestHandlerSelectors.any())
                .paths(PathSelectors.any())
                .build();
    }
}

15. Avoid Circular Dependencies

Best Practice: Design your application to avoid circular dependencies. Use constructor injection wherever possible.

Example:

@Service
public class UserService {
    
    private final UserRepository userRepository;

    @Autowired
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }
}

16. Use Asynchronous Processing

Best Practice: Use @Async for asynchronous method execution.

Example:

@EnableAsync
@SpringBootApplication
public class MyApp {
    public static void main(String[] args) {
        SpringApplication.run(MyApp.class, args);
    }
}

@Service
public class EmailService {

    @Async
    public void sendEmail(String to, String body) {
        // Simulate email sending
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("Email sent to: " + to);
    }
}

17. Optimize Database Access

Best Practice: Use Spring Data JPA efficiently and consider using native queries for complex operations.

Example:

public interface UserRepository extends JpaRepository<User, Long> {

    @Query("SELECT u FROM User u WHERE u.email = ?1")
    User findByEmail(String email);

    @Query(value = "SELECT * FROM users WHERE email = ?1", nativeQuery = true)
    User findByEmailNative(String email);
}

18. Use Constructor Injection

Best Practice: Prefer constructor injection over field injection for better testability and immutability.

Example:

@Service
public class UserService {
    
    private final UserRepository userRepository;

    @Autowired
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }
}

19. Profile Your Application

Best Practice: Profile your application to identify performance bottlenecks.

Example:

Use tools like Spring Boot DevTools and JProfiler.

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-devtools</artifactId>
    <optional>true</optional>
</dependency>

20. Secure Sensitive Information

Best Practice: Store sensitive information securely using Spring Boot's encryption support.

Example:

# application.yml
encrypt:
  key: mysecretkey

# application.properties
# Encrypt passwords and other sensitive data
spring.datasource.password={cipher}encryptedpassword

21. Use Appropriate HTTP Status Codes

Best Practice: Return the correct HTTP status codes from your REST endpoints.

Example:

@GetMapping("/{id}")
public ResponseEntity<User> getUserById(@PathVariable Long id) {
    User user = userService.getUserById(id);
    if (user == null) {
        return ResponseEntity.status(HttpStatus.NOT_FOUND).body(null);
    }
    return ResponseEntity.ok(user);
}
java


### 22. Perform Integration Testing

**Best Practice:** Write integration tests to ensure that your components work together as expected.

*Example:*

```java
@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
public class UserControllerIntegrationTest {

    @Autowired
    private TestRestTemplate restTemplate;

    @Test
    public void testGetUserById() {
        ResponseEntity<User> response = restTemplate.getForEntity("/api/users/1", User.class);
        assertEquals(HttpStatus.OK, response.getStatusCode());
        assertNotNull(response.getBody());
    }
}

23. Use Rate Limiting

Best Practice: Implement rate limiting to protect your application from abuse.

Example: Use libraries like Bucket4j for rate limiting.

<dependency>
    <groupId>com.github.vladimir-bukhtoyarov</groupId>
    <artifactId>bucket4j-spring-boot-starter</artifactId>
    <version>0.2.0</version>
</dependency>
@RestController
@RequestMapping("/api/users")
public class UserController {

    @Bucket4jRateLimit(bucketType = RateLimiterType.SIMPLE, value = "10;5;1")
    @GetMapping("/{id}")
    public ResponseEntity<User> getUserById(@PathVariable Long id) {
        User user = userService.getUserById(id);
        if (user == null) {
            return ResponseEntity.status(HttpStatus.NOT_FOUND).body(null);
        }
        return ResponseEntity.ok(user);
    }
}

24. Configuration Management

Externalize Configuration:

# application.yml
server:
  port: 8080

spring:
  datasource:
    url: jdbc:mysql://localhost:3306/mydb
    username: user
    password: pass

Profiles for Environment-Specific Configurations:

# application-dev.yml
server:
  port: 8081
spring:
  datasource:
    url: jdbc:mysql://localhost:3306/devdb
    username: devuser
    password: devpass
# application-prod.yml
server:
  port: 8080
spring:
  datasource:
    url: jdbc:mysql://localhost:3306/proddb
    username: produser
    password: prodpass

Secure Sensitive Information:

# application.yml
spring:
  datasource:
    url: jdbc:mysql://localhost:3306/mydb
    username: ${DB_USERNAME}
    password: ${DB_PASSWORD}

Use environment variables to pass sensitive information.

25. Dependency Management

Use Dependency Injection:

@Service
public class UserService {
    private final UserRepository userRepository;

    @Autowired
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    public User getUserById(Long id) {
        return userRepository.findById(id).orElse(null);
    }
}

26. Application Design

Follow SOLID Principles:

// Single Responsibility Principle
@Service
public class UserService {
    private final UserRepository userRepository;

    @Autowired
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    public User getUserById(Long id) {
        return userRepository.findById(id).orElse(null);
    }
}

Use DTOs for Data Transfer:

// UserDTO.java
public class UserDTO {
    private Long id;
    private String name;

    // getters and setters
}
// UserController.java
@RestController
@RequestMapping("/users")
public class UserController {
    private final UserService userService;

    @Autowired
    public UserController(UserService userService) {
        this.userService = userService;
    }

    @GetMapping("/{id}")
    public ResponseEntity<UserDTO> getUserById(@PathVariable Long id) {
        User user = userService.getUserById(id);
        UserDTO userDTO = new UserDTO();
        userDTO.setId(user.getId());
        userDTO.setName(user.getName());
        return ResponseEntity.ok(userDTO);
    }
}

27. Performance and Scalability

Enable Caching:

@Configuration
@EnableCaching
public class CacheConfig {
}

@Service
public class UserService {
    private final UserRepository userRepository;

    @Autowired
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    @Cacheable("users")
    public User getUserById(Long id) {
        return userRepository.findById(id).orElse(null);
    }
}

Asynchronous Processing:

@Configuration
@EnableAsync
public class AsyncConfig {
}

@Service
public class NotificationService {
    @Async
    public void sendEmail(String email) {
        // Code to send email
    }
}

28. Testing

Write Unit Tests:

@RunWith(SpringRunner.class)
@SpringBootTest
public class UserServiceTest {
    @Autowired
    private UserService userService;

    @MockBean
    private UserRepository userRepository;

    @Test
    public void testGetUserById() {
        User user = new User();
        user.setId(1L);
        when(userRepository.findById(1L)).thenReturn(Optional.of(user));

        User found = userService.getUserById(1L);
        assertEquals(1L, found.getId().longValue());
    }
}

Integration Tests:

@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
public class UserControllerTest {
    @Autowired
    private TestRestTemplate restTemplate;

    @Test
    public void testGetUserById() {
        ResponseEntity<UserDTO> response = restTemplate.getForEntity("/users/1", UserDTO.class);
        assertEquals(HttpStatus.OK, response.getStatusCode());
    }
}

29. Security

Implement Security Best Practices:

@EnableWebSecurity
public class SecurityConfig extends WebSecurityConfigurerAdapter {
    @Override
    protected void configure(HttpSecurity http) throws Exception {
        http
            .csrf().disable()
            .authorizeRequests()
            .antMatchers("/admin/**").hasRole("ADMIN")
            .anyRequest().authenticated()
            .and()
            .formLogin()
            .permitAll();
    }

    @Override
    protected void configure(AuthenticationManagerBuilder auth) throws Exception {
        auth.inMemoryAuthentication()
            .withUser("user").password("{noop}password").roles("USER")
            .and()
            .withUser("admin").password("{noop}admin").roles("ADMIN");
    }
}

30. Documentation and Code Quality

JavaDoc:

/**
 * Service class for managing users.
 */
@Service
public class UserService {
    private final UserRepository userRepository;

    @Autowired
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    /**
     * Get a user by ID.
     *
     * @param id the user ID
     * @return the user
     */
    public User getUserById(Long id) {
        return userRepository.findById(id).orElse(null);
    }
}

30. Logging

Consistent Logging Strategy:

@Service
public class UserService {
    private static final Logger logger = LoggerFactory.getLogger(UserService.class);

    private final UserRepository userRepository;

    @Autowired
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    public User getUserById(Long id) {
        logger.info("Fetching user with ID: {}", id);
        return userRepository.findById(id).orElse(null);
    }
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment