Close

Spring MVC - Request Scoped Bean

[Last Updated: Mar 28, 2026]

The scope 'request' defines a single bean definition which lives within a single HTTP request. That means for each HTTP request a new bean instance is created. This scope is only valid in the context of a web-aware ApplicationContext.

This scope might be very useful in using helper objects across multiple beans through out a single request. This is particularly useful as compared to passing various parameters through a long chain of methods calls which sometimes becomes unmaintainable and it's difficult to add new parameters.

Example

Since we are going to inject a 'request' scoped bean into a default 'singleton' scoped bean, we have to avoid narrower scope bean DI problem. For that we will use JSR 330 Provider approach. Here's a list of other solution.

Creating request scoped bean

@Component
@Scope(WebApplicationContext.SCOPE_REQUEST)
public class EmployeeDetails implements Serializable {
    private Employee employee;
    private int lastYearSalary;

    public void setEmployee(Employee employee) {
        this.employee = employee;
    }

    public Employee getEmployee() {
        return employee;
    }

    public int getLastYearSalary() {
        return lastYearSalary;
    }

    public void setLastYearSalary(int lastYearSalary) {
        this.lastYearSalary = lastYearSalary;
    }
}
public class Employee {
    private String id;
    private String name;
    private String dept;

    public Employee(String id, String name, String dept) {
        this.id = id;
        this.name = name;
        this.dept = dept;
    }

    public String getId() {
        return id;
    }

    public String getName() {
        return name;
    }

    public String getDept() {
        return dept;
    }
}

Injecting request bean to Controller

@Controller
@RequestMapping("/employees")
public class EmployeeController {
    @Autowired
    private Provider<EmployeeDetails> employeeDetailsProvider;
    @Autowired
    AppService appService;

    @RequestMapping("/{id}")
    public String handler(@PathVariable("id") String employeeId) {
        employeeDetailsProvider.get().setEmployee(getEmployeeById(employeeId));
        appService.findEmployeeSalary();
        return "employee-page";
    }

    private Employee getEmployeeById(String employeeId) {
        //todo: call to backend instead
        Employee employee = new Employee(employeeId, RandomUtil.getFullName(),
                RandomUtil.getAnyOf("Admin", "IT", "Sales"));
        return employee;
    }
}

Injecting request bean to other beans

@Service
public class AppService {
    @Autowired
    private Provider<EmployeeDetails> employeeDetailsProvider;

    public void findEmployeeSalary() {
        EmployeeDetails employeeDetails = employeeDetailsProvider.get();
        Employee employee = employeeDetails.getEmployee();
        employeeDetails.setLastYearSalary(getEmployeeSalary(employee));
    }

    private int getEmployeeSalary(Employee employee) {
        //todo: call backend instead
        int salary = 100 * RandomUtil.getInt(30, 100);
        return salary;
    }
}

Output

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

mvn jetty:run
$ curl -s http://localhost:8080/employees/1
<html>
<body>
<h3>Employee details </h3>
<p>Name: Slone Conover</p>
<p>Dept: Sales</p>
<p>Last year salary: 9600</p>
</body>
</html>

As seen above, all information of EmployeeDetails is there even though they were populated in different places by autowiring. That shows a single instance of EmployeeDetails was used through out the request.

Integration Tests

package com.logicbig.example;

import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.springframework.test.context.junit.jupiter.web.SpringJUnitWebConfig;
import org.springframework.test.web.servlet.assertj.MockMvcTester;
import org.springframework.test.web.servlet.assertj.MvcTestResult;
import org.springframework.web.context.WebApplicationContext;
import static org.assertj.core.api.Assertions.assertThat;

@SpringJUnitWebConfig(MyWebConfig.class)
class EmployeeControllerIntegrationTest {

    private MockMvcTester mockMvc;

    @BeforeEach
    void setup(WebApplicationContext wac) {
        // Initialize MockMvcTester using the WebApplicationContext
        this.mockMvc = MockMvcTester.from(wac);
    }

    @Test
    void givenValidEmployeeId_whenGetEmployee_thenEmployeeDetailsPopulated() {
        // Use the fluent 'get' API and AssertJ assertions
        MvcTestResult result =
                mockMvc.get().uri("/employees/1").exchange();

        assertThat(result).hasStatusOk();

        // Extracting request attributes for scoped-bean verification
        EmployeeDetails details = (EmployeeDetails) result.getRequest()
                                                          .getAttribute(
                                                                  "employeeDetails");

        assertThat(details).isNotNull();
        assertThat(details.getEmployee().getId()).isEqualTo("1");
    }

    @Test
    void twoConsecutiveRequests_shouldReceiveSeparateEmployeeDetailsInstances() {
        MvcTestResult first = mockMvc.get().uri("/employees/1").exchange();
        MvcTestResult second = mockMvc.get().uri("/employees/2").exchange();

        // Verify requests are distinct
        assertThat(first.getRequest()).isNotSameAs(second.getRequest());

        String idFromFirst = extractEmployeeId(first);
        String idFromSecond = extractEmployeeId(second);

        assertThat(idFromFirst).isEqualTo("1");
        assertThat(idFromSecond).isEqualTo("2");
        assertThat(idFromFirst).isNotEqualTo(idFromSecond);
    }

    // Internal helper using AssertJ-style extraction
    private String extractEmployeeId(MvcTestResult result) {
        assertThat(result)
                .request()
                .attributes()
                .extractingByKey("employeeDetails")
                .isInstanceOf(EmployeeDetails.class);

        // 2. Extract (The "Logic")
        EmployeeDetails details = (EmployeeDetails) result.getRequest().getAttribute("employeeDetails");
        return details.getEmployee().getId();
    }
}
mvn clean test -Dtest="EmployeeControllerIntegrationTest"

Output

$ mvn clean test -Dtest="EmployeeControllerIntegrationTest"
[INFO] Scanning for projects...
[WARNING]
[WARNING] Some problems were encountered while building the effective model for com.logicbig.example:request-scope-example: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:request-scope-example >-------------
[INFO] Building request-scope-example 1.0-SNAPSHOT
[INFO] from pom.xml
[INFO] --------------------------------[ war ]---------------------------------
[INFO]
[INFO] --- clean:3.2.0:clean (default-clean) @ request-scope-example ---
[INFO] Deleting D:\example-projects\spring-mvc\scopes\request-scope-example\target
[INFO]
[INFO] --- resources:3.3.1:resources (default-resources) @ request-scope-example ---
[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\scopes\request-scope-example\src\main\resources
[INFO]
[INFO] --- compiler:3.7.0:compile (default-compile) @ request-scope-example ---
[INFO] Changes detected - recompiling the module!
[INFO] Compiling 7 source files to D:\example-projects\spring-mvc\scopes\request-scope-example\target\classes
[INFO]
[INFO] --- resources:3.3.1:testResources (default-testResources) @ request-scope-example ---
[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\scopes\request-scope-example\src\test\resources
[INFO]
[INFO] --- compiler:3.7.0:testCompile (default-testCompile) @ request-scope-example ---
[INFO] Changes detected - recompiling the module!
[INFO] Compiling 1 source file to D:\example-projects\spring-mvc\scopes\request-scope-example\target\test-classes
[INFO]
[INFO] --- surefire:3.2.5:test (default-test) @ request-scope-example ---
[INFO] Using auto detected provider org.apache.maven.surefire.junit4.JUnit4Provider
[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.EmployeeControllerIntegrationTest
[INFO] Tests run: 2, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.483 s -- in com.logicbig.example.EmployeeControllerIntegrationTest
[INFO]
[INFO] Results:
[INFO]
[INFO] Tests run: 2, Failures: 0, Errors: 0, Skipped: 0
[INFO]
[INFO] ------------------------------------------------------------------------
[INFO] BUILD SUCCESS
[INFO] ------------------------------------------------------------------------
[INFO] Total time: 2.636 s
[INFO] Finished at: 2026-03-28T09:20:59+08:00
[INFO] ------------------------------------------------------------------------
INFO: JSR-330 'javax.inject.Inject' annotation found and supported for autowiring

Example Project

Dependencies and Technologies Used:

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

    Version compatibilities of spring-webmvc with this example:

      javax.servlet-api:3.x
    • 3.2.9.RELEASE
    • 3.2.10.RELEASE
    • 3.2.11.RELEASE
    • 3.2.12.RELEASE
    • 3.2.13.RELEASE
    • 3.2.14.RELEASE
    • 3.2.15.RELEASE
    • 3.2.16.RELEASE
    • 3.2.17.RELEASE
    • 3.2.18.RELEASE
    • 4.0.0.RELEASE
    • 4.0.1.RELEASE
    • 4.0.2.RELEASE
    • 4.0.3.RELEASE
    • 4.0.4.RELEASE
    • 4.0.5.RELEASE
    • 4.0.6.RELEASE
    • 4.0.7.RELEASE
    • 4.0.8.RELEASE
    • 4.0.9.RELEASE
    • 4.1.0.RELEASE
    • 4.1.1.RELEASE
    • 4.1.2.RELEASE
    • 4.1.3.RELEASE
    • 4.1.4.RELEASE
    • 4.1.5.RELEASE
    • 4.1.6.RELEASE
    • 4.1.7.RELEASE
    • 4.1.8.RELEASE
    • 4.1.9.RELEASE
    • 4.2.0.RELEASE
    • 4.2.1.RELEASE
    • 4.2.2.RELEASE
    • 4.2.3.RELEASE
    • 4.2.4.RELEASE
    • 4.2.5.RELEASE
    • 4.2.6.RELEASE
    • 4.2.7.RELEASE
    • 4.2.8.RELEASE
    • 4.2.9.RELEASE
    • 4.3.0.RELEASE
    • 4.3.1.RELEASE
    • 4.3.2.RELEASE
    • 4.3.3.RELEASE
    • 4.3.4.RELEASE
    • 4.3.5.RELEASE
    • 4.3.6.RELEASE
    • 4.3.7.RELEASE
    • 4.3.8.RELEASE
    • 4.3.9.RELEASE
    • 4.3.10.RELEASE
    • 4.3.11.RELEASE
    • 4.3.12.RELEASE
    • 4.3.13.RELEASE
    • 4.3.14.RELEASE
    • 4.3.15.RELEASE
    • 4.3.16.RELEASE
    • 4.3.17.RELEASE
    • 4.3.18.RELEASE
    • 4.3.19.RELEASE
    • 4.3.20.RELEASE
    • 4.3.21.RELEASE
    • 4.3.22.RELEASE
    • 4.3.23.RELEASE
    • 4.3.24.RELEASE
    • 4.3.25.RELEASE
    • 4.3.26.RELEASE
    • 4.3.27.RELEASE
    • 4.3.28.RELEASE
    • 4.3.29.RELEASE
    • 4.3.30.RELEASE
    • 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
    • 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)
  • junit-jupiter-engine 6.0.3 (Module "junit-jupiter-engine" of JUnit)
  • jakarta.servlet-api 6.1.0 (Jakarta Servlet API documentation)
  • jakarta.inject-api 2.0.1 (Jakarta Dependency Injection)
  • hamcrest 3.0 (Core API and libraries of hamcrest matcher framework)
  • assertj-core 3.26.3 (Rich and fluent assertions for testing in Java)
  • JDK 25
  • Maven 3.9.11

Spring MVC - Request Scoped Bean Example Select All Download
  • request-scope-example
    • src
      • main
        • java
          • com
            • logicbig
              • example
                • EmployeeDetails.java
          • webapp
            • WEB-INF
              • views
        • test
          • java
            • com
              • logicbig
                • example

    See Also

    Join