前端杂谈 · Web

前端跨域-实践篇

小编 · 1月11日 · 2021年

前言

上文讲到浏览器因为安全的思考,设置了同源策略。这就是跨域的形成原因,也谈到了跨域的一些解决方案。

前端跨域-实践篇

本文就上篇文章的实践,在项目开发时好作为参考资料。

准备环境

在开始之前,先说说实验环境:

前端采用 Vue3 搭建,端口为 8080。

后端采用 Spring Boot 搭建,端口为 8888。

后端测试接口:

@RestController
public class HelloController {
    @GetMapping("/hello")
    public Map hello(){
        HashMap result = new HashMap();
        result.put("message", "Hello World!");
        return result;
    }
}

启动项目,测试一下:

curl localhost:8888/hello
{"message":"Hello World"}

解决前后端跨域可以说就是解决 AJAX 跨域。上文说到四种跨域方案,其中 CORS 和 代理服务器 为常见解决方案。

CORS

使用 @CrossOrigin 注解

@CrossOrigin 可以用在类或者方法上:

@CrossOrigin(origins = "http://localhost:8080")
@RestController
public class HelloController

写在类上,表示该类的所有方法对应的接口浏览器都不会拦截。

@GetMapping("/hello")
@CrossOrigin(origins = "http://localhost:8080")
public String hello()

写在方法上,表示该方法对应的接口浏览器不会拦截。

实现 WebMvcConfigurer

创建一个类 CorsConfig,使用 @Configuration 标识它为配置类;实现 WebMvcConfigurer,重写 addCorsMappings 方法:

@Configuration
public class CorsConfig implements WebMvcConfigurer {

    @Override
    public void addCorsMappings(CorsRegistry registry) {
        registry.addMapping("/**")
                .allowedOrigins("http://localhost:8080");
    }
}

需要注意的是,如果设置 Access-Control-Allow-Origin 为 * ,Access-Control-Allow-Credentials 就不能设置为 true。

Filter

创建一个类 CorsFilter,使用 @Configuration 标识它为配置类;实现 Filter,实现 doFilter 方法:

@Configuration
public class CorsFilter implements Filter {

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) throws IOException, ServletException {
        HttpServletResponse httpResponse = (HttpServletResponse) response;
        httpResponse.setHeader("Access-Control-Allow-Origin", "*");
        filterChain.doFilter(request, response);
    }
}

更多 CORS 头部信息,可以 参考文档

测试

在前端代码中或者用浏览器打开前端监听的端口,输入以下 JavaScript 代码:

fetch('http://localhost:8888/hello')
.then(response => response.json())
.then(json => console.log(json))

返回:

{
    "message": "Hello World"
}

代理服务器

nginx

在开发过程中,前端会自己起一个服务来调试代码。于是 nginx 可以监听 80 端口,分别反向代理前端服务和后台服务。

server {
    listen 80;
    server_name 127.0.0.1;

    location / {
        proxy_pass http://127.0.0.1:8080;
    }

    location /api/ {
        proxy_pass http://127.0.0.1:8888/;
    }
}

需要注意的是,后端反向代理端口后要加上符号 /。否则访问 127.0.0.1/api/hello 就会反向代理到 http://127.0.0.1:8888/api/hello 而不是 http://127.0.0.1:8888/hello。

前端写完代码之后,可以将代码打包成静态文件,使用 nginx 来解析。

server {
    listen 80;
    server_name 127.0.0.1;
    index index.html;
    root /home/kang/vue-demo/dist;

    location / {
        try_files $uri $uri/ /index.html;
    }

    location /api/ {
        proxy_pass http://127.0.0.1:8888/;
    }
}

需要注意的是,如果前端使用 history 来模拟 url,那在代理的过程中需要重写跳转规则:try_files $uri $uri/ /index.html。该语句表示 URL 的跳转由 index.html 管理。删除导致前端项目路由失效且 nginx 响应 404。

测试

在前端项目中,或者用浏览器打开,输入下列代码:

fetch('/api/hello')
.then(response => response.json())
.then(json => console.log(json))

返回:

{
    "message": "Hello World"
}

node

在根目录中打开 vue.config.js(如果没有就新建)。写下如下语句:

module.exports = {
    devServer: {
        proxy: {
            "/api": {
                target: "http://localhost:8888",
                changeOrigin: true,
                pathRewrite: {
                    '^/api': ''
                }
            }
        }
    }
}

这里需要注意的是,这里的 pathRewrite,也就是路径重写。将 /api 前缀会去掉了。这样访问 /api/hello 才是 http://127.0.0.1:8888/hello,而不是 http://127.0.0.1:8888/hello。

测试

在前端项目中,或者用浏览器打开,输入下列代码:

fetch('/api/hello')
.then(response => response.json())
.then(json => console.log(json))

返回:

{
    "message": "Hello World"
}

参考

前端跨域-理论篇

2021-1-11 0