Vue解決跨域問題常見方法詳解
跨域報錯是前端開發(fā)中非常經(jīng)典的一個錯誤,報錯如下
Access to XMLHttpRequest at '......' from origin
'......' has been blocked by CORS policy:
No 'Access-Control-Allow-Origin' header is present on the requested resource.
跨域錯誤源自于瀏覽器的同源策略,想要解決跨域首先要知道什么是同源策略?
一、同源策略?
同源策略:著名的安全策略,URL有三個基本組成部分:協(xié)議+域名或ip+端口,三個必須完全相同稱之為同源,不同源的稱之為跨域
URL 與 URL 對比:
http://localhost:3000/
https://localhost:3000/ 不同源:協(xié)議不同
http://localhost:3000/
http://127.0.0.1:3000/ 不同源:域名或ip不同
http://localhost:3000/
http://localhost:3001/ 不同源:端口不同
http://localhost:3000/
http://localhost:3000/ 同源
http://127.0.0.1:3000/
http://127.0.0.1:3000/ 同源
注意:同源策略不是服務器行為,而是瀏覽器的行為,服務器會正常響應請求,但是如果不同源會被瀏覽器攔截
搭建express服務器——演示
搭建一個express服務器用來演示跨域報錯
安裝express
?npm i express
在app.js文件中
?let express = require('express') ?let app = express() ?? ?app.listen(3000, () => { ? ? ?console.log('服務器已啟動..端口3000.') ?}) ?? ?app.use(express.static('./views')) ?? ?app.get('/getTest', (req, res) => { ? ? ?console.log(req.query) ? ? ?res.send(req.query) ?})
在views/index.html文件中
?<!DOCTYPE html> ?<html> ?? ?<head> ? ? ?<meta charset="UTF-8"> ? ? ?<title>Document</title> ? ? ?<script src="https://unpkg.com/axios/dist/axios.min.js"></script> ? ? ?<script> ? ? ? ? ?function getIpTest() { ? ? ? ? ? ? ?axios({ ? ? ? ? ? ? ? ? ?method: "get", ? ? ? ? ? ? ? ? ?url: "http://127.0.0.1:3000/getTest", ? ? ? ? ? ? ? ? ?params: { uid: 123 }, ? ? ? ? ? ? ?}).then((res) => { ? ? ? ? ? ? ? ? ?console.log(res.data); ? ? ? ? ? ? ?}); ? ? ? ? ?} ? ? ? ? ?function getDnameTest() { ? ? ? ? ? ? ?axios({ ? ? ? ? ? ? ? ? ?method: "get", ? ? ? ? ? ? ? ? ?url: "http://localhost:3000/getTest", ? ? ? ? ? ? ? ? ?params: { uid: 123 }, ? ? ? ? ? ? ?}).then((res) => { ? ? ? ? ? ? ? ? ?console.log(res.data); ? ? ? ? ? ? ?}); ? ? ? ? ?} ? ? ?</script> ?</head> ?? ?<body> ? ? ?<button οnclick="getIpTest()">getIpTest</button> ? ? ?<button οnclick="getDnameTest()">getDnameTest</button> ?</body> ?? ?</html>
打開瀏覽器訪問http://127.0.0.1:3000/
(1)調(diào)用getIpTest發(fā)送請求不會報錯,因為瀏覽器地址欄訪問的服務器是http://127.0.0.1:3000/** ,而方法內(nèi)發(fā)送請求的URL也是http://127.0.0.1:3000/** ,視為同源
(2)調(diào)用getDnameTest發(fā)送請求報錯,因為瀏覽器地址欄訪問的服務器是http://127.0.0.1:3000/** ,而方法內(nèi)發(fā)送請求的URL也是http://localhost:3000/** ,視為跨域
Access to XMLHttpRequest at 'http://localhost:3000/......' from origin
'http://127.0.0.1:3000' has been blocked by CORS policy:
No 'Access-Control-Allow-Origin' header is present on the requested resource.
打開瀏覽器訪問http://localhost:3000/
(3)調(diào)用getDnameTest發(fā)送請求不會報錯,因為瀏覽器地址欄訪問的服務器是http://localhost:3000/ ,而方法內(nèi)發(fā)送請求的URL也是http://localhost:3000/ ,視為同源
(4)調(diào)用getIpTest發(fā)送請求報錯,因為瀏覽器地址欄訪問的服務器是http://localhost:3000/ ,而方法內(nèi)發(fā)送請求的URL也是**http://127.0.0.1:3000/** ,視為跨域
Access to XMLHttpRequest at 'http://127.0.0.1:3000/......' from origin
'http://localhost:3000' has been blocked by CORS policy:
No 'Access-Control-Allow-Origin' header is present on the requested resource.
二、vue如何處理跨域問題?
(1)創(chuàng)建vue項目,安裝axios模塊
?vue create app //創(chuàng)建vue ?npm install axios vue-axios //安裝axios
在main.js中
?import axios from 'axios' ?import VueAxios from 'vue-axios' ?Vue.use(VueAxios, axios)
在views/About.vue中
?<template> ? ?<div> ? ? ?<button @click="getTest()">getTest</button> ? ?</div> ?</template> ?? ?<script> ?export default { ? ?methods: { ? ? ?getTest() { ? ? ? ?this.axios({ ? ? ? ? ?method: "get", ? ? ? ? ?url: "http://127.0.0.1:3000/getTest", ? ? ? ? ?params: { uid: 123 }, ? ? ? ?}).then((res) => { ? ? ? ? ?console.log(res.data); ? ? ? ?}); ? ? ?}, ? ?}, ?}; ?</script>
腳手架項目端口是8080而請求的express服務器端口是3000,不滿足同源策略,發(fā)送請求報跨域錯誤
Access to XMLHttpRequest at 'http://127.0.0.1:3000/......' from origin
'http://127.0.0.1:8080' has been blocked by CORS policy:
No 'Access-Control-Allow-Origin' header is present on the requested resource.
(2)解決辦法
1.配置代理服務器
在vue.config.js文件中
(修改后需要重啟腳手架項目)
?const { defineConfig } = require('@vue/cli-service') ?module.exports = defineConfig({ ? ?transpileDependencies: true, ? ?devServer: { ? ? ?//配置http-proxy代理方式跨域 ? ? ?proxy: { ? ? ? ?// 自定義請求的開頭,使用代理方式處理/demo開頭的請求,/xxx可以自定義 ? ? ? ?"/demo": { ? ? ? ? ?// 往哪個服務器發(fā)請求 ? ? ? ? ?target: "http://127.0.0.1:3000", ? ? ? ? ?pathRewrite: { ? ? ? ? ? ?// ^代表字符串開頭,實際發(fā)送請求時,會把請求開頭的/demo刪除 ? ? ? ? ? ?// 因為/demo并不是請求的一部分,只是個代理的標識 ? ? ? ? ? ?"^/demo": "", ? ? ? ? ?}, ? ? ? ?}, ? ? ? ?// 如果有其他網(wǎng)址也需要跨域則繼續(xù)配置 ? ? ? ?// "/其他的...": { ? ? ? ?// target: "其他的請求地址", ? ? ? ?// pathRewrite: { ? ? ? ?// "^/其他的...": "", ? ? ? ?// }, ? ? ? ?// }, ? ? ?}, ? ?}, ?})
在views/About.vue文件中
<template> ? ?<div> ? ? ?<button @click="getTest()">getTest</button> ? ?</div> ?</template> ?? ?<script> ?export default { ? ?methods: { ? ? ?getTest() { ? ? ? ?this.axios({ ? ? ? ? ?method: "get", ? ? ? ? ?// 在原來的url上去掉http://127.0.0.1:3000換成/demo ? ? ? ? ?url: "/demo/getTest", ? ? ? ? ?params: { uid: 123 }, ? ? ? ?}).then((res) => { ? ? ? ? ?console.log(res.data); ? ? ? ?}); ? ? ?}, ? ?}, ?}; ?</script>
原理:
跨域是瀏覽器的安全策略,服務器和服務器之間發(fā)送請求沒有跨域
在啟動腳手架的時候會啟動一個內(nèi)置web服務器
請求的時候瀏覽器實際并沒有直接和需要請求的服務器通信,而是通過內(nèi)置的web服務器在中轉(zhuǎn)
注意:
項目上線需要把打包后的文件放在服務器上運行,而不是啟動腳手架運行,也就沒有內(nèi)置web服務器做代理,所以此方式只適用于開發(fā)測試階段
上線時一般后端解決跨域問題
需要使用nginx代理或者服務器配置cors(每種語言有自己不同的配置方式)
2.express處理跨域
express中處理跨域需要使用cors模塊
?npm i cors
在app.js文件中
?let express = require('express') ?// 引入cors模塊 ?var cors = require("cors"); ?let app = express() ?? ?app.listen(3000, () => { ? ? ?console.log('服務器已啟動...') ?}) ?? ?// 配置跨域 ?app.use(cors({ ? ? ?// 允許跨域的服務器地址,可以寫多個 ? ? ?origin: ['http://localhost:8080', 'http://127.0.0.1:8080'], ? ? ?// 使用cookie時需要設(shè)置為true ? ? ?credentials: true ?})); ?? ?app.use(express.static('./views')) ?? ?app.get('/getTest', (req, res) => { ? ? ?console.log(req.query) ? ? ?res.send(req.query) ?})
相關(guān)文章
vue.js el-tooltip根據(jù)文字長度控制是否提示toolTip問題
這篇文章主要介紹了vue.js el-tooltip根據(jù)文字長度控制是否提示toolTip問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-02-02vue-extend和vue-component注冊一個全局組件方式
這篇文章主要介紹了vue-extend和vue-component注冊一個全局組件方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-11-11vue2項目導出操作實現(xiàn)方法(后端接口導出、前端直接做導出)
這篇文章主要給大家介紹了關(guān)于vue2項目導出操作實現(xiàn)方法的相關(guān)資料,文中介紹的是后端接口導出、前端直接做導出,通過代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考借鑒價值,需要的朋友可以參考下2024-05-05