nginx解决跨域问题

一. 产生跨域的原因

1.浏览器限制

2.跨域

3.XHR(XMLHttpRequest)请求

二. 解决思路

解决跨域有多重,在这里主要讲用nginx解决跨域

1.JSONP

2.nginx代理

3.浏览器禁止检查跨域

三. 下载安装nginx

  • 选择其中一个版本下载,再解压即可使用
  • 在nginx目录下输入nginx -v,若出现版本号,则安装成功
nginx解决跨域问题

四. nginx反向代理解决跨域(客户端解决跨域)

1.我们使用jquery的ajax发送请求,node开启后台服务

前端代码:

利用jQuery的ajax api发送请求

 <button id="getOK">发送请求OK(客户端解决跨域问题)</button>
 <button id="getNO">发送请求NO(客户端解决跨域问题)</button>
 <script>
 $(document).ready(function () {
 $('#getOK').click(function () {
 $.ajax({
 url:'http://localhost:3000/ok',
 success:function(res) {
 console.log("success",res)
 },
 error:function(err) {
 console.log('fail',err)
 }
 })
 })
 $('#getNO').click(function () {
 $.ajax({
 url:'http://localhost:3000/no',
 success:function(res) {
 console.log("success",res)
 },
 error:function(err) {
 console.log('fail',err)
 }
 })
 })
 }) 
 </script>

后端代码:

利用node的express框架开启服务,并根据url返回json格式的数据,

设置这么多接口的目的是为了后面匹配nginx的location配置的

const express = require('express')
const cookieParser = require('cookie-parser')
var app = express()
var router = express.Router()
router.get('/ok',function (req,res) {
 res.json({
 code:200,
 msg:"isOK"
 }) 
})
router.get('/ok/son',function (req,res) {
 res.json({
 code:200,
 msg:"isOKSon"
 }) 
})
router.get('/ok2',function (req,res) {
 res.json({
 code:200,
 msg:"isOK2"
 }) 
})
router.get('/no',function (req,res) {
 res.json({
 code:200,
 msg:"isNO"
 }) 
})
router.get('/no/son',function (req,res) {
 res.json({
 code:200,
 msg:"isNOSON"
 }) 
})
router.get('/no/son2',function (req,res) {
 res.json({
 code:200,
 msg:"isNOSON2"
 }) 
})
app.use(router)
app.use(cookieParser)
app.listen(3000,function () {
 console.log('listen in 3000')
})

然后开启node服务

nginx解决跨域问题

现在可以测试下接口

nginx解决跨域问题

可以看出,node服务成功开启

现在可以尝试不开启nginx服务直接发送ajax请求会出现什么情况

(注意:发送ajax请求需要以服务器方式打开网页,不能以文件形式)

nginx解决跨域问题

如图,在5500端口请求3000端口出现了跨域问题,这时候就可以开启nginx服务并配置location进行解决

2.配置nginx进行反向代理解决跨域

反向代理的原理就是讲前端的地址和后端的地址用nginx转发到同一个地址下,如5500端口和3000端口都转到3003端口下,具体配置如下:

  • 打开nginx目录下的conf目录里面nginx.conf
  • 为了方便以后测试,我们将配置分离开来,弄成多个文件
  • 在nginx.conf的http对象的最后加上include ../vhost/test.conf;(注意要最后加上分号)
  • 这样就可以在test.conf下单独配置了

具体的location配置规则如下:

server
{
 listen 3003;
 server_name localhost;
 ## = /表示精确匹配路径为/的url,真实访问为http://localhost:5500
 location = / {
 proxy_pass http://localhost:5500;
 }
 ## /no 表示以/no开头的url,包括/no1,no/son,或者no/son/grandson
 ## 真实访问为http://localhost:5500/no开头的url
 ## 若 proxy_pass最后为/ 如http://localhost:3000/;匹配/no/son,则真实匹配为http://localhost:3000/son
 location /no {
 proxy_pass http://localhost:3000;
 }
 ## /ok/表示精确匹配以ok开头的url,/ok2是匹配不到的,/ok/son则可以
 location /ok/ {
 proxy_pass http://localhost:3000;
 }
}

上面代码的意思是将localhost:3003转发为location:5500,也就是说现在访问localhost:3003实际上是访问location:5500,而访问localhost:3003/no则是访问localhost:3000,并以no开头的url

  • 现在我们可以开启nginx服务了,在nginx目录下使用start nginx即可开启服务
nginx解决跨域问题
  • 每次修改配置都需要执行nginx -s reload命令才能生效
nginx解决跨域问题

现在修改前端代码,将之前请求的接口的端口换为3003,如下:

$('#getOK').click(function () {
 $.ajax({
 url:'http://localhost:3003/ok',
 success:function(res) {
 console.log("success",res)
 },
 error:function(err) {
 console.log('fail',err)
 }
 })
 })

在浏览器访问的也不算location:5500,而是localhost:3003了,再次发送请求也不会出现跨域问题了,因为他们都是同一个域了,这就是nginx反向代理

nginx解决跨域问题

五. 后端配置nginx解决跨域(服务端解决跨域)

1. 依旧是ajax+node

这是前端代码

$(document).ready(function () {
 $('#get').click(function () {
 $.ajax({
 url:'http://localhost:3002/ok',
 // 带cookies的请求
 xhrFields:{
 withCredentials:true
 },
 success:function(res) {
 console.log("success",res)
 },
 error:function(err) {
 console.log('fail',err)
 }
 })
 })
 })

后端代码同前面

还有nginx配置如下:

server
{
 listen 3002;
 server_name localhost;
 location /ok {
 proxy_pass http://localhost:3000;
 # 指定允许跨域的方法,*代表所有
 add_header Access-Control-Allow-Methods *;
 # 预检命令的缓存,如果不缓存每次会发送两次请求
 add_header Access-Control-Max-Age 3600;
 # 带cookie请求需要加上这个字段,并设置为true
 add_header Access-Control-Allow-Credentials true;
 # 表示允许这个域跨域调用(客户端发送请求的域名和端口) 
 # $http_origin动态获取请求客户端请求的域 不用*的原因是带cookie的请求不支持*号
 add_header Access-Control-Allow-Origin $http_origin;
 # 表示请求头的字段 动态获取
 add_header Access-Control-Allow-Headers 
 $http_access_control_request_headers;
 # OPTIONS预检命令,预检命令通过时才发送请求
 # 检查请求的类型是不是预检命令
 if ($request_method = OPTIONS){
 return 200;
 }
 }
}

发送预检命令的是非简单请求,具体可以看慕课网ajax跨域完全讲解

实际上不是非简单请求的且不带cookie只需2个字段即可解决跨域

add_header Access-Control-Allow-Methods *;

add_header Access-Control-Allow-Origin $http_origin;

  • 具体效果如下图:
nginx解决跨域问题

这时只需改ajax请求的端口接口,无需修改前端服务器的地址

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,340评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,762评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,329评论 0 329
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,678评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,583评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 47,995评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,493评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,145评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,293评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,250评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,267评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,973评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,556评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,648评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,873评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,257评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,809评论 2 339

推荐阅读更多精彩内容