什么是跨域,为什么浏览器会禁止跨域,以及实现跨域的几种方式
首先我们来想一想
为什么会有跨域这个名词的出现呢?
跨域又是什么呢?为何要跨域?
浏览器的同源策略又是什么?怎么解决?
jsonp又是什么?
跨域的原理又是什么呢?
名词解释:
1、什么是跨域
跨域的产生来源于现代浏览器所通用的‘同源策略’,所谓同源策略,是指只有在地址的:
1. 协议名 https,http
2. 域名 http://a.study.cn http://study.cn
3. 端口名 http://study.cn:8080/json/jsonp/jsonp.html study.cn/json/jsonp/jsonp.html
均一样的情况下,才允许访问相同的cookie、localStorage或是发送Ajax请求等等。若在不同源的情况下访问,就称为跨域。
例如a.cn下面的js不能调用b.cn中的js,对象或数据(因为a.cn和b.cn是不同域),所以跨域就出现了.
同源策略:
请求的url地址,必须与浏览器上的url地址处于同域上,也就是域名,端口,协议相同.
比如:我在本地上的域名是study.cn,请求另外一个域名一段数据
这个时候在浏览器上会报错:
这个就是同源策略的保护,如果浏览器对javascript没有同源策略的保护,那么一些重要的机密网站将会很危险~
1 2 3 4 5 6 | 请求地址 形式 结果 http: //study.cn/test/a.html 同一域名,不同文件夹 成功 http: //study.cn/json/jsonp/jsonp.html 同一域名,统一文件夹 成功 http: //a.study.cn/json/jsonp/jsonp.html 不同域名,文件路径相同 失败 http: //study.cn:8080/json/jsonp/jsonp.html 同一域名,不同端口 失败 https: //study.cn/json/jsonp/jsonp.html 同一域名,不同协议 失败 |
jsonp:
jsonp 全称是JSON with Padding,是为了解决跨域请求资源而产生的解决方案,是一种依靠开发人员创造出的一种非官方跨域数据交互协议。
一个是描述信息的格式,一个是信息传递双方约定的方法。
jsonp的产生:
1.AJAX直接请求普通文件存在跨域无权限访问的问题,不管是静态页面也好.
2.不过我们在调用js文件的时候又不受跨域影响,比如引入jquery框架的,或者是调用相片的时候
3.凡是拥有scr这个属性的标签都可以跨域例如<script><img><iframe>
4.如果想通过纯web端跨域访问数据只有一种可能,那就是把远程服务器上的数据装进js格式的文件里.
5.而json又是一个轻量级的数据格式,还被js原生支持
6.为了便于客户端使用数据,逐渐形成了一种非正式传输协议,人们把它称作JSONP,该协议的一个要点就是允许用户传递一个callback 参数给服务端,
demo1:基于script标签实现跨域
举个例子:我在http://study.cn/json/jsonp/jsonp_2.html下请求一个远程的js文件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | <!DOCTYPE html> <html> <head> <meta charset= "UTF-8" > <title>Insert title here</title> <script type= "text/javascript" > var message = function(data) { alert(data[1].title); }; </script> <script type= "text/javascript" src= "http://web.cn/js/message.js" ></script> </head> <body> <div id= 'testdiv' ></div> </body> </html> |
远程的message.js文件是
message([ {"id":"1", "title":"天津新闻联播,雷人搞笑的男主持人"}, {"id":"2", "title":"楼市告别富得流油 专家:房价下跌是大概率事件"}, {"id":"3", "title":"法国人关注时事 八成年轻人每天阅读新闻"}, {"id":"4", "title":"新闻中的历史,历史中的新闻"}, {"id":"5", "title":"东阳新闻20140222"}, {"id":"6", "title":"23个职能部门要增加新闻发布频次"}, {"id":"7", "title":"《贵州新闻联播》 中国美丽乡村"}, {"id":"8", "title":"朝韩离散家属团聚首轮活动结束"}, {"id":"9", "title":"索契冬奥会一天曝出两例兴奋剂事件"}, {"id":"10", "title":"今天中国多地仍将出现中度霾"} ]);
这个时候我们得到的相应头是:
这样就实现跨域成功了,因为服务端返回数据时会将这个callback参数(message)作为函数名来包裹住JSON数据,这样客户端就可以随意定制自己的函数来自动处理返回数据了。
demo2: 基于script标签实现跨域
让远程js知道它应该调用的本地函数叫什么名字,只要服务端提供的js脚本是动态生成的就好了,这样前台只需要传一个callback参数过去告诉服务端,我需要XXX代码,于是服务端就会得到相应了.
例如 在http://study.cn/json/jsonp/jsonp_3.html页面请求 http://192.168.31.137/train/test/jsonpthree
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 | <!DOCTYPE html> <html> <head> <meta charset= "UTF-8" > <title>Insert title here</title> <script type= "text/javascript" > var messagetow = function(data){ alert(data); }; var url = "http://192.168.31.137/train/test/jsonpthree?callback=messagetow" ; var script = document.createElement( 'script' ); script.setAttribute( 'src' , url); document.getElementsByTagName( 'head' )[0].appendChild(script); </script> </head> <body> </body> </html> |
得到的响应头是:
demo3: 基于jquery跨域
那么如何用jquery来实现我们的跨域呢???jquery已经把跨域封装到ajax上了,而且封装得非常的好,使用起来也特别方便
如果是一般的ajax请求:
1 2 3 4 5 6 7 8 9 10 11 | $.ajax({ url: 'http://192.168.31.137/train/test/testjsonp' , type : 'get' , dataType : 'text' , success:function(data){ alert(data); }, error:function(data){ alert(2); } }); |
那么在浏览器中会报错:
jsonp形式的ajax请求:并且通过get请求的方式传入参数,注意:跨域请求是只能是get请求不能使用post请求
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 | <!DOCTYPE html> <html> <head> <meta charset= "UTF-8" > <title>Insert title here</title> <script type= "text/javascript" src= "./js/jquery.js" ></script> <script type= "text/javascript" > $(document).ready(function(){ var name = 'chenshishuo' ; var sex = 'man' ; var address = 'shenzhen' ; var looks = 'handsome ' ; $.ajax({ type : 'get' , url: 'http://192.168.31.137/train/test/testjsonp' , data : { name : name, sex : sex, address : address, looks : looks, }, cache : false , jsonp: "callback" , jsonpCallback: "success" , dataType : 'jsonp' , success:function(data){ alert(data); }, error:function(data){ alert( 'error' ); } }); }); </script> </head> <body> <input id= 'inputtest' value= '546' name= 'inputtest' > <div id= 'testdiv' ></div> </body> </html> |
jsonp 传递给请求处理程序或页面的,用以获得jsonp回调函数名的参数名(默认为:callback)
jsonpCallback 自定义的jsonp回调函数名称,默认为jQuery自动生成的随机函数名
看看请求头和相应头吧
请求头:jquery会自动带入callback参数,当服务端获取到这个参数后,返回回来.(响应头)
现在是不是明白了跨域的基本原理,和基本的使用方法呢??
上面我们说到img中的src可以自动调用远程图片的(这个比较简单我在这里就不说了)
还有ifram请求:
基于iframe实现的跨域要求两个域具有aa.xx.com,bb.xx.com 这种特点,
也就是两个页面必须属于一个基础域(例如都是xxx.com),使用同一协议和同一端口,这样在两个页面中同时添加document.domain,就可以实现父页面调用子页面的函数
要点就是 :通过修改document.domain来跨子域
demo4: 通过iframe来跨子域
http://a.study.cn/a.html 请求 http://b.study.cn/b.html
在a.html:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | <!DOCTYPE html> <html> <head> <meta charset= "UTF-8" > <title>Insert title here</title> <script type= "text/javascript" > document.domain = 'study.cn' ; function test() { alert(document.getElementById( 'a' ).contentWindow); } </script> </head> <body> <iframe id= 'a' src= 'http://b.study.cn/b.html' onload= 'test()' > </body> </html> |
在b.html:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | <!DOCTYPE html> <html> <head> <meta charset= "UTF-8" > <title>Insert title here</title> <script type= "text/javascript" > document.domain = 'study.cn' ; </script> </head> <body> 我是b.study.cn的body </body> </html> |
运行效果截图:
我们就可以通过js访问到iframe中的各种属性和对象了
如果你想在http://a.study.cn/a.html页面中通过ajax直接请求页面http://b.study.cn/b.html,即使你设置了相同的document.domain也还是不行的.
所以修改document.domain的方法只适用于不同子域的框架(父类与子类)间的交互。
如果想通过使用ajax的方法去与不同子域间的数据交互或者是js调用,只有两种方法,一种是使用jsonp的方法外,还有一种是使用iframe来做一个代理。
原理就是让这个 iframe载入一个与你想要通过ajax获取数据的目标页面处在相同的域的页面,所以这个iframe中的页面是可以正常使用ajax去获取你要的数据 的,
然后就是通过我们刚刚讲得修改document.domain的方法,让我们能通过js完全控制这个iframe,这样我们就可以让iframe去发 送ajax请求,然后收到的数据我们也可以获得了。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通