Spring Boot(三)之找回熟悉的Controller,Service
找回熟悉的Controller,Service Controller哪兒去了?
對于很多習(xí)慣了Spring開發(fā)的同學(xué)來講,Controller,Service,DAO 這些套路突然間都沒了會(huì)有不適感。其實(shí)呢,這些東西還在,只不過對于較簡單的情景下,這些都變成了系統(tǒng)背后幫你做的事情。這一小節(jié)我們就先來看看如何將Controller再召喚回來。召喚回來的好處有哪些呢?首先我們可以自定義API URL的路徑,其次可以對參數(shù)和返回的json結(jié)構(gòu)做一定的處理。
如果要讓 TodoController 可以和 TodoRepository 配合工作的話,我們當(dāng)然需要在 TodoController 中需要引用 TodoRepository。
public class TodoController {
@Autowired
private TodoRepository repository;
//省略其它部分
}
@Autowired 這個(gè)修飾符是用于做依賴性注入的,上面的用法叫做 field injection,直接做類成員的注入。但Spring現(xiàn)在鼓勵(lì)用構(gòu)造函數(shù)來做注入,所以,我們來看看構(gòu)造函數(shù)的注入方法:
public class TodoController {
private TodoRepository repository;
@Autowired
public TodoController(TodoRepository repository){
this.repository = repository;
}
//省略其它部分
}
當(dāng)然我們?yōu)榱丝梢宰孲pring知道這是一個(gè)支持REST API的 Controller ,還是需要標(biāo)記其為 @RestController。由于默認(rèn)的路徑映射會(huì)在資源根用復(fù)數(shù)形式,由于todo是輔音后的o結(jié)尾,按英語習(xí)慣,會(huì)映射成 todoes。但這里用 todos 比 todoes 更舒服一些,所以我們再使用另一個(gè) @RequestMapping("/todos") 來自定義路徑。這個(gè) 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;
}
}
上面的代碼中需要再說明幾個(gè)要點(diǎn):
為什么在類上標(biāo)記 @RequestMapping("/todos") 后在每個(gè)方法上還需要添加 @RequestMapping?類上面定義的 @RequestMapping 的參數(shù)會(huì)默認(rèn)應(yīng)用于所有方法,但如果我們發(fā)現(xiàn)某個(gè)方法需要有自己的特殊值時(shí),就需要定義這個(gè)方法的映射參數(shù)。比如上面例子中 addTodo,路徑也是 todos,但要求 Request的方法是 POST,所以我們給出了 @RequestMapping(method = RequestMethod.POST)。但 getTodo 方法的路徑應(yīng)該是 todos/:id,這時(shí)我們要給出 @RequestMapping(value = "/{id}", method = RequestMethod.GET)
這些方法接受的參數(shù)也使用了各種修飾符,@PathVariable 表示參數(shù)是從路徑中得來的,而 @RequestBody 表示參數(shù)應(yīng)該從 Http Request的body 中解析,類似的 @RequestHeader 表示參數(shù)是 Http Request的Header中定義的。
在可以測試之前,我們還需要使用 @Repository 來標(biāo)記 TodoRepository,以便于Spring可以在依賴注入時(shí)可以找到這個(gè)類。
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
做一下測試:

測試一下
Controller Service呢?在哪里?
熟悉Spring的童鞋肯定會(huì)問,我們剛才的做法等于直接是Controller訪問Data了,隔離不夠啊。其實(shí)我覺得有很多時(shí)候,這種簡單設(shè)計(jì)是挺好的,因?yàn)闃I(yè)務(wù)還沒有到達(dá)那步,過于復(fù)雜的設(shè)計(jì)其實(shí)沒啥太大意義。但這里我們還是一步步來實(shí)踐一下,找回大家熟悉的感覺。
回到原來的熟悉模式再簡單不過的,新建一個(gè) TodoService 接口,定義一下目前的增刪改查幾個(gè)操作:
public interface TodoService {
Todo addTodo(Todo todo);
Todo deleteTodo(String id);
List<Todo> findAll(String userId);
Todo findById(String id);
Todo update(Todo todo);
}
為預(yù)防我們以后使用 MySQL 等潛在的 “可擴(kuò)展性”,我們給這個(gè)接口的實(shí)現(xiàn)命名為 MongoTodoServiceImpl,然后把 Controller 中的大部分代碼拿過來改改就行了。當(dāng)然為了系統(tǒng)可以找到這個(gè)依賴并注入需要的類中,我們標(biāo)記它為 @Service
@Service
public class MongoTodoServiceImpl implements TodoService{
private final TodoRepository repository;
@Autowired
MongoTodoServiceImpl(TodoRepository repository) {
this.repository = repository;
}
@Override
public Todo addTodo(Todo todo) {
return repository.insert(todo);
}
@Override
public Todo deleteTodo(String id) {
Todo deletedTodo = repository.findOne(id);
repository.delete(id);
return deletedTodo;
}
@Override
public List<Todo> findAll(String userId) {
return repository.findByUserId(new ObjectId(userId));
}
@Override
public Todo findById(String id) {
return repository.findOne(id);
}
@Override
public Todo update(Todo todo) {
repository.save(todo);
return todo;
}
}
最后把Controller中的所有方法改為使用Service的簡單調(diào)用就大功告成了。
public class TodoController {
private TodoService service;
@Autowired
public TodoController(TodoService service){
this.service = service;
}
@RequestMapping(method = RequestMethod.GET)
public List<Todo> getAllTodos(@RequestHeader(value = "userId") String userId) {
return service.findAll(userId);
}
@RequestMapping(method = RequestMethod.POST)
Todo addTodo(@RequestBody Todo addedTodo) {
return service.addTodo(addedTodo);
}
@RequestMapping(value = "/{id}", method = RequestMethod.GET)
public Todo getTodo(@PathVariable String id) {
return service.findById(id);
}
@RequestMapping(value = "/{id}", method = RequestMethod.PUT)
Todo updateTodo(@PathVariable String id, @RequestBody Todo updatedTodo) {
updatedTodo.setId(id);
return service.update(updatedTodo);
}
@RequestMapping(value = "/{id}", method = RequestMethod.DELETE)
Todo removeTodo(@PathVariable String id) {
return service.deleteTodo(id);
}
}
說實(shí)話如果每個(gè)簡單類都這么寫,我深深地趕腳背離了Spring Boot的意圖,雖然你能舉出1000個(gè)理由這么做有好處。類似的,DAO或DTO要寫起來也很簡單,但我還是建議在業(yè)務(wù)沒有復(fù)雜之前還是享受Spring Boot帶給我們的便利吧。
以上所述是小編給大家介紹的Spring Boot(三)之找回熟悉的Controller,Service,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
Java實(shí)現(xiàn)班級(jí)管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)班級(jí)管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02
Mybatis之collection標(biāo)簽中javaType和ofType屬性的區(qū)別說明
這篇文章主要介紹了Mybatis之collection標(biāo)簽中javaType和ofType屬性的區(qū)別說明,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12
Maven中dependencyManagement管理項(xiàng)目依賴項(xiàng)
在開發(fā)?Java?項(xiàng)目時(shí),管理和協(xié)調(diào)依賴項(xiàng)的版本號(hào)是一項(xiàng)重要而繁瑣的任務(wù),本文主要介紹了Maven中dependencyManagement管理項(xiàng)目依賴項(xiàng),具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01

