Vue 3.x+axios跨域方案的踩坑指南


Posted in Javascript onJuly 04, 2019

缘起

最近实验课上需要重构以前写过的一个项目(垃圾堆),需要添加发生邮件提醒的功能,记得以前写过一个PHP版的实现,所以想把PHP写的功能整理成一个服务,然后在前端调用。但是这个项目是JavaWeb,也就是说我需要面对跨域的问题。不过本篇文章,讲的并不是如何解决这样的跨域问题,而是我在找如何解决这个问题的路上遇到的坑。

其实,在前端工程化大行其道的现在,前后端已经分离开来,前端为了提高工作流效率往往自己开一个小型的服务器,就比如webpack.devServer。这样在前端调用后端接口的时候必然会面临跨域的问题,

如题,Vue3.x + axios 跨域方案 就是解决这里的跨域问题。这里的跨域是基于webpack的devServer的代理功能(proxy)来实现开发环境中的跨域,也就是说本篇所讨论的并不能解决生产环境下的跨域问题,因为webpack.devServer是DevDependencies,一旦打包上线,这个proxy代理就会失效。但是这并不妨碍我们开发中使用跨域来提高开发效率和体验。

开始填坑

其实这个问题解决起来很简单,网上也是很多教程,为了文章完整性,我这里也做一个尽量完备的展示,介绍如何配置Vue3.x来实现跨域 。

vue.config.js中devServer.proxy的配置解析

Vue3.x的CLI工具比Vue2.x的CLI工具构建的项目要简化很多,根目录下只有./src和./public文件夹,所以网上很多教程说config目录下的vue.config.js是说的vue2.x版本。那么对于Vue3.x版本,构建也很简单,直接在根目录里建一个vue.config.js配置文件就可以了,我们直接看devServer.proxy里的代码:

我这里devServer的地址是:localhost:8080/,需要代理的地址是:localhost/index/phpinfo.php (我自己写的一个测试跨域用的php,返回一个‘ok')

下面是根据上面的地址需要配置的proxy对象

devServer : {
  proxy : {
   '/index' : {
    target : 'http://localhost/index',
    // ws : true,
    changeOrigin : true,
    pathRewrite : {
     '^/index' : ''
    }
   }
  }
 }

大部分教程到这里就停止了,但是我在这里做一个扩展,为了让读者理解这里的配置是如何起作用的(以下内容整理自http-proxy-middleware的npm描述里,http-proxy-middleware是一个npm模块,是proxy的底层原理实现)。

         foo://example.com:8042/over/there?name=ferret#nose
         \_/   \______________/\_________/ \_________/ \__/
          |           |            |            |        |
       scheme     authority       path        query   fragment

以我上面的配置为例,'/index'这个key在http-proxy-middleware中被称为context——用来决定哪些请求需要被target对应的主机地址(这里是http://localhost/index)代理,它可以是 字符串,含有通配符的字符串,或是一个数组,分别对应于path matching(路径匹配)wildcard path matching(通配符路径匹配)multiple path matching(多路径匹配),而这里的path指的就是上图所标识的path段。

简言之,这个key就是匹配path的,一旦匹配到符合的path,就会把请求转发的代理主机去,而代理主机的地址就是target字段对应的内容。

那pathRewrit是什么意思呢?意如其名,路径重写。就是把模式(这里是^/index)匹配到的path重写为对应的路径(这里是'',相当于删除了这个匹配到的路径)。除了删除,还有在原有路径上添加一个基础路径,或是改写一个路径的方式,这可以参考http-proxy-middleware的npm描述的option.pathRewrite章节 。

在Vue中使用axios

这个使用任意一个ajax封装的库都是可行的,axios,jquery.ajax或者是vue-resource都是可以的。
在Vue中使用axios,网上有两种方法,一种是将axios加入Vue的原型里,我更推荐第二种方法:

npm install axios vue-axios
import axios from 'axios';
import VueAxios from 'vue-axios';
Vue.use(VueAxios,axios);

以我上面的proxy配置为基础,想要让代理成功转发到localhost/index/phpinfo.php,在Vue实例中axios需要这样写访问地址:

this.axios.get('/index/phpinfo.php').then((res)=>{
  console.log(res);
  })

我们来分析这些代码整个发挥作用的原理是什么?首先,axios去访问/index/phpinfo.php,这是个相对地址,所以真实访问地址其实是localhost:8080/index/phpinfo.php,然而/index/phpinfo.php被我们配置的/index匹配到了 ,所以访问被proxy代理,那转发到哪个路径呢?在pathRewrite中,我们将模式^/index的路径清除了,所以最终的访问路径是 target+pathRewrite+ 剩余的部分 , 这样也就是 http://localhost/index++/phpinfo.php

坑点

可能出现即使配置了proxy,但是依然没有任何卵用。

  • 大部分情况是因为你的proxy配置和你的访问路径不匹配,或者即使匹配到了,但是转发出去的地址不对,没有命中后端给的API
  • 或者看看axios,有没有使用正确姿势?
  • 还有一点,或许你看到返回的response里的url依然显示的是本地主机,但是数据已经正常返回,这是正常的,因为我们访问的本来就是本地主机,只不过proxy转发了这个请求到一个新的地址。

后续

本篇只解决了开发环境下的跨域问题,实际线上还不能跨域,目前这里有一些方案:

  • Nginx反向代理跨域
  • JSONP
  • CORS

下一次讨论这个跨域问题,尝试解决。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

Javascript 相关文章推荐
Prototype Template对象 学习
Jul 19 Javascript
JavaScript全排列的六种算法 具体实现
Jun 29 Javascript
详解javascript的变量与标识符
Jan 04 Javascript
jQuery实现获取h1-h6标题元素值的方法
Mar 06 Javascript
Angular2.js实现表单验证详解
Jun 23 Javascript
使用Node.js实现ORM的一种思路详解(图文)
Oct 24 Javascript
Angular 实现输入框中显示文章标签的实例代码
Nov 07 Javascript
Vue CLI3.0中使用jQuery和Bootstrap的方法
Feb 28 jQuery
layui--select使用以及下拉框实现键盘选择的例子
Sep 24 Javascript
javascript canvas实现简易时钟例子
Sep 05 Javascript
js+canvas绘制图形验证码
Sep 21 Javascript
JavaScript实现轮播图效果
Oct 30 Javascript
Vue.js递归组件实现组织架构树和选人功能
Jul 04 #Javascript
vue-cli配置flexible过程详解
Jul 04 #Javascript
vue动态配置模板 'component is'代码
Jul 04 #Javascript
react 移动端实现列表左滑删除的示例代码
Jul 04 #Javascript
jQuery删除/清空指定元素的所有子节点实例代码
Jul 04 #jQuery
小程序中canvas的drawImage方法参数使用详解
Jul 04 #Javascript
vue如何限制只能输入正负数及小数
Jul 04 #Javascript
You might like
php二维数组用键名分组相加实例函数
2013/11/06 PHP
php获取远程文件大小
2015/10/20 PHP
JavaScript 数组的 uniq 方法
2008/01/23 Javascript
jquery中focus()函数实现当对象获得焦点后自动把光标移到内容最后
2013/09/29 Javascript
javascript浏览器窗口之间传递数据的方法
2015/01/20 Javascript
js实现YouKu的漂亮搜索框效果
2015/08/19 Javascript
[原创]Bootstrap 中下拉菜单修改成鼠标悬停直接显示
2016/04/14 Javascript
jQuery的文档处理程序详解
2016/05/10 Javascript
Nodejs连接mysql并实现增、删、改、查操作的方法详解
2018/01/04 NodeJs
React组件refs的使用详解
2018/02/09 Javascript
vue使用一些外部插件及样式的配置代码
2019/11/18 Javascript
vue使用nprogress实现进度条
2019/12/09 Javascript
vue中echarts图表大小适应窗口大小且不需要刷新案例
2020/07/19 Javascript
在Vue 中实现循环渲染多个相同echarts图表
2020/07/20 Javascript
vue.js 输入框输入值自动过滤特殊字符替换中问标点操作
2020/08/31 Javascript
[03:16]DOTA2完美大师赛主赛事首日集锦
2017/11/23 DOTA
Python操作MySQL简单实现方法
2015/01/26 Python
Python3.x版本中新的字符串格式化方法
2015/04/24 Python
给大家整理了19个pythonic的编程习惯(小结)
2019/09/25 Python
基于python实现百度语音识别和图灵对话
2020/11/02 Python
PyTorch 中的傅里叶卷积实现示例
2020/12/11 Python
css3 clip实现圆环进度条的示例代码
2018/02/07 HTML / CSS
苹果中国官方网站:Apple中国
2016/07/22 全球购物
Mixbook加拿大:照片书,照片卡,剪贴簿,年历和日历
2017/02/21 全球购物
校园十大歌手策划书
2014/02/01 职场文书
餐厅执行经理岗位职责范本
2014/02/26 职场文书
毕业生实习期转正自我鉴定
2014/09/26 职场文书
2014年外联部工作总结
2014/11/17 职场文书
婚礼答谢词
2015/01/04 职场文书
学校光盘行动倡议书
2015/04/28 职场文书
办公室年度工作总结2015
2015/05/21 职场文书
优秀班主任工作总结2015
2015/05/25 职场文书
《角的初步认识》教学反思
2016/02/17 职场文书
一波干货,会议主持词开场白范文
2019/05/06 职场文书
Python中的turtle画箭头,矩形,五角星
2022/03/16 Python
Win11自动黑屏怎么办 Win11自动黑屏设置教程
2022/07/15 数码科技