前后端分离,解决vue+axios跨域和proxyTable不生效等问题

看到我这篇文章前可能你以前看过很多类似的文章。至少我是这样的,因为一直没有很好的解决问题。

正文

当我们通过webstorm等IDE开发工具启动项目的时候,通过命令控制台可以观察到启动项目的命令
如下:

webpack-dev-server --inline --progress --config build/webpack.dev.conf.js

我们很容易发现,用到了一个配置文件build/webpack.dev.conf.js
我因为实在逼得找不到原因,明明按照网上的文章配置的,但是跨域依旧没解决,总会遇到各种各样的问题。

不得已之下,看了下项目的启动配置文件。看完后,感觉自己搞懂了怎么回事,有感。

开始

首先我们要确认后端的跨域是允许的。

首先确保后端跨域没问题,目的是解决前端的问题

Springboot为例可以通过下面注解允许某个接口(注解方法上与@RequestMapping,@GetMapping等相关注解一起使用),或一批接口(注解在类上)

@CrossOrigin(originPatterns = "*", allowCredentials = "true", origins = "*")

如果是springsecurity 登录接口,简单代码片段列举,通过csrf().disable() 去允许接口可以跨域

protected void configure(HttpSecurity http) throws Exception {
    http.csrf().disable()

随着版本的迭代有些配置可能会变更,我们可以通过观看源码找到那些内容发生了变更,然后自己尝试去解决问题。

在网上你会看到各种各样的文章,有人说是改config/vue.config.js,也有人说是改config/index.js 去解决跨域问题。

以我这里为例子,会发现项目中没有vue.config.js与网上某些文章不一样。
在这里插入图片描述
通过观看build/webpack.dev.conf.js 源码
截取部分源码,会发现require('../config')引入了config/index.js
最主要的是我们看到了源码中有config.dev.host等相关的内容,
其中我们发现代理相关的词proxy: config.dev.proxyTable, 从这里我们就可以知道用的是proxyTable的配置来代理后端解决跨域问题。
如果你发现项目用的名称不叫 proxyTable 只需要找到对应位置去修改就行了

'use strict'
const utils = require('./utils')
const webpack = require('webpack')
const config = require('../config')
const merge = require('webpack-merge')
const path = require('path')
const baseWebpackConfig = require('./webpack.base.conf')
const CopyWebpackPlugin = require('copy-webpack-plugin')
const HtmlWebpackPlugin = require('html-webpack-plugin')
const FriendlyErrorsPlugin = require('friendly-errors-webpack-plugin')
const portfinder = require('portfinder')

const HOST = process.env.HOST
const PORT = process.env.PORT && Number(process.env.PORT)

const devWebpackConfig = merge(baseWebpackConfig, {
  module: {
    rules: utils.styleLoaders({ sourceMap: config.dev.cssSourceMap, usePostCSS: true })
  },
  // cheap-module-eval-source-map is faster for development
  devtool: config.dev.devtool,

  // these devServer options should be customized in /config/index.js
  devServer: {
    clientLogLevel: 'warning',
    historyApiFallback: {
      rewrites: [
        { from: /.*/, to: path.posix.join(config.dev.assetsPublicPath, 'index.html') },
      ],
    },
    hot: true,
    contentBase: false, // since we use CopyWebpackPlugin.
    compress: true,
    host: HOST || config.dev.host,
    port: PORT || config.dev.port,
    open: config.dev.autoOpenBrowser,
    overlay: config.dev.errorOverlay
      ? { warnings: false, errors: true }
      : false,
    publicPath: config.dev.assetsPublicPath,
    proxy: config.dev.proxyTable,
    quiet: true, // necessary for FriendlyErrorsPlugin
    watchOptions: {
      poll: config.dev.poll,
    }
  },

我们确定了需要改的内容是config/index.js 中的 config.dev.proxyTable 字段配置

proxyTable: {
  '/api': {
    target: 'http://localhost:8081', // 你请求的第三方接口,后端地址
    changeOrigin: true,// 在本地会创建一个虚拟服务端,然后发送请求的数据,并同时接收请求的数据,这样服务端和服务端进行数据的交互就不会有跨域问题
    ws: true,
    pathRewrite: { // 路径重写,
      '^/api': ''// 替换target中的请求地址
    }
  }
}

UI 的端口是8080,后台的端口是8081
我们希望用axios去访问8081返回我们想要的数据。
上面的配置意思是,会将 url 以/api 开头通过axios发送的请求会被拦截和代理。
例如我想访问后端的登录接口http://localhost:8081/user/login 去登录返回 token 和user的信息

一般我们会这样做

axios.post(
	'/user/login', 
	this.loginForm, 
	{headers: {'Content-Type': 'application/x-www-form-urlencoded'}})
.then(res => {
  // TODO
})

如果代理没生效,那么它发的请求就是http://localhost:8080/user/login 不是我们期望的8081端口。

实际上代理生效后,通过浏览器的Network也看不出来访问的是8081的还是8080端口的。
你只能在Network中看到,request header是8080端口。
因此很多人搞不清。
通过后端打断点,我确认了该请求是否访问到后端api。


上述给的例子中是否有发现proxyTable/api以及内部的pathRewrite配置,
但是我的axios请求中没有/api前缀,因此上述实际proxyTable并没有用,还需要给url带上/api前缀才可以代理生效

修改main.js加入下面片段,将前缀/api 通过总入口方式带入即可。(这样url就不要自己手动拼上/api

import axios from 'axios'
Vue.prototype.$axios = axios
axios.defaults.baseURL = '/api'
Vue.config.productionTip = false

最后重启vue项目检验一下。

Header信息如下,8080是前端的端口,Request URL: http://localhost:8080/api/user/login这里只是显示成前端的8080,实际上调用的是8081后端的api http://localhost:8081/user/login 后端的springboot接口是@PostMapping("/user/login")
我们可以明显看到后端有返回json数据过来。
在这里插入图片描述

在这里插入图片描述

### 关于Spring BootVue.js前后端分离架构中的问题解决方案 #### 后端配置CORS支持 在Spring Boot应用程序中,可以通过多种方式来启用源资源共享(CORS),从而允许来自其他名的请求。一种常见做法是在控制器级别或全局范围内应用`@CrossOrigin`注解[^1]。 另一种更灵活的方法是通过Java配置类实现自定义的CorsConfigurationSource bean: ```java @Configuration public class WebConfig implements WebMvcConfigurer { @Override public void addCorsMappings(CorsRegistry registry) { registry.addMapping("/**") // 对所有路径生效 .allowedOrigins("*") // 允许任何来源访问(生产环境建议指定具体地址) .allowedMethods("GET", "POST", "PUT", "DELETE", "OPTIONS") .maxAge(3600); // 预检请求的有效期为一小时 } } ``` 此方法能够精确控制哪些URL模式、HTTP动词以及原点被授权进行通信,并且还可以设置预检请求的最大有效期以减少不必要的握手次数[^2]。 #### 前端调整Axios默认选项 对于基于Vue.js构建的应用程序来说,在发起AJAX请求之前修改axios库的基础配置也是一种有效的手段。可以在项目的入口文件(main.js或其他合适位置)加入如下代码片段: ```javascript import axios from 'axios'; // 设置公共参数 axios.defaults.baseURL = 'http://localhost:8181'; // 后端API服务器地址 axios.defaults.headers.post['Content-Type'] = 'application/json;charset=UTF-8'; axios.defaults.withCredentials = true; // 如果需要携带cookie则开启此项 ``` 上述操作不仅简化了每次发送请求时重复输入基础URL的过程,同时也确保了即使存在多个异步调用也能统一管理认证凭证等问题[^3]。 #### 浏览器开发者工具排查 当遇到实际运行过程中仍然存在的错误提示时,利用浏览器内置的开发者工具(F12键打开),特别是Network面板下的XHR标签页,可以帮助快速定位并解决问题所在。注意观察响应头(Response Headers)部分是否存在Access-Control-Allow-Origin字段及其对应的值是否符合预期设定。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

诗水人间

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值