重拾后端之Spring Boot(一):REST API的搭建可以这样简单
重拾后端之Spring Boot(二):MongoDb的无缝集成
重拾后端之Spring Boot(三):找回熟悉的Controller,Service
重拾后端之Spring Boot(四):使用 JWT 和 Spring Security 保护 REST API
重拾后端之Spring Boot(五):跨域、自定义查询及分页
重拾后端之Spring Boot(六):热加载、容器和多项目
找回熟悉的Controller,Service
Controller哪儿去了?
对于很多习惯了Spring开发的同学来讲,Controller
,Service
,DAO
这些套路突然间都没了会有不适感。其实呢,这些东西还在,只不过对于较简单的情景下,这些都变成了系统背后帮你做的事情。这一小节我们就先来看看如何将Controller再召唤回来。召唤回来的好处有哪些呢?首先我们可以自定义API URL的路径,其次可以对参数和返回的json结构做一定的处理。
如果要让 TodoController
可以和 TodoRepository
配合工作的话,我们当然需要在 TodoController
中需要引用 TodoRepository
。
public class TodoController {
@Autowired
private TodoRepository repository;
//省略其它部分
}
@Autowired
这个修饰符是用于做依赖性注入的,上面的用法叫做 field injection
,直接做类成员的注入。但Spring现在鼓励用构造函数来做注入,所以,我们来看看构造函数的注入方法:
public class TodoController {
private TodoRepository repository;
@Autowired
public TodoController(TodoRepository repository){
this.repository = repository;
}
//省略其它部分
}
当然我们为了可以让Spring知道这是一个支持REST API的 Controller
,还是需要标记其为 @RestController
。由于默认的路径映射会在资源根用复数形式,由于todo是辅音后的o结尾,按英语习惯,会映射成 todoes
。但这里用 todos
比 todoes
更舒服一些,所以我们再使用另一个 @RequestMapping("/todos")
来自定义路径。这个 Controller
中的其它方法比较简单,就是利用repository中的方法去增删改查即可。
package dev.local.todo;
import org.bson.types.ObjectId;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
import java.util.List;
@RestController
@RequestMapping("/todos")
public class TodoController {
private TodoRepository repository;
@Autowired
public TodoController(TodoRepository repository){
this.repository = repository;
}
@RequestMapping(method = RequestMethod.GET)
public List<Todo> getAllTodos(@RequestHeader(value = "userId") String userId) {
return repository.findByUserId(new ObjectId(userId));
}
@RequestMapping(method = RequestMethod.POST)
Todo addTodo(@RequestBody Todo addedTodo) {
return repository.insert(addedTodo);
}
@RequestMapping(value = "/{id}", method = RequestMethod.GET)
public Todo getTodo(@PathVariable String id) {
return repository.findOne(id);
}
@RequestMapping(value = "/{id}", method = RequestMethod.PUT)
Todo updateTodo(@PathVariable String id, @RequestBody Todo updatedTodo) {
updatedTodo.setId(id);
return repository.save(updatedTodo);
}
@RequestMapping(value = "/{id}", method = RequestMethod.DELETE)
Todo removeTodo(@PathVariable String id) {
Todo deletedTodo = repository.findOne(id);
repository.delete(id);
return deletedTodo;
}
}
上面的代码中需要再说明几个要点:
- 为什么在类上标记
@RequestMapping("/todos")
后在每个方法上还需要添加@RequestMapping
?类上面定义的@RequestMapping
的参数会默认应用于所有方法,但如果我们发现某个方法需要有自己的特殊值时,就需要定义这个方法的映射参数。比如上面例子中addTodo
,路径也是todos
,但要求 Request的方法是POST
,所以我们给出了@RequestMapping(method = RequestMethod.POST)
。但getTodo
方法的路径应该是todos/:id
,这时我们要给出@RequestMapping(value = "/{id}", method = RequestMethod.GET)
- 这些方法接受的参数也使用了各种修饰符,
@PathVariable
表示参数是从路径中得来的,而@RequestBody
表示参数应该从 Http Request的body
中解析,类似的@RequestHeader
表示参数是 Http Request的Header中定义的。
在可以测试之前,我们还需要使用 @Repository
来标记 TodoRepository
,以便于Spring可以在依赖注入时可以找到这个类。
package dev.local.todo;
import org.bson.types.ObjectId;
import org.springframework.data.mongodb.repository.MongoRepository;
import org.springframework.stereotype.Repository;
import java.util.List;
/**
* Created by wangpeng on 2017/1/26.
*/
@Repository
public interface TodoRepository extends MongoRepository<Todo, String>{
List<Todo> findByUserId(ObjectId userId);
}
接下来就可以用PostMan做一下测试:
Angular从零到一
网友评论