同源策略规定了若是两个 url 的协议、域名、端口中有任何一个不等,就认定它们跨源了。javascript
JSONP实现跨域请求的原理简单的说,就是动态建立<script>标签,而后利用<script>的src 不受同源策略约束来跨域获取数据。前端
JSONP 由两部分组成:回调函数和数据
。回调函数是当响应到来时应该在页面中调用的函数。回调函数的名字通常是在请求中指定的。而数据就是传入回调函数中的 JSON 数据。
动态建立<script>标签,设置其src,回调函数在src中设置:vue
var script = document.createElement("script"); script.src = "https://api.douban.com/v2/book/search?q=javascript&count=1&callback=handleResponse"; document.body.insertBefore(script, document.body.firstChild);
在页面中,返回的JSON做为参数传入回调函数中,咱们经过回调函数来来操做数据。java
function handleResponse(response){ // 对response数据进行操做代码 console.log(response) }
JSONP目前仍是比较流行的跨域方式,虽然JSONP使用起来方便,可是也存在一些问题:
首先, JSONP 是从其余域中加载代码执行。若是其余域不安全,极可能会在响应中夹带一些恶意代码,而此时除了彻底放弃 JSONP 调用以外,没有办法追究。所以在使用不是你本身运维的 Web 服务时,必定得保证它安全可靠。
JSONP 具备直接访问响应文本的优势,可是要想确认 JSONP 是否请求失败并不容易,由于 script 标签的 onerror 事件还未获得浏览器普遍的支持,此外它仅能支持 GET 方式调用。node
整个CORS通讯过程,都是浏览器自动完成,不须要用户参与。对于开发者来讲,CORS通讯与同源的AJAX通讯没有差异,代码彻底同样。浏览器一旦发现AJAX请求跨源,就会自动添加一些附加的头信息,有时还会多出一次附加的请求,但用户不会有感受。react
所以,实现CORS通讯的关键是服务器。只要服务器实现了CORS接口,就能够跨源通讯。webpack
一个经常使用的完整的跨域头:ios
let express=require("express"); let app=express(); app.use(function(req,res,next){ //若是在webpack里配置了代理,那么这些响应头都不要了 //只容许8080访问 res.header('Access-Control-Allow-Origin','http://localhost:8080'); //服务容许客户端发的方法 res.header('Access-Control-Allow-Methods','GET,POST,DELETE,PUT'); //服务器容许的请求头 res.header('Access-Control-Allow-Headers','Content-Type,Accept'); //跨域携带cookie 容许客户端把cookie发过来 res.header('Access-Control-Allow-Credentials','true'); //若是请求的方法是OPTIONS,那么意味着客户端只要响应头,直接结束响应便可 if(req.method == 'OPTIONS'){ res.end(); }else{ next(); } }); app.listen(3000);
有一位网友写的很清楚,在此再也不赘述,你们能够直接查看他的文章:
跨域二三事,很难详细的跨域分析nginx
在实际工做中,先后端配合并非那么默契,若是后台只给我接口,不能修改后台,怎么跨域?
在vue项目和react项目中的config文件中,都有一个proxy代理设置,这个就是用来在开发环境下进行跨域的。对其进行设置就能实现跨域。
经过vue-cli脚手架搭建出来的项目,修改config文件夹下的index.js中的proxyTable就能实现:web
module.exports = { dev: { env: { NODE_ENV: '"development"' }, //proxy // 只能在开发环境中进行跨域,上线了要进行反向代理nginx设置 proxyTable: { //这里理解成用‘/api’代替target里面的地址,后面组件中咱们掉接口时直接用api代替 好比我要调用'http://40.00.100.100:3002/user/add',直接写‘/api/user/add’便可 '/api': { target: 'http://news.baidu.com',//你要跨域的网址 好比 'http://news.baidu.com', secure: true, // 若是是https接口,须要配置这个参数 changeOrigin: true,//这个参数是用来回避跨站问题的,配置完以后发请求时会自动修改http header里面的host,可是不会修改别的 pathRewrite: { '^/api': '/api'//路径的替换规则 //这里的配置是正则表达式,以/api开头的将会被用用‘/api’替换掉,假如后台文档的接口是 /api/list/xxx //前端api接口写:axios.get('/api/list/xxx') , 被处理以后实际访问的是:http://news.baidu.com/api/list/xxx } } },
npm run dev
前端端口号通常是:http://localhost:8080
proxyTable:{}
这段代码,替换掉便可:module.exports = { dev: { proxyTable: { '/api': { target: 'http://localhost:8000', secure: true, changeOrigin: true, pathRewrite: { '^/api': '/api' } } },
注意本身须要在当前文件夹下提早准备一个list.json的文件,用来读取数据,返回数据。fs.readFile('./list.json','utf8',cb)
let express = require('express'); let app = express(); let fs = require('fs'); let list = require('./list'); let bodyParser = require('body-parser'); app.use(bodyParser.json()); app.use(express.static(__dirname)); //read function read(cb) { //用来读取数据的,注意本身在mock文件夹下准备一些数据 fs.readFile('./list.json','utf8',function (err,data) { if(err || data.length === 0){ cb([]); // 若是有错误 或者文件没长度 就是空数组 }else{ cb(JSON.parse(data)); // 将读出来的内容转化成对象 } }) } //write function write(data,cb) { // 写入内容 fs.writeFile('./list.json',JSON.stringify(data),cb) } // 注意 没有设置跨域头 app.get('/api/list',function (req,res) { res.json(list); }); app.listen(8000,()=>{ console.log('8000 is ok'); });
import axios from 'axios'; axios.interceptors.response.use((res)=>{ return res.data; // 在这里统一拦截结果 把结果处理成res.data }); export function getLists() { return axios.get('/api/list'); }
随便在一个文件中引入api测试一下 打印出来接口返回的数据 import {getLists} from '../../api/index' export default { async created(){ let dataList=await getLists(); console.log(dataList,"我请求的数据"); },
上线要进行nginx反向代理设置,同时应区分环境变量,具体设置请看图片: