Close

Spring MVC - Data Binding with Immutable Bean Classes in Spring 5

[Last Updated: May 13, 2026]

Starting Spring 5, data binding can work with immutable classes now. Previously there must be a no argument constructor and suitable setters to initialize the bean against the request parameters.

From Spring issue #19763:

We're now automatically detecting data classes with a single public constructor, resolving the constructor arguments against the request parameters, as long as the parameter names are retained or an @ConstructorProperties annotation is declared. This works for Spring MVC as well as WebFlux.

@ConstructorProperties is a Java annotation meant to use on a constructor to indicate how the parameters of that constructor correspond to the constructed bean's getter methods.

Java 8+ Parameters Flag

With -parameters compiler flag, we can omit @ConstructorProperties if constructor parameter names match exactly.

Example

Immutable Beans

package com.logicbig.example;

import java.beans.ConstructorProperties;

public class CustomerInfo {
    private String customerId;
    private String zipCode;

    @ConstructorProperties({"id", "zip"})
    public CustomerInfo(String customerId, String zipCode) {
        this.customerId = customerId;
        this.zipCode = zipCode;
    }

    public String getCustomerId() {
        return customerId;
    }

    public String getZipCode() {
        return zipCode;
    }

    @Override
    public String toString() {
        return "CustomerInfo{" +
                "customerId='" + customerId + '\'' +
                ", zipCode='" + zipCode + '\'' +
                '}';
    }
}
public class OrderInfo {

    private final String id;
    private final String zip;

    public OrderInfo(String id, String zip) {
        this.id = id;
        this.zip = zip;
    }

    public String getId() {
        return id;
    }

    public String getZip() {
        return zip;
    }

    @Override
    public String toString() {
        return "OrderInfo{" +
                "id='" + id + '\'' +
                ", zip='" + zip + '\'' +
                '}';
    }
}

Example Controller

@Controller
public class CustomerController {

    @ResponseBody
    @GetMapping("/customer")
    public String getCustomerInfo(CustomerInfo ci) {
        return ci.toString();
    }

    @ResponseBody
    @GetMapping("/order")
    public String getCustomerInfo(OrderInfo oi) {
        return oi.toString();
    }
}

JavaConfig

@EnableWebMvc
@Configuration
@ComponentScan
public class AppConfig {
}

Running

To try examples, run embedded Jetty (configured in pom.xml of example project below):

mvn jetty:run

Output:

$ curl -s "http://localhost:8080/customer?id=23&zip=1111"
CustomerInfo{customerId='23', zipCode='1111'}
$ curl -s "http://localhost:8080/order?id=23&zip=1111"
OrderInfo{id='23', zip='1111'}

In versions before Spring 5.0, following exception would be thrown:

SEVERE: Servlet.service() for servlet [springDispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [com.logicbig.example.CustomerInfo]: No default constructor found; nested exception is java.lang.NoSuchMethodException: com.logicbig.example.CustomerInfo.



()] with root cause
java.lang.NoSuchMethodException: com.logicbig.example.CustomerInfo.



()
at java.lang.Class.getConstructor0(Class.java:3082)
......





Integration Tests

package com.logicbig.example;

import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.junit.jupiter.web.SpringJUnitWebConfig;
import org.springframework.test.web.servlet.MockMvc;
import org.springframework.test.web.servlet.setup.MockMvcBuilders;
import org.springframework.web.context.WebApplicationContext;
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.content;
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;

@SpringJUnitWebConfig(AppConfig.class)
class CustomerControllerIntegrationTest {

    @Autowired
    private WebApplicationContext webApplicationContext;

    private MockMvc mockMvc;

    @BeforeEach
    void setUp() {
        this.mockMvc = MockMvcBuilders
                .webAppContextSetup(webApplicationContext)
                .build();
    }

    @Test
    void testCustomerInfoBinding() throws Exception {
        mockMvc.perform(get("/customer")
                                .param("id", "23")
                                .param("zip", "1111"))
               .andExpect(status().isOk())
               .andExpect(content().string(
                       "CustomerInfo{customerId='23', zipCode='1111'}"));
    }

    @Test
    @DisplayName("Should bind query parameters to OrderInfo immutable bean")
    void testOrderInfoBinding() throws Exception {
        mockMvc.perform(get("/order")
                                .param("id", "23")
                                .param("zip", "1111"))
               .andExpect(status().isOk())
               .andExpect(content().string("OrderInfo{id='23', zip='1111'}"));
    }
}
mvn clean test -Dtest="CustomerControllerIntegrationTest"

Output

$ mvn clean test -Dtest="CustomerControllerIntegrationTest"
[INFO] Scanning for projects...
[WARNING]
[WARNING] Some problems were encountered while building the effective model for com.logicbig.example:spring-5-immutable-class-data-binding:war:1.0-SNAPSHOT
[WARNING] 'build.plugins.plugin.version' for org.apache.maven.plugins:maven-war-plugin is missing. @ line 42, column 21
[WARNING]
[WARNING] It is highly recommended to fix these problems because they threaten the stability of your build.
[WARNING]
[WARNING] For this reason, future Maven versions might no longer support building such malformed projects.
[WARNING]
[INFO]
[INFO] -----< com.logicbig.example:spring-5-immutable-class-data-binding >-----
[INFO] Building spring-5-immutable-class-data-binding 1.0-SNAPSHOT
[INFO] from pom.xml
[INFO] --------------------------------[ war ]---------------------------------
[INFO]
[INFO] --- clean:3.2.0:clean (default-clean) @ spring-5-immutable-class-data-binding ---
[INFO] Deleting D:\example-projects\spring-mvc\data-binding\spring-5-immutable-class-data-binding\target
[INFO]
[INFO] --- resources:3.3.1:resources (default-resources) @ spring-5-immutable-class-data-binding ---
[WARNING] Using platform encoding (UTF-8 actually) to copy filtered resources, i.e. build is platform dependent!
[INFO] skip non existing resourceDirectory D:\example-projects\spring-mvc\data-binding\spring-5-immutable-class-data-binding\src\main\resources
[INFO]
[INFO] --- compiler:3.8.0:compile (default-compile) @ spring-5-immutable-class-data-binding ---
[INFO] Changes detected - recompiling the module!
[INFO] Compiling 5 source files to D:\example-projects\spring-mvc\data-binding\spring-5-immutable-class-data-binding\target\classes
[INFO]
[INFO] --- resources:3.3.1:testResources (default-testResources) @ spring-5-immutable-class-data-binding ---
[WARNING] Using platform encoding (UTF-8 actually) to copy filtered resources, i.e. build is platform dependent!
[INFO] skip non existing resourceDirectory D:\example-projects\spring-mvc\data-binding\spring-5-immutable-class-data-binding\src\test\resources
[INFO]
[INFO] --- compiler:3.8.0:testCompile (default-testCompile) @ spring-5-immutable-class-data-binding ---
[INFO] Changes detected - recompiling the module!
[INFO] Compiling 1 source file to D:\example-projects\spring-mvc\data-binding\spring-5-immutable-class-data-binding\target\test-classes
[INFO]
[INFO] --- surefire:3.2.5:test (default-test) @ spring-5-immutable-class-data-binding ---
[INFO] Using auto detected provider org.apache.maven.surefire.junitplatform.JUnitPlatformProvider
[WARNING] file.encoding cannot be set as system property, use <argLine>-Dfile.encoding=...</argLine> instead
[INFO]
[INFO] -------------------------------------------------------
[INFO] T E S T S
[INFO] -------------------------------------------------------
[INFO] Running com.logicbig.example.CustomerControllerIntegrationTest
[INFO] Tests run: 2, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.503 s -- in com.logicbig.example.CustomerControllerIntegrationTest
[INFO]
[INFO] Results:
[INFO]
[INFO] Tests run: 2, Failures: 0, Errors: 0, Skipped: 0
[INFO]
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
[INFO] Total time: 2.801 s
[INFO] Finished at: 2026-05-13T18:44:40+08:00
[INFO] ------------------------------------------------------------------------
INFO: Completed initialization in 4 ms
INFO: Completed initialization in 1 ms

Example Project

Dependencies and Technologies Used:

  • spring-webmvc 7.0.6 (Spring Web MVC)
     Version Compatibility: 5.0.0.RELEASE - 7.0.6Version List
    ×

    Version compatibilities of spring-webmvc with this example:

      javax.servlet-api:3.x
    • 5.0.0.RELEASE
    • 5.0.1.RELEASE
    • 5.0.2.RELEASE
    • 5.0.3.RELEASE
    • 5.0.4.RELEASE
    • 5.0.5.RELEASE
    • 5.0.6.RELEASE
    • 5.0.7.RELEASE
    • 5.0.8.RELEASE
    • 5.0.9.RELEASE
    • 5.0.10.RELEASE
    • 5.0.11.RELEASE
    • 5.0.12.RELEASE
    • 5.0.13.RELEASE
    • 5.0.14.RELEASE
    • 5.0.15.RELEASE
    • 5.0.16.RELEASE
    • 5.0.17.RELEASE
    • 5.0.18.RELEASE
    • 5.0.19.RELEASE
    • 5.0.20.RELEASE
    • 5.1.0.RELEASE
    • 5.1.1.RELEASE
    • 5.1.2.RELEASE
    • 5.1.3.RELEASE
    • 5.1.4.RELEASE
    • 5.1.5.RELEASE
    • 5.1.6.RELEASE
    • 5.1.7.RELEASE
    • 5.1.8.RELEASE
    • 5.1.9.RELEASE
    • 5.1.10.RELEASE
    • 5.1.11.RELEASE
    • 5.1.12.RELEASE
    • 5.1.13.RELEASE
    • 5.1.14.RELEASE
    • 5.1.15.RELEASE
    • 5.1.16.RELEASE
    • 5.1.17.RELEASE
    • 5.1.18.RELEASE
    • 5.1.19.RELEASE
    • 5.1.20.RELEASE
    • 5.2.0.RELEASE
    • 5.2.1.RELEASE
    • 5.2.2.RELEASE
    • 5.2.3.RELEASE
    • 5.2.4.RELEASE
    • 5.2.5.RELEASE
    • 5.2.6.RELEASE
    • 5.2.7.RELEASE
    • 5.2.8.RELEASE
    • 5.2.9.RELEASE
    • 5.2.10.RELEASE
    • 5.2.11.RELEASE
    • 5.2.12.RELEASE
    • 5.2.13.RELEASE
    • 5.2.14.RELEASE
    • 5.2.15.RELEASE
    • 5.2.16.RELEASE
    • 5.2.17.RELEASE
    • 5.2.18.RELEASE
    • 5.2.19.RELEASE
    • 5.2.20.RELEASE
    • 5.2.21.RELEASE
    • 5.2.22.RELEASE
    • 5.2.23.RELEASE
    • 5.2.24.RELEASE
    • 5.2.25.RELEASE
    • 5.3.0
    • 5.3.1
    • 5.3.2
    • 5.3.3
    • 5.3.4
    • javax.servlet-api:4.x
    • 5.3.5
    • 5.3.6
    • 5.3.7
    • 5.3.8
    • 5.3.9
    • 5.3.10
    • 5.3.11
    • 5.3.12
    • 5.3.13
    • 5.3.14
    • 5.3.15
    • 5.3.16
    • 5.3.17
    • 5.3.18
    • 5.3.19
    • 5.3.20
    • 5.3.21
    • 5.3.22
    • 5.3.23
    • 5.3.24
    • 5.3.25
    • 5.3.26
    • 5.3.27
    • 5.3.28
    • 5.3.29
    • 5.3.30
    • 5.3.31
    • 5.3.32
    • 5.3.33
    • 5.3.34
    • 5.3.35
    • 5.3.36
    • 5.3.37
    • 5.3.38
    • 5.3.39
    • javax.* -> jakarta.*
      jakarta.servlet-api:6.x
      Java 17 min
    • 6.0.0
    • 6.0.1
    • 6.0.2
    • 6.0.3
    • 6.0.4
    • 6.0.5
    • 6.0.6
    • 6.0.7
    • 6.0.8
    • 6.0.9
    • 6.0.10
    • 6.0.11
    • 6.0.12
    • 6.0.13
    • 6.0.14
    • 6.0.15
    • 6.0.16
    • 6.0.17
    • 6.0.18
    • 6.0.19
    • 6.0.20
    • 6.0.21
    • 6.0.22
    • 6.0.23
    • 6.1.0
    • 6.1.1
    • 6.1.2
    • 6.1.3
    • 6.1.4
    • 6.1.5
    • 6.1.6
    • 6.1.7
    • 6.1.8
    • 6.1.9
    • 6.1.10
    • 6.1.11
    • 6.1.12
    • 6.1.13
    • 6.1.14
    • 6.1.15
    • 6.1.16
    • 6.1.17
    • 6.1.18
    • 6.1.19
    • 6.1.20
    • 6.1.21
    • 6.2.0
    • 6.2.1
    • 6.2.2
    • 6.2.3
    • 6.2.4
    • 6.2.5
    • 6.2.6
    • 6.2.7
    • 6.2.8
    • 6.2.9
    • 6.2.10
    • 6.2.11
    • 6.2.12
    • 6.2.13
    • 6.2.14
    • 6.2.15
    • 6.2.16
    • 6.2.17
    • 6.2.18
    • 7.0.0
    • 7.0.1
    • 7.0.2
    • 7.0.3
    • 7.0.4
    • 7.0.5
    • 7.0.6

    Versions in green have been tested.

  • spring-test 7.0.6 (Spring TestContext Framework)
  • jakarta.servlet-api 6.1.0 (Jakarta Servlet API documentation)
  • junit-jupiter-engine 6.0.3 (Module "junit-jupiter-engine" of JUnit)
  • hamcrest 3.0 (Core API and libraries of hamcrest matcher framework)
  • JDK 25
  • Maven 3.9.11

Sporing 5 - Data Binding with Immutable Bean Classes Select All Download
  • spring-5-immutable-class-data-binding
    • src
      • main
        • java
          • com
            • logicbig
              • example
                • CustomerInfo.java
        • test
          • java
            • com
              • logicbig
                • example

    See Also

    Join