JavaScript实现的双向跨域插件分享


Posted in Javascript onJanuary 31, 2015

由于浏览器(同源策略)限制,JavaScript 跨域的问题,一直是一个颇为棘手的问题。HTML5 提供了跨文档消息传输的功能,在网页文档之间互相接收与发送信息。使用这个功能,不仅同源(域 + 端口号)的 Web 网页之间可以互相通信,还可以在两个不同域名之间实现跨域通信。

跨文档消息传输Cross Document Messaging提供了postMessage方法在不同网页文档之间互相传递数据,支持实时消息传递。现在很多浏览器都将支持这个功能,比如Google Chrome 2.0+、Internet Explorer 8.0+、Firefox 3.0+、Opera 9.6+、Safari 4.0+等

那么,IE6、IE7等不支持 HTML5的浏览器怎么办?

可以使用window.name方法,因为window.name的修改不涉及跨域问题,虽然使用起来不是特别理想,但效果还可以接受。
但是,我们总不能每次涉及到跨域都去写一遍window.postMessage、window.addEventListener、window.name等等这些内容吧。

为此,我把这整个跨域过程抽象出来,封装成一个JavaScript 插件,解决双向跨域问题,实现不同网页文档之间的实时通信,可以在两个不同域名之间实现跨域通信。

demo下载地址:http://xiazai.3water.com/201501/other/jcrossdomain_v2.rar,版本v2

javascript跨域插件jcrossdomain.js

(function (win){

  /** 

   * 没有开花的树 

   * 2013/12/07 17:12 

   */ 

  var _jcd = {

    isInited : false,

    elmt : false,

    hash : '',

    delims : ',',

    rand : function(){

      return (new Date).getTime()

    },

    msg : function(){

      alert('Warning: You must call init function at first');

    },

    init : function(callback, elmt){

      if(_jcd.isInited == true)

        return;

      _jcd.isInited = true;

      _jcd.elmt = elmt;

      if(win.postMessage){

        //浏览器支持 HTML5 postMessage 方法

        if(win.addEventListener){

          //支持火狐、谷歌等浏览器

          win.addEventListener("message", function(ev){

            callback.call(win, ev.data);

          },false);

        }else if(win.attachEvent){

          //支持IE浏览器

          win.attachEvent("onmessage", function(ev){

            callback.call(win, ev.data);

          });

        }

        _jcd.msg = function(data){

          _jcd.elmt.postMessage(data, '*');

        }

      }else{

        //浏览器不支持 HTML5 postMessage 方法,如IE6、7

        setInterval(function(){

          if (win.name !== _jcd.hash) {

            _jcd.hash = win.name;

            callback.call(win, _jcd.hash.split(_jcd.delims)[1]);

          }

        }, 50);

        _jcd.msg = function(data){

          _jcd.elmt.name = _jcd.rand() + _jcd.delims + data;

        }

      }

    }

  };
  var jcd = {
    initParent : function(callback, iframeId){

      _jcd.init(callback, document.getElementById(iframeId).contentWindow);

    },
    initChild : function(callback){

      _jcd.init(callback, win.parent);

    },
    sendMessage : function(data){

      _jcd.msg(data);

    }
  };

  win.jCrossDomain = jcd;

})(window);

父网页中调用方法:

//自定义回调函数

var cb = function(msg){

  alert("get msg:" + msg);

};
//初始化,载入回调函数和 iframe 的id

jCrossDomain.initParent(cb, 'iframeA');
//发消息

jCrossDomain.sendMessage('hello, child');

子网页中调用方法:

//自定义回调函数

var cb = function(msg){

  alert("get msg:" + msg);

};
//初始化,载入回调函数

jCrossDomain.initChild(cb);
//发消息

jCrossDomain.sendMessage('hello, parent');

模拟测试小提示:
为了实现不同域之间的通信,可以在操作系统的 hosts 文件添加两个域名,进行模拟。

hosts 文件中添加两个不同的域名
127.0.0.1  parent.com
127.0.0.1  child.com

程序猿的进化过程:

JavaScript实现的双向跨域插件分享

Javascript 相关文章推荐
jquery ajax请求方式与提示用户正在处理请稍等
Sep 01 Javascript
JavaScript中使用Object.create()创建对象介绍
Dec 30 Javascript
jquery插件qrcode在线生成二维码
Apr 26 Javascript
元素绑定click点击事件方法
Jun 08 Javascript
纯HTML5制作围住神经猫游戏-附源码下载
Aug 23 Javascript
Bootstrap嵌入jqGrid,使你的table牛逼起来
May 05 Javascript
jQuery实现简单的抽奖游戏
May 05 jQuery
判断jQuery是否加载完成,没完成继续判断的解决方法
Dec 06 jQuery
对vux点击事件的优化详解
Aug 28 Javascript
node微信开发之获取access_token+自定义菜单
Mar 17 Javascript
vue项目中引入vue-datepicker插件的详解
May 14 Javascript
深入了解Hybrid App技术的相关知识
Jul 17 Javascript
JavaScript判断变量是否为空的自定义函数分享
Jan 31 #Javascript
分享两个手机访问pc网站自动跳转手机端网站代码
Dec 24 #Javascript
js判断登录与否并确定跳转页面的方法
Jan 30 #Javascript
jQuery知识点整理
Jan 30 #Javascript
浅谈jquery回调函数callback的使用
Jan 30 #Javascript
jQuery修改class属性和CSS样式整理
Jan 30 #Javascript
javascript中AJAX用法实例分析
Jan 30 #Javascript
You might like
解析php利用正则表达式解决采集内容排版的问题
2013/06/20 PHP
php 获取本地IP代码
2013/06/23 PHP
PHP对文件进行加锁、解锁实例
2015/01/23 PHP
php ucwords() 函数将字符串中每个单词的首字符转换为大写(实现代码)
2016/05/12 PHP
Thinkphp框架开发移动端接口(2)
2016/08/18 PHP
完美解决Thinkphp3.2中插入相同数据的问题
2017/08/01 PHP
Laravel访问出错提示:`Warning: require(/vendor/autoload.php): failed to open stream: No such file or di解决方法
2019/04/02 PHP
prototype1.4中文手册
2006/09/22 Javascript
javascript 自动转到命名锚记
2009/01/10 Javascript
写js时遇到的一些小问题
2010/12/06 Javascript
jquery二级导航内容均分的原理及实现
2013/08/13 Javascript
Javascript Ajax异步读取RSS文档具体实现
2013/12/12 Javascript
探寻Javascript执行效率问题
2014/11/12 Javascript
重写document.write实现无阻塞加载js广告(补充)
2014/12/12 Javascript
jQuery圆形统计图开发实例
2015/01/04 Javascript
运行Node.js的IIS扩展iisnode安装配置笔记
2015/03/02 Javascript
JavaScript模版引擎的基本实现方法浅析
2016/02/15 Javascript
Javascript实现前端简单的路由实例
2016/09/11 Javascript
解决URL地址中的中文乱码问题的办法
2017/02/10 Javascript
vue项目中使用Hbuilder打包app 设置沉浸式状态栏的方法
2018/10/22 Javascript
微信小程序出现wx.getLocation再次授权问题的解决方法分析
2019/01/16 Javascript
用element的upload组件实现多图片上传和压缩的示例代码
2019/02/12 Javascript
浅谈js闭包理解
2019/03/28 Javascript
Vue路由对象属性 .meta $route.matched详解
2019/11/04 Javascript
JS实现选项卡插件的两种写法(jQuery和class)
2020/12/30 jQuery
讲解Python中的标识运算符
2015/05/14 Python
Python实现多线程抓取妹子图
2015/08/08 Python
python 读取更新中的log 或其它文本方式
2019/12/24 Python
Pytorch Tensor基本数学运算详解
2019/12/30 Python
介绍一下Linux文件的记录形式
2013/09/29 面试题
有多年工作经验的自我评价
2014/03/02 职场文书
音乐课外活动总结
2015/05/09 职场文书
小学班主任研修日志
2015/11/13 职场文书
python 自动刷新网页的两种方法
2021/04/20 Python
JavaScript+HTML实现学生信息管理系统
2021/04/20 Javascript
Android移动应用开发指南之六种布局详解
2022/09/23 Java/Android