前端跨域的6种方式

同源策略

  • 浏览器对不同源的脚本或者文本的访问方式进行的限制,就是一种浏览器的安全机制。
  • 同源:相同的协议、域名、端口号
    同源策略限制的不同源之间的交互主要针对的是js中的XMLHttpRequest等请求

跨域

  • 前置条件是我们在WEB服务器或者服务端脚本中设置ACCESS-CONTROL-ALLOW-ORIGIN头部,如果设置了这些头部并允许某些域名跨域访问,则浏览器就会跳过同源策略的限制返回对应的内容。

1.JSONP (需要后台配合)------(向后台请求数据的跨域)

  • jsonp的原理:
    1.通过动态插入<script>标签来实现跨域资源访问的
    2.server定义好的那个用来设置返回数据中执行函数名的那个函数,就叫jsonpCallback
    3.jsonpCallback后面跟的value必须是全局作用域下的一个函数(和客户端注册的callback名称一致即可)
    4.server返回的数据格式是固定的functionName(/* jsonData */)
 <script type="text/javascript">
    function jsonpCallback(result) { 
      alert(result.msg);
    }
</script>
<script type="text/javascript" src="http://crossdomain.com/jsonServerResponse?jsonp=jsonpCallback"></script>
  1. 简述原理与过程:首先在客户端注册一个callback, 然后把callback的名字传给服务器。此时,服务器先生成 json 数据。 然后以 javascript 语法的方式,生成一个function , function 名字就是传递上来的参数 jsonp。最后将 json 数据直接以入参的方式,放置到 function 中,这样就生成了一段 js 语法的文档,返回给客户端。
  2. 客户端浏览器,解析script标签,并执行返回的 javascript 文档,此时数据作为参数,传入到了客户端预先定义好的 callback 函数里。(动态执行回调函数)
  • jsonp的特点:
    1.jsonp是通过script的src属性去加载跨域资源的,所以jsonp请求都是get请求
    2.get系有的特点jsonp都有
    3.所有的jspon接口必须含有一个jsonpCallback,否则不是合法的接口
    4.所有的jsonp接口必须按照固定的格式返回 functionName(/* jsonData */)

    原生封装
    //jsonpcallback ---后台执行的函数名,一个字符串
    //callback---前端执行的函数
    (function(){
      var count=1;//使每次的cbName都不重复
      this.jsonp=function (url,data,jsonpcallback,callback){
        var cbName='cb'+count++;//构造全局函数名
        var callbackName='window.jsonp.'+cbName;//函数是全局的
        window.jsonp[cbName]=function(data){
          try{
            callback(data);
          }
          script.parentNode.removeChild('script');
          delete window.jsonp[cbName]//执行完就清除
        }
        var src=tools.padStringToURL(url,data);//向url后面拼接参数
        src=tools.padStringToURL(src,jsonpcallback+'='+callbackName);
        var script=document.creatElement('script');//生成script标签
        script.src=src;
      }
      var tools={
        padStringToURL:function(url,param){
          var param=this.encodeToURIString(param);
          if(!param)return;
          return /\?/.test(url)?url+'&'+param:url+'?'+param
        },
        encodeToURIString:function(data){
          if(!data)return;
          if(typeof data==='string'){
            return data;
          }
          var ary=[];
          for(var n in data){
            if(data.hasOwnProperty(n)){
              ary.push(encodURIComponent(n)+'='+encodURIComponent(data[n]));
            }
          }
          return ary.join('&')//将每组键值对用&连接
        }
      }
    })()
    jsonp("http://suggestion.baidu.com/su", {wd: word}, "cb", function (data) {
              console.log(data)
          })
    
    
    jquery的JSONP
    $.ajax({
      async:false,
      url: 'http://跨域的dns/document!searchJSONResult.action',
      type: "GET",
      dataType: 'jsonp',
      jsonp: 'jsoncallback',
      data: qsData,
      timeout: 5000,
      beforeSend: function(){
          //jsonp 方式此方法不被触发.原因可能是dataType如果指定为jsonp的话,就已经不是ajax事件了
      },
      success: function (json) {//客户端jquery预先定义好的callback函数,成功获取跨域服务器上的json数据后,会动态执行这个callback函数
          if(json.actionErrors.length!=0){
              alert(json.actionErrors);
          }
          genDynamicContent(qsData,type,json);
      },
      complete: function(XMLHttpRequest, textStatus){
          $.unblockUI({ fadeOut: 10 });
      },
      error: function(xhr){
          //jsonp 方式此方法不被触发.原因可能是dataType如果指定为jsonp的话,就已经不是ajax事件了
          //请求出错处理
          alert("请求出错(请检查相关度网络状况.)");
      }
    });
    
    
    es6+promise封装JSONP
    /**
    url:是接口
    params:是参数
    cb:是全局下回调函数名,后台返回 cb(data) ,相当于执行了cb函数,那么客户    端就可以在函数中任意操作data了
    **/
    function jsonp({url,params,cb}){
      return new Promise((resolve,reject)=>{
         let script = document.createElement('script');
         params={...params,cb};
         let arrs = [];
         for(let key in params){
           arrs.push(`${key}=${params[key]}`);
         }
         script.src = `${url}?${arrs.join('&')}`;
         document.body.appendChild(script);
         window[cb] = function (data) {
           resolve(data);
           document.body.removeChild(script);
         }
      })
    };
    jsonp({url:'XXX',data:'XXXX',cb:'XXX'}).then(console.log(data))
    
    

2.Proxy代理 :(接口的临时代理,解决跨域)

https://www.jianshu.com/p/0c301c46685b

3.cors(兼容性IE8以上,跨域请求数据的方式 )

利用浏览器提供的跨域API实现跨域请求

  • 注意
    1.需要服务端设置响应首部Access-Control-Allow-Origin
    2.可以使用get、post、head、delete、put、options这些http方法
var getCors=function(){
  if(window.XDomainRequest){
    return new XDomainRequest();
  }
  if(window.XMLHttpRequest){
    var xhr=new XMLHttpRequest();
    if(xhr.withCredentials!==void 0){
      return xhr;
    }
    throw new Error('不支持cors')
  }
   throw new Error('不支持cors')
}
var cors=getCors();
cors.open('post','urlname.....',false);
cors.onload=function(){
  if(cors.state==200){
    cors.responseText//拿到返回值
  }
}
cors.send(data);
cors的jq用法
$.ajax({
  url:'',
  data:'',
  dataType:'返回的数据类型',
  corssDomain:true,
  success:function(data){
    console.log(data)
  }
})

4.document.domain + iframe (能力有限:同一网站的二级页面之间数据传递)

前提条件:这两个域名必须属于同一个基础域名!而且所用的协议,端口都要一致,否则无法利用document.domain进行跨域
有另一种情况,两个子域名:
  • aaa.xxx.com
  • bbb.xxx.com

可以通过Javascript,将两个页面的domain改成一样的,
需要在a.html里与b.html里都加入:
document.domain = "xxx.com";
这样这两个页面就可以互相操作了。也就是实现了同一基础域名之间的"跨域"。

只要能将document.domain改成一样的就可以跨域
关于iframe

一、在使用iframe的页面,要操作这个iframe里面的DOM元素可以用:

  • contentWindow、contentDocument
    获取iframe里面的window对象,再通过这个对象,获取到里面的DOM元素
    例子:
var ifr = document.getElementById("iframe");
ifr.contentWindow.document.getElementById("XXXXX")

二、在iframe本页面,要操作这个iframe的父页面的DOM元素(即嵌套这个iframe的页面)可以用:

  • window.parent、window.top
var ifr = document.getElementByTagName("iframe");
ifr.parent.document.getElementById("XXXXX")

5.postMessage(HTML5中的XMLHttpRequest Level 2中的API实现页面件的数据传递)

  • 在发送数据窗口执行:otherWindow.postMessage(msg,origin)
  1. otherWindow:表示接受数据的窗口的window对象,包括iframe的contentWindw和通过window.open打开的新窗口。
  2. data表示要发送的数据,包扩字符串和对象(ie9以下不支持,可以利用字符串和json互换)。
  3. origin表示接收的域名。
var win = iframe.contentWindow||其他的window对象;  
win.postMessage(data,'http://wozien.com'); //发送数据data给http://wozien.com
window.onmessage = function (e) {
        console.log(e.data);//拿到数据:回给你
      }
  • 在接受的窗口监听window的message事件,回掉函数参数接受一个事件对象event,包括的属性有:
  1. data:接受的数据
  2. origin:发送端的域
  3. source:发送端的DOMWindow对象
window.onmessage = function(e){  
    if(e.origin !== 'http://localhost') return;  //判断是否可信任
    console.log(e.origin+' '+e.data);  //拿到数据
    e.source.postMessage('回给你',e.origin)
}  

6.web sockets:(只有在支持web socket协议的服务器上才能正常工作,与http的长连接的区别是:这个协议是双向的)

  • websocket约定了一个通信的规范,通过一个握手的机制,客户端(浏览器)和服务器(webserver)之间能建立一个类似tcp的连接,从而方便c-s之间的通信

客户端:

<div>user input:<input type="text"></div>
<script src="./socket.io.js"></script>
<script>
var socket = io('http://www.domain2.com:8080');

// 连接成功处理
socket.on('connect', function() {
    // 监听服务端消息
    socket.on('message', function(msg) {
        console.log('data from server: ---> ' + msg); 
    });

    // 监听服务端关闭
    socket.on('disconnect', function() { 
        console.log('Server socket has closed.'); 
    });
});

document.getElementsByTagName('input')[0].onblur = function() {
    socket.send(this.value);
};
</script>

node端:

var http = require('http');
var socket = require('socket.io');

// 启http服务
var server = http.createServer(function(req, res) {
    res.writeHead(200, {
        'Content-type': 'text/html'
    });
    res.end();
});

server.listen('8080');
console.log('Server is running at port 8080...');

// 监听socket连接
socket.listen(server).on('connection', function(client) {
    // 接收信息
    client.on('message', function(msg) {
        client.send('hello:' + msg);
        console.log('data from client: ---> ' + msg);
    });

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

推荐阅读更多精彩内容