Spring - 如何正确使用@Autowired 来防止控制器/MockMvc 为空?
Spring - How to properly use @Autowired to prevent controller / MockMvc from being null?
我正在尝试 运行 一些单元测试并遇到了一个问题,我确信这源于对自动装配的误解。我有一个单元测试 class,其中我试图在 MockMvc 和 REST 控制器上使用 @Autowired
——它们最终都为空。
我看到一些消息来源试图解释为什么会发生这种情况(包括这个 More of Less post and a helpful Whosebug post,它给了我一些见解,但并没有完全帮助我解决我的问题)。
下面是我为重现此问题而创建的示例项目的相关源代码。
ManagerControllerTest.java
@RunWith(SpringRunner.class)
@WebMvcTest(ManagerController.class)
public class ManagerControllerTest {
@Autowired
private MockMvc mockMvc;
@MockBean
private Manager manager;
@Autowired
private ManagerController controller;
@Test
public void controllerNotNull() throws Exception {
assertThat(controller).isNotNull();
}
@Test
public void testStoreSomething() throws Exception {
String path = "/manager/store-something/";
MockHttpServletRequestBuilder builder = MockMvcRequestBuilders.get(path)
.characterEncoding("UTF-8")
.contentType(MediaType.APPLICATION_JSON);
MvcResult result = mockMvc.perform(builder).andReturn();
assertEquals(HttpStatus.CREATED, result.getResponse().getStatus());
}
}
controllerNotNull()
测试结果在
java.lang.AssertionError: Expecting actual not to be null
不过,奇怪的是,当我在 Java 中创建一个新的 Gradle 项目并将我的示例代码从 post 导入其中时,controllerNotNull()
通过了。
并且 testStoreSomething()
结果是
java.lang.NullPointerException at
com.example.sandbox.rest.ManagerControllerTest.testStoreSomething(ManagerControllerTest.java:46)
问题就在这里:我误解了什么?我究竟做错了什么?我可以从控制器中删除 @Autowired
并用 new ManagerController()
实例化它,但我留下了 MockMvc 问题。
ManagerController.java
@Controller
@RequestMapping(value = "/manager/")
public class ManagerController {
Manager manager = new Manager(new StringStorage());
@PostMapping(value = "store-something")
private ResponseEntity<?> storeSomething(String str) {
manager.storeSomething(str);
return new ResponseEntity<>(CREATED);
}
}
Manager.java
public class Manager {
private final Storage storage;
public Manager(Storage storage) {
this.storage = storage;
}
public void storeSomething(String str) {
storage.store(str);
}
}
Storage.java
public interface Storage {
void store(String str);
}
字符串Storage.java
public class StringStorage implements Storage {
Map<String, String> stringMap;
@Override
public void store(String str) {
stringMap.put(str, str);
}
}
Application.java
@SpringBootApplication
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
build.gradle 已经从原来的 post 编辑(使用 JUnit4),但问题仍然存在。
repositories {
jcenter()
}
apply plugin: 'java'
apply plugin: 'eclipse'
dependencies {
compile group: 'org.springframework.boot', name: 'spring-boot-starter-web', version: '2.0.0.RELEASE'
compile group: 'org.springframework.boot', name: 'spring-boot-starter-tomcat', version: '2.0.0.RELEASE'
testCompile group: 'org.springframework.boot', name: 'spring-boot-starter-test', version: '2.0.0.RELEASE'
testImplementation 'org.junit.jupiter:junit-jupiter-api:4.0.0'
testImplementation 'org.junit.jupiter:junit-jupiter-params:4.0.0'
testCompile group: 'org.mockito', name: 'mockito-core', version: '2.17.0'
testCompile group: 'org.junit.jupiter', name: 'junit-jupiter-engine', version: "4.0.0"
testCompile group: 'org.junit.platform', name: 'junit-platform-launcher', version: "1.3.1"
}
将注释 AutoConfigureMockMvc 添加到您的测试中 class
@RunWith(SpringRunner.class)
@AutoConfigureMockMvc
public class ManagerControllerTest {
@Autowired
private MockMvc mockMvc;
@MockBean
private Manager manager;
@Autowired
private ManagerController controller;
@Test
public void controllerNotNull() throws Exception {
assertThat(controller).isNotNull();
}
@Test
public void testStoreSomething() throws Exception {
this.mvc.perform(get("/manager/store-something/"))
.contentType(MediaType.APPLICATION_JSON)
.andExpect(status().isOk())
}
或者,您也可以将 TestRestTemplate 与 @SpringBootTest(webEnvironment= SpringBootTest.WebEnvironment.RANDOM_PORT)
一起使用
@RestController
@RequestMapping(value = "/manager")
public class ManagerController {
@Autowired
Manager manager;
@PostMapping(value = "/store-something")
private ResponseEntity<?> storeSomething(String str) {
manager.storeSomething(str);
return new ResponseEntity<>(CREATED);
}
}
@Component
public class Manager {
@Autowired
private Storage storage;
public void storeSomething(String str) {
storage.store(str);
}
}
public interface Storage {
void store(String str);
}
@Service
public class StringStorage implements Storage {
Map<String, String> stringMap = new HashMap<>();
@Override
public void store(String str) {
stringMap.put(str, str);
}
}
@SpringBootApplication
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
测试class:
@RunWith(SpringRunner.class)
@SpringBootTest
@AutoConfigureMockMvc
public class ApplicationTest {
@Autowired
private MockMvc mockMvc;
@Test
public void shouldReturnDefaultMessage() throws Exception {
this.mockMvc.perform(post("/manager/store-something")).andDo(print()).andExpect(status().isOk())
.andExpect(content().string(containsString("Hello World")));
}
}
我正在尝试 运行 一些单元测试并遇到了一个问题,我确信这源于对自动装配的误解。我有一个单元测试 class,其中我试图在 MockMvc 和 REST 控制器上使用 @Autowired
——它们最终都为空。
我看到一些消息来源试图解释为什么会发生这种情况(包括这个 More of Less post and a helpful Whosebug post,它给了我一些见解,但并没有完全帮助我解决我的问题)。
下面是我为重现此问题而创建的示例项目的相关源代码。
ManagerControllerTest.java
@RunWith(SpringRunner.class)
@WebMvcTest(ManagerController.class)
public class ManagerControllerTest {
@Autowired
private MockMvc mockMvc;
@MockBean
private Manager manager;
@Autowired
private ManagerController controller;
@Test
public void controllerNotNull() throws Exception {
assertThat(controller).isNotNull();
}
@Test
public void testStoreSomething() throws Exception {
String path = "/manager/store-something/";
MockHttpServletRequestBuilder builder = MockMvcRequestBuilders.get(path)
.characterEncoding("UTF-8")
.contentType(MediaType.APPLICATION_JSON);
MvcResult result = mockMvc.perform(builder).andReturn();
assertEquals(HttpStatus.CREATED, result.getResponse().getStatus());
}
}
controllerNotNull()
测试结果在
java.lang.AssertionError: Expecting actual not to be null
不过,奇怪的是,当我在 Java 中创建一个新的 Gradle 项目并将我的示例代码从 post 导入其中时,controllerNotNull()
通过了。
并且 testStoreSomething()
结果是
java.lang.NullPointerException at com.example.sandbox.rest.ManagerControllerTest.testStoreSomething(ManagerControllerTest.java:46)
问题就在这里:我误解了什么?我究竟做错了什么?我可以从控制器中删除 @Autowired
并用 new ManagerController()
实例化它,但我留下了 MockMvc 问题。
ManagerController.java
@Controller
@RequestMapping(value = "/manager/")
public class ManagerController {
Manager manager = new Manager(new StringStorage());
@PostMapping(value = "store-something")
private ResponseEntity<?> storeSomething(String str) {
manager.storeSomething(str);
return new ResponseEntity<>(CREATED);
}
}
Manager.java
public class Manager {
private final Storage storage;
public Manager(Storage storage) {
this.storage = storage;
}
public void storeSomething(String str) {
storage.store(str);
}
}
Storage.java
public interface Storage {
void store(String str);
}
字符串Storage.java
public class StringStorage implements Storage {
Map<String, String> stringMap;
@Override
public void store(String str) {
stringMap.put(str, str);
}
}
Application.java
@SpringBootApplication
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
build.gradle 已经从原来的 post 编辑(使用 JUnit4),但问题仍然存在。
repositories {
jcenter()
}
apply plugin: 'java'
apply plugin: 'eclipse'
dependencies {
compile group: 'org.springframework.boot', name: 'spring-boot-starter-web', version: '2.0.0.RELEASE'
compile group: 'org.springframework.boot', name: 'spring-boot-starter-tomcat', version: '2.0.0.RELEASE'
testCompile group: 'org.springframework.boot', name: 'spring-boot-starter-test', version: '2.0.0.RELEASE'
testImplementation 'org.junit.jupiter:junit-jupiter-api:4.0.0'
testImplementation 'org.junit.jupiter:junit-jupiter-params:4.0.0'
testCompile group: 'org.mockito', name: 'mockito-core', version: '2.17.0'
testCompile group: 'org.junit.jupiter', name: 'junit-jupiter-engine', version: "4.0.0"
testCompile group: 'org.junit.platform', name: 'junit-platform-launcher', version: "1.3.1"
}
将注释 AutoConfigureMockMvc 添加到您的测试中 class
@RunWith(SpringRunner.class)
@AutoConfigureMockMvc
public class ManagerControllerTest {
@Autowired
private MockMvc mockMvc;
@MockBean
private Manager manager;
@Autowired
private ManagerController controller;
@Test
public void controllerNotNull() throws Exception {
assertThat(controller).isNotNull();
}
@Test
public void testStoreSomething() throws Exception {
this.mvc.perform(get("/manager/store-something/"))
.contentType(MediaType.APPLICATION_JSON)
.andExpect(status().isOk())
}
或者,您也可以将 TestRestTemplate 与 @SpringBootTest(webEnvironment= SpringBootTest.WebEnvironment.RANDOM_PORT)
一起使用@RestController
@RequestMapping(value = "/manager")
public class ManagerController {
@Autowired
Manager manager;
@PostMapping(value = "/store-something")
private ResponseEntity<?> storeSomething(String str) {
manager.storeSomething(str);
return new ResponseEntity<>(CREATED);
}
}
@Component
public class Manager {
@Autowired
private Storage storage;
public void storeSomething(String str) {
storage.store(str);
}
}
public interface Storage {
void store(String str);
}
@Service
public class StringStorage implements Storage {
Map<String, String> stringMap = new HashMap<>();
@Override
public void store(String str) {
stringMap.put(str, str);
}
}
@SpringBootApplication
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
测试class:
@RunWith(SpringRunner.class)
@SpringBootTest
@AutoConfigureMockMvc
public class ApplicationTest {
@Autowired
private MockMvc mockMvc;
@Test
public void shouldReturnDefaultMessage() throws Exception {
this.mockMvc.perform(post("/manager/store-something")).andDo(print()).andExpect(status().isOk())
.andExpect(content().string(containsString("Hello World")));
}
}