本文主要介绍跨域问题介绍并提供了四种解决办法。
准确的来说是浏览器存在跨域问题,浏览器为了安全考虑,也就是同源策略的限制,会拒绝跨域请求。
在这里同源策略是指:请求时拥有相同的协议、域名、端口,其中只要有一项不满足就被视为跨域。
主机(http://www.smz.com) | 是否跨域 | 原因 |
---|---|---|
https://www.smz.com | 是 | 协议不同 |
http://www.smz.com:8001 | 是 | 端口不同 |
http://www.baidu.com | 是 | 域名不同 |
http://www.smz.com/index.html | 否 | 符合三个条件 |
跨域产生:
前端,这里直接请求,就会产生跨域问题
fetch('http://localhost:3000/api/json').then(res=>res.json()).then(res=>{
console.log(res)
})
后端,这里不做处理直接返回数据
app.get('/api/json',(req,res)=>{
res.send({name:'smz'})
})
这里浏览器报错,是因为端口不一致导致跨域
这里值得注意的是,在发送跨域请求时,浏览器会先发送一个OPTIONS
请求(预检请求),用来获取服务器对跨域的支持情况。以下几种跨域请求都会触发预检请求,如果服务端在请求头中返回了适当的跨域头,则允许发送实际请求,否则拒绝。而其他简单请求则会正常发送。
GET
、HEAD
、POST
。Accept
、Accept-Language
、Content-Language
、Content-Type
(仅限于application/x-www-form-urlencoded
、multipart/form-data
、text/plain
)。这种方式需要前后端一起协商解决,原理就是动态创建script标签,因为script标签src属性不受跨域限制。但是script标签只能发送get请求,是不安全的。
后端返回的是一个函数 这个函数前端需先定义好 他会把返回的值注入到这个函数的参数中,其中具有代表性的就是百度搜索跨域请求就是jsonp实现的。
前端:
<script>
let text = document.querySelector('#text')
// 动态创建script标签,添加scr请求地址拼接函数名称
const jsonp = (name) =>{
let script = document.createElement('script')
script.src = 'http://localhost:3000/api/jsonp?callback=' + name
document.body.appendChild(script)
return new Promise((resolve)=>{
window[name] = (data) =>{
resolve(data)
}
})
}
// 利用函数接收返回的参数
jsonp(`callback${new Date().getTime()}`).then(res=>{
text.textContent = res
console.log(res)
})
</script>
后端:这里用Node
const express = require('express')
const app = express()
app.get('/api/jsonp',(req,res)=>{
const {callback} = req.query // 函数名
res.send(`${callback}('hello jsonp')`) //返回并将数据充当函数参数返回
})
app.listen(3000,()=>{
console.log('server is running')
})
跨域问题可以通过以下几种方式解决:
JSONP(JSON with Padding):通过动态创建
需要注意的是,跨域问题只存在于浏览器中,对于服务器端来说并不存在跨域问题。因此,以上解决方案都是在浏览器端实现的。
利用vite或者webpack通过代理,将请求转发到对应的服务器上。
vite或者webpack配置文件,这里用vite
import {defineConfig} from 'vite'
export default defineConfig({
server:{
proxy:{
'/api':{
target:'http://localhost:3000',
changeOrigin:true,// 开启跨域
}
}
}
})
请求:
fetch('/api/json').then(res=>res.json()).then(res=>{
console.log(res)
})
这里浏览器显示的是前端端口发送的请求,实际上已经通过请求转发给3000端口了,是vite开启了一个Node服务,在Node服务这里做了请求转发的操作。
只在开发环境生效,项目上线得用nginx
后端设置跨域很简单,只要在返回请求头中带入允许跨域的接口地址就行Access-Control-Allow-Origin
将其值设置为*可以放行全部,但是不建议这样放行全部,可以指定ip。
res.setHeader('Access-Control-Allow-Origin','*')
其实在产生跨域问题时,浏览器就提示我们这样的做法了
在项目上线时,通过nginx代理进行请求转发。
在nginx配置文件nginx.conf中加入代理配置
location /api {
proxy_pass http://127.0.0.1:5000;
}
本文记录了,跨域问题在前后端的解决方法,以及不同环境下的解决方案。