本文我们将使用Spring MVC 4实现 CRUD Restful WebService , 通过RestTemplate写一个 REST 客户端,定义这些服务. 我们也可以通过外部的一些客户端来测试这些服务。

简短 & 快速介绍REST

REST表示 Representational State Transfer(表示性状态转换).

它是可以用来设计web services的框架,可以被不同的客户端调用。

核心思想是:使用简单的HTTP协议来实现调用,而不是CORBA, RPC 或者 SOAP等负责的机制。

在Rest 基础设计中,资源使用以下动词进行操作。

  • 创建资源 : 使用 HTTP POST
  • 获取资源 : 使用 HTTP GET
  • 更新资源 : 使用 HTTP PUT
  • 删除资源 : 使用 HTTP DELETE

也意味着,你作为Rest 服务开发者或者客户,应该遵循以上的标准。

尽管没有限制必须返回的类型,但是一般基于Web services的Rest返回JSON或者XML作为响应。

客户端可以指定(使用HTTP Accept header)他们想要的资源类型吗,服务器返回需要的资源。

指明资源的Content-Type。如果想详细的理解 restful可以参考这里:StackOverflow link

基于Rest的Controller(控制器)

我们的 REST API :

  • GET 方式请求 /api/user/ 返回用户列表
  • GET 方式请求 /api/user/1返回id为1的用户
  • POST 方式请求 /api/user/ 通过user对象的JSON 参数创建新的user对象
  • PUT 方式请求 /api/user/3 更新id为3的发送json格式的用户对象
  • DELETE 方式请求/api/user/4删除 ID为 4的user对象
  • DELETE 方式请求/api/user/删除所有user
  1. package com.websystique.springmvc.controller;
  2. import java.util.List;
  3. import org.springframework.beans.factory.annotation.Autowired;
  4. import org.springframework.http.HttpHeaders;
  5. import org.springframework.http.HttpStatus;
  6. import org.springframework.http.MediaType;
  7. import org.springframework.http.ResponseEntity;
  8. import org.springframework.web.bind.annotation.PathVariable;
  9. import org.springframework.web.bind.annotation.RequestBody;
  10. import org.springframework.web.bind.annotation.RequestMapping;
  11. import org.springframework.web.bind.annotation.RequestMethod;
  12. import org.springframework.web.bind.annotation.RestController;
  13. import org.springframework.web.util.UriComponentsBuilder;
  14. import com.websystique.springmvc.model.User;
  15. import com.websystique.springmvc.service.UserService;
  16. @RestController
  17. public class HelloWorldRestController {
  18. @Autowired
  19. UserService userService;  //Service which will do all data retrieval/manipulation work
  20. //-------------------Retrieve All Users--------------------------------------------------------
  21. @RequestMapping(value = "/user/", method = RequestMethod.GET)
  22. public ResponseEntity<List<User>> listAllUsers() {
  23. List<User> users = userService.findAllUsers();
  24. if(users.isEmpty()){
  25. return new ResponseEntity<List<User>>(HttpStatus.NO_CONTENT);//You many decide to return HttpStatus.NOT_FOUND
  26. }
  27. return new ResponseEntity<List<User>>(users, HttpStatus.OK);
  28. }
  29. //-------------------Retrieve Single User--------------------------------------------------------
  30. @RequestMapping(value = "/user/{id}", method = RequestMethod.GET, produces = MediaType.APPLICATION_JSON_VALUE)
  31. public ResponseEntity<User> getUser(@PathVariable("id") long id) {
  32. System.out.println("Fetching User with id " + id);
  33. User user = userService.findById(id);
  34. if (user == null) {
  35. System.out.println("User with id " + id + " not found");
  36. return new ResponseEntity<User>(HttpStatus.NOT_FOUND);
  37. }
  38. return new ResponseEntity<User>(user, HttpStatus.OK);
  39. }
  40. //-------------------Create a User--------------------------------------------------------
  41. @RequestMapping(value = "/user/", method = RequestMethod.POST)
  42. public ResponseEntity<Void> createUser(@RequestBody User user,    UriComponentsBuilder ucBuilder) {
  43. System.out.println("Creating User " + user.getName());
  44. if (userService.isUserExist(user)) {
  45. System.out.println("A User with name " + user.getName() + " already exist");
  46. return new ResponseEntity<Void>(HttpStatus.CONFLICT);
  47. }
  48. userService.saveUser(user);
  49. HttpHeaders headers = new HttpHeaders();
  50. headers.setLocation(ucBuilder.path("/user/{id}").buildAndExpand(user.getId()).toUri());
  51. return new ResponseEntity<Void>(headers, HttpStatus.CREATED);
  52. }
  53. //------------------- Update a User --------------------------------------------------------
  54. @RequestMapping(value = "/user/{id}", method = RequestMethod.PUT)
  55. public ResponseEntity<User> updateUser(@PathVariable("id") long id, @RequestBody User user) {
  56. System.out.println("Updating User " + id);
  57. User currentUser = userService.findById(id);
  58. if (currentUser==null) {
  59. System.out.println("User with id " + id + " not found");
  60. return new ResponseEntity<User>(HttpStatus.NOT_FOUND);
  61. }
  62. currentUser.setName(user.getName());
  63. currentUser.setAge(user.getAge());
  64. currentUser.setSalary(user.getSalary());
  65. userService.updateUser(currentUser);
  66. return new ResponseEntity<User>(currentUser, HttpStatus.OK);
  67. }
  68. //------------------- Delete a User --------------------------------------------------------
  69. @RequestMapping(value = "/user/{id}", method = RequestMethod.DELETE)
  70. public ResponseEntity<User> deleteUser(@PathVariable("id") long id) {
  71. System.out.println("Fetching & Deleting User with id " + id);
  72. User user = userService.findById(id);
  73. if (user == null) {
  74. System.out.println("Unable to delete. User with id " + id + " not found");
  75. return new ResponseEntity<User>(HttpStatus.NOT_FOUND);
  76. }
  77. userService.deleteUserById(id);
  78. return new ResponseEntity<User>(HttpStatus.NO_CONTENT);
  79. }
  80. //------------------- Delete All Users --------------------------------------------------------
  81. @RequestMapping(value = "/user/", method = RequestMethod.DELETE)
  82. public ResponseEntity<User> deleteAllUsers() {
  83. System.out.println("Deleting All Users");
  84. userService.deleteAllUsers();
  85. return new ResponseEntity<User>(HttpStatus.NO_CONTENT);
  86. }
  87. }

详解:

@RestController :首先我们使用的是Spring 4的新注解 @RestController注解.

此注解避免了每个方法都要加上@ResponseBody注解。也就是说@RestController 自己戴上了 @ResponseBody注解,看以看作是

@Controller 和 @ResponseBody的结合体。

@RequestBody : 如果方法参数被 @RequestBody注解,Spring将绑定HTTP请求体到那个参数上。如果那样做,Spring将根据请求中的ACCEPT或者 Content-Type header(私下)使用 HTTP Message converters 来将http请求体转化为domain对象。

@ResponseBody : 如果方法加上了@ResponseBody注解,Spring返回值到响应体。如果这样做的话,Spring将根据请求中的 Content-Type header(私下)使用 HTTP Message converters 来将domain对象转换为响应体。

ResponseEntity 是一个真实数据.它代表了整个 HTTP 响应(response). 它的好处是你可以控制任何对象放到它内部。

你可以指定状态码、头信息和响应体。它包含你想要构建HTTP Response 的信息。

@PathVariable 此注解意味着一个方法参数应该绑定到一个url模板变量[在'{}'里的一个]中

一般来说你,要实现REST API in Spring 4 需要了解@RestController , @RequestBody, ResponseEntity 和 @PathVariable 这些注解 .另外, spring 也提供了一些支持类帮助你实现一些可定制化的东西。

MediaType : 带着 @RequestMapping 注解,通过特殊的控制器方法你可以额外指定,MediaType来生产或者消耗。

发布和测试此API

 http://localhost:8080/Spring4MVCCRUDRestService.
 
想要测试此API,我将使用POSTMAN这个外部客户端,接下来我们也将写我们自己的客户端。
1. 获取所有用户
 
打开 POSTMAN工具,选择请求类型为GET,指明uri
 
注意:我们没有指明任何HTTP头。点击 发送,将接收到所有用户的列表
 
 
也要注意HTTP 200 响应。
 
你也许好奇为什么此响应通过JSON字符串发送的,在响应里的Content-Type 头说明了这个。
因为我们添加了JACKSON
  1. <dependency>
  2. <groupId>com.fasterxml.jackson.core</groupId>
  3. <artifactId>jackson-databind</artifactId>
  4. <version>2.5.3</version>
  5. </dependency

因为Spring在类路径发现了这个库,它调用了内置的MappingJackson2HttpMessageConverter 转换器将响应(对象集合)转换为JSON格式。

 Spring内置转换器的好处是,大部分情况下只要把库放到类路径,即可完成转换。当然了有时候我们也需要
采用我们的API。比如,如果我们像也提供XML格式的话,我们需要对User类加上JAXB注解。
 
2. 获取单个用户
 
GET方式 指定/user/1 
 
现在试着发送一个带有错误识别码的GET请求,将收到一个HTTP 404
 
3.创建一个 User
选择POST方法,指明uri /user/ 指明POSTMAN Body选项卡,选择application/json类型
 你要注意POSTMAN自动添加了Content-Type 头信息
 
记住: Accept header包含client能给识别的类型。 Content-Type header表示数据的实际类型。
 
点击发送以后 将收到 HTTP 200 没有响应体(api里面没有在响应体发送任何东西)
 
你可以查询新创建的用户
 
这是实现REST的普通实现方式。但是也没人阻止你为POST或者PUT方式响应体里发送内容。但是这还是REST 的API?值得怀疑。
不管怎样,我们试着创建同一个用户时,你将获得HTTP冲突的响应。
 
4.更新用户
发送一个HTTP PUT 请求来更新用户。
注意:这次我们接收到了响应体。这是因为在控制器的方法实现里我们发送了数据。再次强调,有的人也许不在响应体里面发送更新的详情,只发送位置头(和创建用户一样)。
5.删除用户
6 删除所有用户
7.删除用户后验证

根据RestTemplate 写REST Client

 
Postman是测试Rest Api的超好用的工具,但是如果你想完整的消化REST,可以尝试自己写一个。
最出名的Htpp 客户端是HttpClient( Apache HttpComponents )。
但是用它来访问REST service则相对少见。
Spring的 RestTemplate随之出现。RestTemplate 提供了高级方法,来响应者6种主要的HTTP方法。
 
HTTP 方法和对应的 RestTemplate方法:
 

  • HTTP GET : getForObject, getForEntity
  • HTTP PUT : put(String url, Object request, String…​urlVariables)
  • HTTP DELETE : delete
  • HTTP POST : postForLocation(String url, Object request, String…​ urlVariables), postForObject(String url, Object request, ClassresponseType, String…​ uriVariables)
  • HTTP HEAD : headForHeaders(String url, String…​ urlVariables)
  • HTTP OPTIONS : optionsForAllow(String url, String…​ urlVariables)
  • HTTP PATCH and others : exchange execute

定义 Rest client , 定义REST services

  1. package com.websystique.springmvc;
  2. import java.net.URI;
  3. import java.util.LinkedHashMap;
  4. import java.util.List;
  5. import org.springframework.web.client.RestTemplate;
  6. import com.websystique.springmvc.model.User;
  7. public class SpringRestTestClient {
  8. public static final String REST_SERVICE_URI = "http://localhost:8080/Spring4MVCCRUDRestService";
  9. /* GET */
  10. @SuppressWarnings("unchecked")
  11. private static void listAllUsers(){
  12. System.out.println("Testing listAllUsers API-----------");
  13. RestTemplate restTemplate = new RestTemplate();
  14. List<LinkedHashMap<String, Object>> usersMap = restTemplate.getForObject(REST_SERVICE_URI+"/user/", List.class);
  15. if(usersMap!=null){
  16. for(LinkedHashMap<String, Object> map : usersMap){
  17. System.out.println("User : id="+map.get("id")+", Name="+map.get("name")+", Age="+map.get("age")+", Salary="+map.get("salary"));;
  18. }
  19. }else{
  20. System.out.println("No user exist----------");
  21. }
  22. }
  23. /* GET */
  24. private static void getUser(){
  25. System.out.println("Testing getUser API----------");
  26. RestTemplate restTemplate = new RestTemplate();
  27. User user = restTemplate.getForObject(REST_SERVICE_URI+"/user/1", User.class);
  28. System.out.println(user);
  29. }
  30. /* POST */
  31. private static void createUser() {
  32. System.out.println("Testing create User API----------");
  33. RestTemplate restTemplate = new RestTemplate();
  34. User user = new User(0,"Sarah",51,134);
  35. URI uri = restTemplate.postForLocation(REST_SERVICE_URI+"/user/", user, User.class);
  36. System.out.println("Location : "+uri.toASCIIString());
  37. }
  38. /* PUT */
  39. private static void updateUser() {
  40. System.out.println("Testing update User API----------");
  41. RestTemplate restTemplate = new RestTemplate();
  42. User user  = new User(1,"Tomy",33, 70000);
  43. restTemplate.put(REST_SERVICE_URI+"/user/1", user);
  44. System.out.println(user);
  45. }
  46. /* DELETE */
  47. private static void deleteUser() {
  48. System.out.println("Testing delete User API----------");
  49. RestTemplate restTemplate = new RestTemplate();
  50. restTemplate.delete(REST_SERVICE_URI+"/user/3");
  51. }
  52. /* DELETE */
  53. private static void deleteAllUsers() {
  54. System.out.println("Testing all delete Users API----------");
  55. RestTemplate restTemplate = new RestTemplate();
  56. restTemplate.delete(REST_SERVICE_URI+"/user/");
  57. }
  58. public static void main(String args[]){
  59. listAllUsers();
  60. getUser();
  61. createUser();
  62. listAllUsers();
  63. updateUser();
  64. listAllUsers();
  65. deleteUser();
  66. listAllUsers();
  67. deleteAllUsers();
  68. listAllUsers();
  69. }
  70. }

重启服务器,运行上面的程序。

下面是输出:
  1. Testing listAllUsers API-----------
  2. User : id=1, Name=Sam, Age=30, Salary=70000.0
  3. User : id=2, Name=Tom, Age=40, Salary=50000.0
  4. User : id=3, Name=Jerome, Age=45, Salary=30000.0
  5. User : id=4, Name=Silvia, Age=50, Salary=40000.0
  6. Testing getUser API----------
  7. User [id=1, name=Sam, age=30, salary=70000.0]
  8. Testing create User API----------
  9. Location : http://localhost:8080/Spring4MVCCRUDRestService/user/5
  10. Testing listAllUsers API-----------
  11. User : id=1, Name=Sam, Age=30, Salary=70000.0
  12. User : id=2, Name=Tom, Age=40, Salary=50000.0
  13. User : id=3, Name=Jerome, Age=45, Salary=30000.0
  14. User : id=4, Name=Silvia, Age=50, Salary=40000.0
  15. User : id=5, Name=Sarah, Age=51, Salary=134.0
  16. Testing update User API----------
  17. User [id=1, name=Tomy, age=33, salary=70000.0]
  18. Testing listAllUsers API-----------
  19. User : id=1, Name=Tomy, Age=33, Salary=70000.0
  20. User : id=2, Name=Tom, Age=40, Salary=50000.0
  21. User : id=3, Name=Jerome, Age=45, Salary=30000.0
  22. User : id=4, Name=Silvia, Age=50, Salary=40000.0
  23. User : id=5, Name=Sarah, Age=51, Salary=134.0
  24. Testing delete User API----------
  25. Testing listAllUsers API-----------
  26. User : id=1, Name=Tomy, Age=33, Salary=70000.0
  27. User : id=2, Name=Tom, Age=40, Salary=50000.0
  28. User : id=4, Name=Silvia, Age=50, Salary=40000.0
  29. User : id=5, Name=Sarah, Age=51, Salary=134.0
  30. Testing all delete Users API----------
  31. Testing listAllUsers API-----------
  32. No user exist----------

完整的例子

 

更新pom.xml添加项目依赖

  1. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  2. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
  3. <modelVersion>4.0.0</modelVersion>
  4. <groupId>com.websystique.springmvc</groupId>
  5. <artifactId>Spring4MVCCRUDRestService</artifactId>
  6. <packaging>war</packaging>
  7. <version>1.0.0</version>
  8. <name>Spring4MVCCRUDRestService Maven Webapp</name>
  9. <properties>
  10. <springframework.version>4.2.0.RELEASE</springframework.version>
  11. <jackson.version>2.5.3</jackson.version>
  12. </properties>
  13. <dependencies>
  14. <dependency>
  15. <groupId>org.springframework</groupId>
  16. <artifactId>spring-webmvc</artifactId>
  17. <version>${springframework.version}</version>
  18. </dependency>
  19. <dependency>
  20. <groupId>org.springframework</groupId>
  21. <artifactId>spring-tx</artifactId>
  22. <version>${springframework.version}</version>
  23. </dependency>
  24. <dependency>
  25. <groupId>com.fasterxml.jackson.core</groupId>
  26. <artifactId>jackson-databind</artifactId>
  27. <version>${jackson.version}</version>
  28. </dependency>
  29. <dependency>
  30. <groupId>javax.servlet</groupId>
  31. <artifactId>javax.servlet-api</artifactId>
  32. <version>3.1.0</version>
  33. </dependency>
  34. </dependencies>
  35. <build>
  36. <pluginManagement>
  37. <plugins>
  38. <plugin>
  39. <groupId>org.apache.maven.plugins</groupId>
  40. <artifactId>maven-compiler-plugin</artifactId>
  41. <version>3.2</version>
  42. <configuration>
  43. <source>1.7</source>
  44. <target>1.7</target>
  45. </configuration>
  46. </plugin>
  47. <plugin>
  48. <groupId>org.apache.maven.plugins</groupId>
  49. <artifactId>maven-war-plugin</artifactId>
  50. <version>2.4</version>
  51. <configuration>
  52. <warSourceDirectory>src/main/webapp</warSourceDirectory>
  53. <warName>Spring4MVCCRUDRestService</warName>
  54. <failOnMissingWebXml>false</failOnMissingWebXml>
  55. </configuration>
  56. </plugin>
  57. </plugins>
  58. </pluginManagement>
  59. <finalName>Spring4MVCCRUDRestService</finalName>
  60. </build>
  61. </project>

User Service

  1. package com.websystique.springmvc.service;
  2. import java.util.List;
  3. import com.websystique.springmvc.model.User;
  4. public interface UserService {
  5. User findById(long id);
  6. User findByName(String name);
  7. void saveUser(User user);
  8. void updateUser(User user);
  9. void deleteUserById(long id);
  10. List<User> findAllUsers();
  11. void deleteAllUsers();
  12. public boolean isUserExist(User user);
  13. }
  1. package com.websystique.springmvc.service;
  2. import java.util.ArrayList;
  3. import java.util.Iterator;
  4. import java.util.List;
  5. import java.util.concurrent.atomic.AtomicLong;
  6. import org.springframework.stereotype.Service;
  7. import org.springframework.transaction.annotation.Transactional;
  8. import com.websystique.springmvc.model.User;
  9. @Service("userService")
  10. @Transactional
  11. public class UserServiceImpl implements UserService{
  12. private static final AtomicLong counter = new AtomicLong();
  13. private static List<User> users;
  14. static{
  15. users= populateDummyUsers();
  16. }
  17. public List<User> findAllUsers() {
  18. return users;
  19. }
  20. public User findById(long id) {
  21. for(User user : users){
  22. if(user.getId() == id){
  23. return user;
  24. }
  25. }
  26. return null;
  27. }
  28. public User findByName(String name) {
  29. for(User user : users){
  30. if(user.getName().equalsIgnoreCase(name)){
  31. return user;
  32. }
  33. }
  34. return null;
  35. }
  36. public void saveUser(User user) {
  37. user.setId(counter.incrementAndGet());
  38. users.add(user);
  39. }
  40. public void updateUser(User user) {
  41. int index = users.indexOf(user);
  42. users.set(index, user);
  43. }
  44. public void deleteUserById(long id) {
  45. for (Iterator<User> iterator = users.iterator(); iterator.hasNext(); ) {
  46. User user = iterator.next();
  47. if (user.getId() == id) {
  48. iterator.remove();
  49. }
  50. }
  51. }
  52. public boolean isUserExist(User user) {
  53. return findByName(user.getName())!=null;
  54. }
  55. private static List<User> populateDummyUsers(){
  56. List<User> users = new ArrayList<User>();
  57. users.add(new User(counter.incrementAndGet(),"Sam",30, 70000));
  58. users.add(new User(counter.incrementAndGet(),"Tom",40, 50000));
  59. users.add(new User(counter.incrementAndGet(),"Jerome",45, 30000));
  60. users.add(new User(counter.incrementAndGet(),"Silvia",50, 40000));
  61. return users;
  62. }
  63. public void deleteAllUsers() {
  64. users.clear();
  65. }
  66. }

Model (模型)类

  1. package com.websystique.springmvc.model;
  2. public class User {
  3. private long id;
  4. private String name;
  5. private int age;
  6. private double salary;
  7. public User(){
  8. id=0;
  9. }
  10. public User(long id, String name, int age, double salary){
  11. this.id = id;
  12. this.name = name;
  13. this.age = age;
  14. this.salary = salary;
  15. }
  16. public long getId() {
  17. return id;
  18. }
  19. public void setId(long id) {
  20. this.id = id;
  21. }
  22. public String getName() {
  23. return name;
  24. }
  25. public void setName(String name) {
  26. this.name = name;
  27. }
  28. public int getAge() {
  29. return age;
  30. }
  31. public void setAge(int age) {
  32. this.age = age;
  33. }
  34. public double getSalary() {
  35. return salary;
  36. }
  37. public void setSalary(double salary) {
  38. this.salary = salary;
  39. }
  40. @Override
  41. public int hashCode() {
  42. final int prime = 31;
  43. int result = 1;
  44. result = prime * result + (int) (id ^ (id >>> 32));
  45. return result;
  46. }
  47. @Override
  48. public boolean equals(Object obj) {
  49. if (this == obj)
  50. return true;
  51. if (obj == null)
  52. return false;
  53. if (getClass() != obj.getClass())
  54. return false;
  55. User other = (User) obj;
  56. if (id != other.id)
  57. return false;
  58. return true;
  59. }
  60. @Override
  61. public String toString() {
  62. return "User [id=" + id + ", name=" + name + ", age=" + age
  63. + ", salary=" + salary + "]";
  64. }
  65. }
配置类
  1. package com.websystique.springmvc.configuration;
  2. import org.springframework.context.annotation.ComponentScan;
  3. import org.springframework.context.annotation.Configuration;
  4. import org.springframework.web.servlet.config.annotation.EnableWebMvc;
  5. @Configuration
  6. @EnableWebMvc
  7. @ComponentScan(basePackages = "com.websystique.springmvc")
  8. public class HelloWorldConfiguration {
  9. }
 

初始化类

  1. package com.websystique.springmvc.configuration;
  2. import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;
  3. public class HelloWorldInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {
  4. @Override
  5. protected Class<?>[] getRootConfigClasses() {
  6. return new Class[] { HelloWorldConfiguration.class };
  7. }
  8. @Override
  9. protected Class<?>[] getServletConfigClasses() {
  10. return null;
  11. }
  12. @Override
  13. protected String[] getServletMappings() {
  14. return new String[] { "/" };
  15. }
  16. }

为你的REST API添加CORS支持

当访问REST API时,你可能需要面对“同源策略”问题。
错误如下:
” No ‘Access-Control-Allow-Origin’ header is present on the requested resource. Origin ‘http://127.0.0.1:8080′ is therefore not allowed access.” OR
” XMLHttpRequest cannot load http://abc.com/bla. Origin http://localhost:12345 is not allowed by Access-Control-Allow-Origin.”
一般来说,在服务器端,我们在响应中返回额外的CORS访问控制头,实现跨域链接。
用 Spring的话,我么可以写一个简单的过滤器为每个响应添加CORS特征头。
  1. package com.websystique.springmvc.configuration;
  2. import java.io.IOException;
  3. import javax.servlet.Filter;
  4. import javax.servlet.FilterChain;
  5. import javax.servlet.FilterConfig;
  6. import javax.servlet.ServletException;
  7. import javax.servlet.ServletRequest;
  8. import javax.servlet.ServletResponse;
  9. import javax.servlet.http.HttpServletResponse;
  10. public class CORSFilter implements Filter {
  11. public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
  12. System.out.println("Filtering on...........................................................");
  13. HttpServletResponse response = (HttpServletResponse) res;
  14. response.setHeader("Access-Control-Allow-Origin", "*");
  15. response.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, OPTIONS, DELETE");
  16. response.setHeader("Access-Control-Max-Age", "3600");
  17. response.setHeader("Access-Control-Allow-Headers", "x-requested-with");
  18. chain.doFilter(req, res);
  19. }
  20. public void init(FilterConfig filterConfig) {}
  21. public void destroy() {}
  22. }
需要将其添加在Spring 配置中:
  1. package com.websystique.springmvc.configuration;
  2. import javax.servlet.Filter;
  3. import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;
  4. public class HelloWorldInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {
  5. @Override
  6. protected Class<?>[] getRootConfigClasses() {
  7. return new Class[] { HelloWorldConfiguration.class };
  8. }
  9. @Override
  10. protected Class<?>[] getServletConfigClasses() {
  11. return null;
  12. }
  13. @Override
  14. protected String[] getServletMappings() {
  15. return new String[] { "/" };
  16. }
  17. @Override
  18. protected Filter[] getServletFilters() {
  19. Filter [] singleton = { new CORSFilter()};
  20. return singleton;
  21. }
  22. }

REST API 开发的更多相关文章

  1. 第六代智能英特尔® 酷睿™ 处理器图形 API 开发人员指南

    欢迎查看第六代智能英特尔® 酷睿™ 处理器图形 API 开发人员指南,该处理器可为开发人员和最终用户提供领先的 CPU 和图形性能增强.各种新特性和功能以及显著提高的性能. 本指南旨在帮助软件开发人员 ...

  2. 天气预报API开发

    天气预报API开发 一.        寻觅篇 最近想要跟着视频练习一下利用API开发一个天气预报系统,就在网上找了一下可以用的API,结果好多都已经失效了... 1.       百度车联网天气预报 ...

  3. 高性能PHP框架thinkphp5.0.0 Beta发布-为API开发而设计

    ThinkPHP V5.——为API开发而设计的高性能框架 ThinkPHP5..0版本是一个颠覆和重构版本,采用全新的架构思想,引入了很多的PHP新特性,优化了核心,减少了依赖,实现了真正的惰性加载 ...

  4. ASP.NET Core Web API 开发-RESTful API实现

    ASP.NET Core Web API 开发-RESTful API实现 REST 介绍: 符合REST设计风格的Web API称为RESTful API. 具象状态传输(英文:Representa ...

  5. 基于.Net Framework 4.0 Web API开发(2):ASP.NET Web APIs 参数传递方式详解

    概述:  ASP.NET Web API 的好用使用过的都知道,没有复杂的配置文件,一个简单的ApiController加上需要的Action就能工作.调用API过程中参数的传递是必须的,本节就来谈谈 ...

  6. API 开发实践

    整个2015年,如果要给自己打上一个标签的话,那应该就是 API. 在各个不同的系统中定制各种 API 框架. 在做商城对接各种电商 ERP 的 API 开发中,我采用的是兼容SHOPEX 的 API ...

  7. 报表引擎API开发入门— EJB程序数据源

    我们前面讲了几个数据源,今天我们来讲一下EJB数据源,这篇讲完我们数据源这部分就讲完了.数据连接不需要直接访问数据库,而是使用EJB做为数据源.FR通过定义程序数据集使用EJB的相关类获取到EJB数据 ...

  8. Rest API 开发 学习笔记(转)

    Rest API 开发 学习笔记 概述 REST 从资源的角度来观察整个网络,分布在各处的资源由URI确定,而客户端的应用通过URI来获取资源的表示方式.获得这些表徵致使这些应用程序转变了其状态.随着 ...

  9. Windows下mock环境搭建-加速项目Api开发

    本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 公司进行技术部拆分,以项目制作为新的开发模式,前端+移动端+后端,于是加速Api开发变得很有必要,准 ...

  10. 淘宝API开发系列---阿里.聚石塔.开放平台的使用

    好久没有继续跟进淘宝的API使用了,有很多做相关应用的同行都来咨询,很多都因为自己开发工作比较忙而没有来得及好的处理,前几天,有一个朋友叫帮忙指导如何使用淘宝API,由于原来有一些成熟的例子应用,因此 ...

随机推荐

  1. [翻译] BAFluidView

    BAFluidView https://github.com/antiguab/BAFluidView This view and it's layer create a 2D fluid anima ...

  2. 浅析NSTextContainer

    浅析NSTextContainer TextKit中的NSTextContainer有点晦涩难懂,如果想用TextKit实现文本分页的效果,你是必须要使用NSTextContainer的...... ...

  3. Centos7下Mysql通过.frm和.ibd恢复数据

    通过.frm和.ibd文件恢复表结构和数据 这里以hue数据库中的desktop_document2表为例 分成两步骤,先去表结构,再取数据,最好在一个用完就可以删除的数据库中进行 取表结构篇: 1. ...

  4. React onPaste 获取粘贴板的值

    React 中, 获取 粘贴板的值, 使用下面的方法 console.log(e.clipboardData.getData('Text')); 如果是 JS 中的 onpaste 事件, 则使用 v ...

  5. Programming Assignment 2: Deques and Randomized Queues

    编程作业二 作业链接:Deques and Randomized Queues & Checklist 我的代码:Deque.java & RandomizedQueue.java & ...

  6. java Math数学工具及Random随机函数

    Math类包含用于执行基本数学运算的方法,如绝对值.对数.平方根和三角函数.它是一个final类,其中定义的都是一些常量和静 态方法.常用方法如下:public static double sqrt( ...

  7. vue-cli项目打包优化(webpack3.0)

    1.修改source-map配置:此配置能大大减少打包后文件体积. a.首先修改 /config/index.js 文件: // /config/index.js dev环境:devtool: 'ev ...

  8. Python之美[从菜鸟到高手]--2+2=5

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/yueguanghaidao/article/details/35644165     今天在伯乐在线 ...

  9. 【openjudge】【字符串+模拟】1777:文件结构“图”

    [题目传送门:]戳 [描述:] 在计算机上看到文件系统的结构通常很有用.Microsoft Windows上面的"explorer"程序就是这样的一个例子.但是在有图形界面之前,没 ...

  10. Docker实战(四)之Docker数据管理

    在生产环境中使用Docker的过程中,往往需要对数据进行持久化,或者需要在多个容器之间进行数据共享,这必然涉及到容器的数据管理操作. 容器中管理数据主要有两种形式: 数据卷:容器内数据直接映射到本地主 ...