SpringBoot与Vue项目的跨域问题

SpringBoot与Vue项目的跨域问题

1. 项目初始化

1.1 SpringBoot项目

在这里插入图片描述

package com.icebear.controller;

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
  *
  * @author icebear
  * @date 2023/2/17
  * @info 测试sessionID
  */
@RestController
@RequestMapping("/api/tests")
public class TestController {

    @GetMapping("/getSession")
    public String getSession(HttpServletRequest request, HttpServletResponse response) {
        return request.getSession().getId();
    }
}

1.2 Vue项目

在这里插入图片描述

// main.js
import { createApp } from 'vue'
import App from './App.vue'

import axios from "axios";

axios.defaults.baseURL = 'http://localhost:8080/api'

const app = createApp(App)
app.mount('#app')
app.config.globalProperties.$axios = axios

<!-- App.vue -->
<template>
  <div class="btn-div">
    <button @click="send">发送请求,获取sessionId</button>
  </div>
  <div class="text-div">
    后面显示获取的sessionId:
    <span v-text="sessionId" style="color: red"></span>
  </div>
</template>

<script>

export default {
  name: 'App',
  data () {
    return {
      sessionId: ''
    }
  },
  methods: {
    send() {
      this.$axios.get('/tests/getSession')
        .then(res => {
          console.log(res)
          this.sessionId = res.data
        })
        .catch(err => {
          console.log(err)
        })
    }
  }
}
</script>

<style scoped>

</style>

2. 跨域问题及解决

2.1 跨域问题出现

在启动SpringBoot项目和Vue项目之后,点击浏览器页面内的按钮,发送请求,此时浏览器的控制台会出现如下错误:

2.2 SpringBoot解决跨域问题

2.2.1 使用@CrossOrigin注解
package com.icebear.controller;

import org.springframework.web.bind.annotation.CrossOrigin;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
  *
  * @author icebear
  * @date 2023/2/17
  * @info 测试sessionID
  */
@RestController
@RequestMapping("/api/tests")
// 使用@CrossOrigin注解处理跨域问题
@CrossOrigin
public class TestController {

    @GetMapping("/getSession")
    public String getSession(HttpServletRequest request, HttpServletResponse response) {
        return request.getSession().getId();
    }
}

浏览器结果如下:

在这里插入图片描述
在这里插入图片描述

2.2.2 实现WebMvcConfigurer接口的addCorsMappings方法

在这里插入图片描述

package com.icebear.config;

import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.CorsRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
/**
  *
  * @author icebear
  * @date 2023/2/17
  * @info 方法二:实现WebMvcConfigurer的addCorsMappings方法处理跨域
  */
@Configuration
public class MyCrosConfig implements WebMvcConfigurer {
    @Override
    public void addCorsMappings(CorsRegistry registry) {
        registry.addMapping("/**") // 所有接口
                .allowCredentials(true) // 是否发送cookie
                .allowedOriginPatterns("*") // 支持域
                .allowedMethods("GET", "POST", "PUT", "DELETE") // 支持方法
                .allowedHeaders("*")
                .exposedHeaders("*");
    }
}

浏览器结果如下:

在这里插入图片描述
在这里插入图片描述

2.2.3 添加CorsFilter

在这里插入图片描述

package com.icebear.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.cors.CorsConfiguration;
import org.springframework.web.cors.UrlBasedCorsConfigurationSource;
import org.springframework.web.filter.CorsFilter;

import java.util.Arrays;
/**
  *
  * @author icebear
  * @date 2023/2/17
  * @info 方法三:添加CorsFilter解决跨域问题
  */
@Configuration
public class MyCorsFilter {

    @Bean // 不要忘记添加此注解
    public CorsFilter corsFilter() {
        // 创建CORS配置对象
        CorsConfiguration config = new CorsConfiguration();
        config.addAllowedOriginPattern("*"); // 设置支持域
        config.setAllowCredentials(true); // 是否发送cookie
        config.setAllowedMethods(Arrays.asList("GET", "POST", "PUT", "DELETE")); // 支持的请求方式
        config.addAllowedHeader("*"); // 允许的原始请求头部信息
        config.addExposedHeader("*"); // 暴露的头部信息
        // 添加地址映射
        UrlBasedCorsConfigurationSource corsConfigurationSource = new UrlBasedCorsConfigurationSource();
        corsConfigurationSource.registerCorsConfiguration("/**", config);
        // 返回CorsFilter对象
        return new CorsFilter(corsConfigurationSource);
    }
}

浏览器结果如下:

在这里插入图片描述
在这里插入图片描述

3. 服务器部署后没有Cookie

3.1 环境模拟

3.1.1 软件准备
  1. linux虚拟机
  2. 安装java运行环境
  3. 安装nginx:LINUX安装nginx详细步骤
3.1.2 部署项目
  1. Vue项目构建成功后上传虚拟机;
  2. SpringBoot项目打包成功后上传虚拟机;
3.1.3 模拟域名
  1. 使用命令ifconfig获取虚拟机ip地址,例如:192.168.1.3
  2. 修改windows的hosts文件,将域名和虚拟机的ip地址做映射;

如何修改hosts文件?几种修改hosts文件的方法

在这里插入图片描述

  1. 修改nginx配置,指定域名对应的端口和项目位置;
http {

    ......
    
    # 二级域名转发
    server {
        listen 80;
        server_name bear.iceindex.xyz;
        # root /www/iceindex.xyz/dist;
        # 二级域名对应的Vue项目转发
        location / {
            proxy_pass http://0.0.0.0:8091;
        }
    }
    # Vue项目的实际访问地址
    server {
        listen 8091;
        server_name localhost;
        root /www/iceindex.xyz/dist;
        location / {
            root /www/iceindex.xyz/dist;
            try_files $uri $uri/ /index.html;
        }
    }
}

3.2 运行项目

3.2.1 前后端部署在同一台服务器中

3.2.1.1 通过IP:Port方式访问项目

浏览器输入Vue项目的IP:Port访问,并发送请求,结果如下:

在这里插入图片描述

解决方法:

修改axios的默认配置:

import { createApp } from 'vue'
import App from './App.vue'

import axios from "axios";

// 本地SpringBoot项目地址
// axios.defaults.baseURL = 'http://localhost:8080/api'

// 服务器SpringBoot项目地址
axios.defaults.baseURL = 'http://192.168.1.3:8080/api'

// 添加下面一行可以获取cookie
axios.defaults.withCredentials = true

const app = createApp(App)
app.mount('#app')
app.config.globalProperties.$axios = axios

修改后结果:

从下图的结果中可以看出,成功获得了JSESSIONID所对应的cookie

在这里插入图片描述

3.2.1.2 通过域名访问项目

在浏览器输入Vue项目配置的域名地址进行访问,并发送请求,结果如下:

从下图中的结果中可以看出,没有获得JSESSIONID所对应的cookie

在这里插入图片描述
在这里插入图片描述

解决方法:

  1. 修改axiosbaseUrl配置:
import { createApp } from 'vue'
import App from './App.vue'

import axios from "axios";

// 本地SpringBoot项目地址
// axios.defaults.baseURL = 'http://localhost:8080/api'

// 服务器SpringBoot项目地址 IP对应
// axios.defaults.baseURL = 'http://192.168.1.3:8080/api'

// 服务器SpringBoot项目地址 域名对应
axios.defaults.baseURL = 'http://bear.iceindex.xyz/api'

// 添加下面一行可以获取cookie
axios.defaults.withCredentials = true

const app = createApp(App)
app.mount('#app')
app.config.globalProperties.$axios = axios

  1. 修改nginx配置:
{
    # 二级域名转发
    server {
        listen 80;
        server_name bear.iceindex.xyz;
        # root /www/iceindex.xyz/dist;
        # 二级域名对应的Vue项目转发
        location / {
            proxy_pass http://0.0.0.0:8091;
        }
        # 二级域名对应的SpringBoot项目转发
        location /api {
            proxy_pass http://0.0.0.0:8080;
        }
    }
    # Vue项目的实际访问地址
    server {
        listen 8091;
        server_name localhost;
        root /www/iceindex.xyz/dist;
        location / {
            root /www/iceindex.xyz/dist;
            try_files $uri $uri/ /index.html;
        }
    }
}

修改后的结果:

在这里插入图片描述
在这里插入图片描述

3.2.2 前后端部署在不同服务器中

由于前后端部署在不同的服务器中,Vue项目的地址和SpringBoot项目的地址是不同的,模拟的方式为:Vue项目中的axios的baseUrl设置为SpringBoot项目地址:

import { createApp } from 'vue'
import App from './App.vue'

import axios from "axios";

// 本地SpringBoot项目地址
// axios.defaults.baseURL = 'http://localhost:8080/api'

// 服务器SpringBoot项目地址 IP对应
// axios.defaults.baseURL = 'http://192.168.1.3:8080/api'

// 服务器SpringBoot项目地址 域名对应
axios.defaults.baseURL = 'http://bear.iceindex.xyz/api'

// 添加下面一行可以获取cookie
axios.defaults.withCredentials = true

const app = createApp(App)
app.mount('#app')
app.config.globalProperties.$axios = axios

本地运行Vue项目结果:
在这里插入图片描述

解决方法:

将协议升级为https,并设置Cookie的SameSite值为None,Secure值改为true

详情参考:【跨域】一篇文章彻底解决跨域设置cookie问题!

4. 其他参考

解决vue+springboot前后端分离项目,前端跨域访问sessionID不一致导致的session为null问题

springboot-vue跨域详解

SpringBoot 解决跨域问题的 5 种方案!

手把手教 Nginx 部署 Vue 项目

【vue项目部署CSS失效】VUE部署后css样式加载无效和失效多种情况解决方案

vue项目部署 nginx 配置

新版本chrome浏览器带来的跨域请求cookie丢失问题

解决新版谷歌Chrome浏览器Cookie跨域失效问题

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Spring BootVue解决跨域问题的方法有以下几种: 1. 在Spring Boot中配置跨域访问 可以在Spring Boot的配置文件中添加如下配置: ``` @Configuration public class CorsConfig { @Bean public WebMvcConfigurer corsConfigurer() { return new WebMvcConfigurerAdapter() { @Override public void addCorsMappings(CorsRegistry registry) { registry.addMapping("/**") .allowedOrigins("*") .allowedMethods("GET", "POST", "PUT", "DELETE", "OPTIONS") .allowedHeaders("*") .allowCredentials(true) .maxAge(360); } }; } } ``` 这样就可以允许所有的跨域请求了。 2. 在Vue中配置代理 可以在Vue的配置文件中添加如下配置: ``` module.exports = { devServer: { proxy: { '/api': { target: 'http://localhost:808', changeOrigin: true, pathRewrite: { '^/api': '' } } } } } ``` 这样就可以将所有以/api开头的请求代理到Spring Boot的808端口上了。 3. 使用CORS插件 可以在Vue中使用CORS插件来解决跨域问题。可以使用如下代码: ``` import cors from 'cors'; app.use(cors()); ``` 这样就可以允许所有的跨域请求了。 ### 回答2: 在开发Web应用程序时,通常会涉及到多个域之间的交互,例如从前端Vue应用程序发送Ajax请求到后端Spring Boot应用程序时。但是,由于浏览器的同源策略,跨域请求将被拒绝。 解决跨域问题有许多方法,下面将讨论如何使用Spring BootVue解决跨域问题。 首先,可以在Spring Boot应用程序的控制器类上添加@CrossOrigin注解来实现跨源请求。例如: ```java @RestController @RequestMapping("/api") @CrossOrigin public class MyRestController { //... 省略其他代码 } ``` 这将允许来自任何源的Ajax请求访问该控制器中的资源。 另一种解决跨域问题的方法是使用Vue中的代理。在Vue的config/index.js文件中,可以将代理配置如下: ```javascript module.exports = { dev: { proxyTable: { '/api': { target: 'http://localhost:8080', changeOrigin: true, pathRewrite: { '^/api': '', }, }, }, }, }; ``` 这将允许Vue应用程序向以“/api”开头的URL发出请求,并将其代理到Spring Boot应用程序的“http://localhost:8080”地址。changeOrigin选项设置为true将更改host头以匹配目标URL的主机。 最后,还可以将CORS配置添加到Spring Boot应用程序的配置类中。具体而言,可以将以下内容添加到@Configuration类中: ```java @Configuration public class MyConfig { @Bean public WebMvcConfigurer corsConfigurer() { return new WebMvcConfigurerAdapter() { @Override public void addCorsMappings(CorsRegistry registry) { registry.addMapping("/api/**") .allowedOrigins("http://localhost:8081") .allowedMethods("GET", "POST") .allowedHeaders("*") .allowCredentials(true) .maxAge(3600); } }; } } ``` 这将允许来自“http://localhost:8081”地址的Ajax请求访问以“/api”开头的URL,并允许使用GET和POST方法。还可以指定允许所有标头,并设置允许凭据和缓存生存期。 总之,Spring BootVue都提供了多种解决跨域问题的方法。选择适合自己的方法,可以使应用程序更加健壮和可扩展。 ### 回答3: 随着Web技术不断发展,很多网站和应用需要同时使用多个不同的域名,这就产生了跨域问题跨域问题的解决方法有很多种,其中比较流行的方式是使用SpringBootVue。 首先,我们需要明确什么是跨域。跨域是指一个网站访问另一个域名下的资源时,由于浏览器的同源策略限制,会导致访问失败。例如,一个网站(域名为A)要访问另一个网站(域名为B)的数据,由于B网站与A网站域名不同,浏览器会拒绝此次访问。因此,我们需要使用SpringBootVue解决跨域问题。 使用SpringBoot解决跨域问题SpringBoot提供了解决跨域问题的方式,只需要在Controller中添加@CrossOrigin注解即可。例如: ``` @RestController @RequestMapping("/api") @CrossOrigin(origins = "*", maxAge = 3600) public class ApiController { //... } ``` 其中,@CrossOrigin注解的origins属性指定允许跨域访问的源地址,*代表允许任何源地址访问;maxAge属性指定允许访问的时间,单位为秒。 使用Vue解决跨域问题Vue可以使用axios库发送HTTP请求,实现跨域访问。在axios发送请求时,我们需要设置请求头,告诉服务器允许跨域访问。例如: ``` import axios from 'axios' axios.defaults.baseURL = 'http://localhost:8080/api' axios.defaults.headers.post['Content-Type'] = 'application/json;charset=UTF-8' axios.defaults.headers.common['Access-Control-Allow-Origin'] = '*' export default { getList (params) { return axios.post('/list', params) }, //... } ``` 其中,axios.defaults.headers.common['Access-Control-Allow-Origin'] = '*'设置了允许跨域访问的源地址为*,代表任何源地址均可访问。 综上所述,SpringBootVue都提供了解决跨域问题的方式,我们只需要按照要求设置相应参数即可。这样,我们就能够轻松实现跨域访问,提高我们的开发效率。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值