Spring Boot实战:Restful API的构建

在现在的开发流程中,为了最大程度实现前后端的分离,通常后端接口只提供数据接口,由前端通过Ajax请求从后端获取数据并进行渲染再展示给用户。我们用的最多的方式就是后端会返回给前端一个JSON字符串,前端解析JSON字符串生成JavaScript的对象,然后再做处理。本文就来演示一下Spring boot如何实现这种模式,本文重点会讲解如何设计一个Restful的API,并通过Spring boot来实现相关的API。不过,为了大家更好的了解Restful风格的API,我们先设计一个传统的数据返回接口,这样大家可以对比着来理解。

一、非Restful接口的支持

  我们这里以文章列表为例,实现一个返回文章列表的接口,代码如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

@Controller

@RequestMapping("/article")

public class ArticleController {

 

    @Autowired

    private ArticleService articleService;

 

    @RequestMapping("/list.json")

    @ResponseBody

    public List<Article> listArticles(String title, Integer pageSize, Integer pageNum) {

        if (pageSize == null) {

            pageSize = 10;

        }

        if (pageNum == null) {

            pageNum = 1;

        }

        int offset = (pageNum - 1) * pageSize;

        return articleService.getArticles(title, 1L, offset, pageSize);

    }

}

  这个ArticleService的实现很简单,就是简单的封装了ArticleMapper的操作,ArticleMapper的内容大家可以参考上一篇的文章,ArticleService的实现类如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

@Service

public class ArticleServiceImpl implements ArticleService {

 

    @Autowired

    private ArticleMapper articleMapper;

 

    @Override

    public Long saveArticle(@RequestBody Article article) {

        return articleMapper.insertArticle(article);

    }

 

    @Override

    public List<Article> getArticles(String title,Long userId,int offset,int pageSize) {

        Article article = new Article();

        article.setTitle(title);

        article.setUserId(userId);

        return articleMapper.queryArticlesByPage(article,offset,pageSize);

    }

 

    @Override

    public Article getById(Long id) {

        return articleMapper.queryById(id);

    }

 

    @Override

    public void updateArticle(Article article) {

        article.setUpdateTime(new Date());

        articleMapper.updateArticleById(article);

    }

}

  运行Application.java这个类,然后访问:http://locahost:8080/article/list.json,就可以看到如下的结果:

  ArticleServiceImpl这个类是一个很普通的类,只有一个Spring的注解@Service,标识为一个bean以便于通过Spring IoC容器来管理。我们再来看看ArticleController这个类,其实用过Spring MVC的人应该都熟悉这几个注解,这里简单解释一下:

  @Controller 标识一个类为控制器。

  @RequestMapping URL的映射。

  @ResponseBody 返回结果转换为JSON字符串。

  @RequestBody 表示接收JSON格式字符串参数。

  通过这个三个注解,我们就能轻松的实现通过URL给前端返回JSON格式数据的功能。不过大家肯定有点疑惑,这不都是Spring MVC的东西吗?跟Spring boot有什么关系?其实Spring boot的作用就是为我们省去了配置的过程,其他功能确实都是Spring与Spring MVC来为我们提供的,大家应该记得Spring boot通过各种starter来为我们提供自动配置的服务,我们的工程里面之前引入过这个依赖:

1

2

3

4

<dependency>

      <groupId>org.springframework.boot</groupId>

      <artifactId>spring-boot-starter-web</artifactId>

</dependency>

  这个是所有Spring boot的web工程都需要引入的jar包,也就是说只要是Spring boot的web的工程,都默认支持上述的功能。这里我们进一步发现,通过Spring boot来开发web工程,确实为我们省了许多配置的工作。

 

二、Restful API设计

  好了,我们现在再来看看如何实现Restful API。实际上Restful本身不是一项什么高深的技术,而只是一种编程风格,或者说是一种设计风格。在传统的http接口设计中,我们一般只使用了get和post两个方法,然后用我们自己定义的词汇来表示不同的操作,比如上面查询文章的接口,我们定义了article/list.json来表示查询文章列表,可以通过get或者post方法来访问。而Restful API的设计则通过HTTP的方法来表示CRUD相关的操作。因此,除了get和post方法外,还会用到其他的HTTP方法,如PUT、DELETE、HEAD等,通过不同的HTTP方法来表示不同含义的操作。下面是我设计的一组对文章的增删改查的Restful API:

接口URLHTTP方法接口说明
 /article POST 保存文章
 /article/{id} GET 查询文章列表
 /article/{id} DELETE 删除文章
 /article/{id} PUT 更新文章信息

   这里可以看出,URL仅仅是标识资源的路劲,而具体的行为由HTTP方法来指定。

 

三、Restful API实现

  现在我们再来看看如何实现上面的接口,其他就不多说,直接看代码:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

@RestController

@RequestMapping("/rest")

public class ArticleRestController {

 

    @Autowired

    private ArticleService articleService;

 

    @RequestMapping(value = "/article", method = POST, produces = "application/json")

    public WebResponse<Map<String, Object>> saveArticle(@RequestBody Article article) {

        article.setUserId(1L);

        articleService.saveArticle(article);

        Map<String, Object> ret = new HashMap<>();

        ret.put("id", article.getId());

        WebResponse<Map<String, Object>> response = WebResponse.getSuccessResponse(ret);

        return response;

    }

 

    @RequestMapping(value = "/article/{id}", method = DELETE, produces = "application/json")

    public WebResponse<?> deleteArticle(@PathVariable Long id) {

        Article article = articleService.getById(id);

        article.setStatus(-1);

        articleService.updateArticle(article);

        WebResponse<Object> response = WebResponse.getSuccessResponse(null);

        return response;

    }

 

    @RequestMapping(value = "/article/{id}", method = PUT, produces = "application/json")

    public WebResponse<Object> updateArticle(@PathVariable Long id, @RequestBody Article article) {

        article.setId(id);

        articleService.updateArticle(article);

        WebResponse<Object> response = WebResponse.getSuccessResponse(null);

        return response;

    }

 

    @RequestMapping(value = "/article/{id}", method = GET, produces = "application/json")

    public WebResponse<Article> getArticle(@PathVariable Long id) {

        Article article = articleService.getById(id);

        WebResponse<Article> response = WebResponse.getSuccessResponse(article);

        return response;

    }

}

  我们再来分析一下这段代码,这段代码和之前代码的区别在于:

  (1)我们使用的是@RestController这个注解,而不是@Controller,不过这个注解同样不是Spring boot提供的,而是Spring MVC4中的提供的注解,表示一个支持Restful的控制器。

  (2)这个类中有三个URL映射是相同的,即都是/article/{id},这在@Controller标识的类中是不允许出现的。这里的可以通过method来进行区分,produces的作用是表示返回结果的类型是JSON。

  (3)@PathVariable这个注解,也是Spring MVC提供的,其作用是表示该变量的值是从访问路径中获取。

  所以看来看去,这个代码还是跟Spring boot没太多的关系,Spring boot也仅仅是提供自动配置的功能,这也是Spring boot用起来很舒服的一个很重要的原因,因为它的侵入性非常非常小,你基本感觉不到它的存在。

 

四、测试

  代码写完了,怎么测试?除了GET的方法外,都不能直接通过浏览器来访问,当然,我们可以直接通过postman来发送各种http请求。不过我还是比较支持通过单元测试类来测试各个方法。这里我们就通过Junit来测试各个方法:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

@RunWith(SpringJUnit4ClassRunner.class)

@SpringBootTest(classes = Application.class)

public class ArticleControllerTest {

 

    @Autowired

    private ArticleRestController restController;

 

    private MockMvc mvc;

 

    @Before

    public void setUp() throws Exception {

        mvc = MockMvcBuilders.standaloneSetup(restController).build();

    }

 

    @Test

    public void testAddArticle() throws Exception {

        Article article = new Article();

        article.setTitle("测试文章000000");

        article.setType(1);

        article.setStatus(2);

        article.setSummary("这是一篇测试文章");

        Gson gosn = new Gson();

        RequestBuilder builder = MockMvcRequestBuilders

                .post("/rest/article")

                .accept(MediaType.APPLICATION_JSON)

                .contentType(MediaType.APPLICATION_JSON_UTF8)

                .content(gosn.toJson(article));

 

        MvcResult result = mvc.perform(builder).andReturn();

        System.out.println(result.getResponse().getContentAsString());

    }

 

    @Test

    public void testUpdateArticle() throws Exception {

        Article article = new Article();

        article.setTitle("更新测试文章");

        article.setType(1);

        article.setStatus(2);

        article.setSummary("这是一篇更新测试文章");

        Gson gosn = new Gson();

        RequestBuilder builder = MockMvcRequestBuilders

                .put("/rest/article/1")

                .accept(MediaType.APPLICATION_JSON)

                .contentType(MediaType.APPLICATION_JSON_UTF8)

                .content(gosn.toJson(article));

        MvcResult result = mvc.perform(builder).andReturn();

    }

 

    @Test

    public void testQueryArticle() throws Exception {

        RequestBuilder builder = MockMvcRequestBuilders

                .get("/rest/article/1")

                .accept(MediaType.APPLICATION_JSON)

                .contentType(MediaType.APPLICATION_JSON_UTF8);

        MvcResult result = mvc.perform(builder).andReturn();

        System.out.println(result.getResponse().getContentAsString());

    }

 

    @Test

    public void testDeleteArticle() throws Exception {

        RequestBuilder builder = MockMvcRequestBuilders

                .delete("/rest/article/1")

                .accept(MediaType.APPLICATION_JSON)

                .contentType(MediaType.APPLICATION_JSON_UTF8);

        MvcResult result = mvc.perform(builder).andReturn();

    }

}

  执行结果这里就不给大家贴了,大家有兴趣的话可以自己实验一下。整个类要说明的点还是很少,主要这些东西都与Spring boot没关系,支持这些操作的原因还是上一篇文章中提到的引入对应的starter:

1

2

3

4

5

<dependency>

      <groupId>org.springframework.boot</groupId>

      <artifactId>spring-boot-starter-test</artifactId>

      <scope>test</scope>

</dependency>

  因为要执行HTTP请求,所以这里使用了MockMvc,ArticleRestController通过注入的方式实例化,不能直接new,否则ArticleRestController就不能通过Spring IoC容器来管理,因而其依赖的其他类也无法正常注入。通过MockMvc我们就可以轻松的实现HTTP的DELETE/PUT/POST等方法了。

五、总结

  本文讲解了如果通过Spring boot来实现Restful的API,其实大部分东西都是Spring和Spring MVC提供的,Spring boot只是提供自动配置的功能。但是,正是这种自动配置,为我们减少了很多的开发和维护工作,使我们能更加简单、高效的实现一个web工程,从而让我们能够更加专注于业务本身的开发,而不需要去关心框架的东西。这篇文章中我们提到了可以通过postman和junit的方式来访问Restful 接口,下篇文章我们会介绍另外一种方式来访问,有兴趣的可以继续关注一下。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
课程的实战源码是我在 GitHub 上开源项目 spring-boot-projects 的其一个项目代码,目前已有 2300 多个 star,项目截图如下: 由于项目比较受大家欢迎,因此心就出现了将它做成教学视频的想法,也就是你现在了解的这个课程《SpringBoot入门及前后端分离项目开发》,本课程是一个 Spring Boot 技术栈的实战课程,课程共分为 3 大部分,前面两个部分为基础环境准备和相关概念介绍,第三个部分是 Spring Boot 项目实践开发。Spring Boot 介绍、前后端分离、API 规范等内容旨在让读者更加熟悉 SpringBoot 及企业开发需要注意的事项并具有使用 SpringBoot 技术进行基本功能开发的能力;这最后的项目实战为课程的主要部分,我会带着大家实际的开发一个前后端分离的 Spring Boot 实践项目,让大家实际操作并从无到有开发一个线上项目,并学习到一定的开发经验以及其的开发技巧,旨在让读者具有将 Spring Boot 真正应用于项目开发的能力; 以下为实践项目的页面和功能展示,分别为:登录页面 列表页面(分页功能) 图片上传功能 富文本编辑器整合使用 实践项目的主要功能和页面就是这些,通过项目展示大家也能够感受到,在实际应用开发的高频次功能都已经实现,稍加修改就可以运用到企业开发,整个项目的开发模式为前后端分离的模式,即 Spring Boot 提供后端接口,前端页面通过 Ajax 异步调用接口的方式与后端服务器进行交互并将数据填充至页面,这也是目前企业开发比较重用的开发模式,希望大家能够了解并且能够实际的上手开发。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值