jQuery的观察者模式详解


Posted in Javascript onDecember 22, 2014

在jQuery中,on方法可以为元素绑定事件,trigger方法可以手动触发事件,围绕这2个方法,我们来体验jQuery中的观察者模式(Observer Pattern)。

■ on方法绑定内置事件,自然触发

比如,我们给页面的body元素绑定一个click事件,这样写。

 <head>

<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>

    <title></title>

    <script src="Scripts/jquery-2.1.1.min.js"></script>

    <script type="text/javascript">

        $(function() {

            $('body').on('click', function () {

                console.log('被点击了~~');

            });

        });      

    </script>

</head>

<body>

    <h1>hello</h1>

</body>

以上,我们只有点击body,才能触发click事件。也就是说,当给页面元素绑定内置事件后,事件的触发是在内置事件发生的那刻。

■ on方法绑定内置事件,手动触发

使用trigger方法,也可以手动触发元素绑定的内置事件。

     <script src="Scripts/jquery-2.1.1.min.js"></script>
    <script type="text/javascript">
        $(function() {
            $('body').on('click', function () {
                console.log('被点击了~~');
            });
            $('body').trigger('click');
        });     
    </script>
以上,无需点击body,在页面加载完毕,body自动触发了click事件。

■ on方法绑定自定义事件,手动触发

我们知道,click是jquery内置的事件,那么,是否可以自定义事件,并手动触发呢?

    <script src="Scripts/jquery-2.1.1.min.js"></script>

    <script type="text/javascript">

        $(function() {

            $('body').on('someclick', function () {

                console.log('被点击了~~');

            });

            $('body').trigger('someclick');

        });      

    </script>

以上,我们自定义了一个someclick事件,得到的结果和上面一样。

于是,我们发现:我们可以为元素绑定自定义事件,并且用trigger方法触发该事件。

当然,自定义事件的名称可以按照"命名空间.自定义事件名称"的形式来写,比如app.someclick,这在大型项目中尤其有用,这样可以有效避免自定义事件名称冲突。

如果从"发布订阅"这个角度来看,on方法相当于订阅者、观察者,trigger方法相当于发布者。

■ 从"异步获取json数据"来体验jQuery观察者模式

在根目录下,有一个data.json的文件。

{
    "one" : "Hello",
    "two" : "World"
}
现在,通过异步的方式来获取json数据。

    <script src="Scripts/jquery-2.1.1.min.js"></script>

    <script type="text/javascript">

        $(function () {

            $.getJSON('data.json', function(data) {

                console.log(data);

            });

        });      

    </script>

jQuery的观察者模式详解 

如果用一个全局变量来接收异步获取的json数据。

    <script src="Scripts/jquery-2.1.1.min.js"></script>

    <script type="text/javascript">

        $(function () {

            var data;

            $.getJSON('data.json', function(results) {

                data = results;

            });

            console.log(data);

        });      

    </script>

jQuery的观察者模式详解

这次,我们得到的结果却是undefined,这是为什么?
--因为,当$.getJSON方法还在获取数据的时候,就已经执行console.log(data),而此时data还没有数据。

如何解决这个问题呢?
--如果在$.getJSON方法之外先定义好需要执行的方法,然后在$.getJSON方法的回调函数里真正触发这个方法,不就解决了吗?

    <script src="Scripts/jquery-2.1.1.min.js"></script>

    <script type="text/javascript">

        $(function () {

            $.getJSON('data.json', function(results) {

                $(document).trigger('app.myevent', results); //相当于发布

            });

            $(document).on('app.myevent', function(e, results) { //相当于订阅

                console.log(results);

            });

        });      

    </script>

jQuery的观察者模式详解

以上,on方法就像一个订阅者,它订阅了自定义事件app.myevent;而trigger方法就像一个发布者,它发布事件和参数后,才真正让订阅者方法得以执行。

■ jQuery观察者模式的扩展方法

为此,我们还可以为jQuery观察者模式专门写一个扩展方法。

    <script src="Scripts/jquery-2.1.1.min.js"></script>

    <script type="text/javascript">

        $(function () {

            $.getJSON('data.json', function (results) {

                $.publish('app.myevent', results);

            });

            $.subscribe('app.myevent', function(e, results) {

                console.log(results);

            });

        });

        (function($) {

            var o = $({});//自定义事件对象

            $.each({

                trigger: 'publish',

                on: 'subscribe',

                off: 'unsubscribe'

            }, function(key, val) {

                jQuery[val] = function() {

                    o[key].apply(o, arguments);

                };

            });

        })(jQuery);

    </script>

jQuery的观察者模式详解

以上,定义了全局的publish和subscribe方法,我们在任何时候都可以调用。

    <script src="Scripts/jquery-2.1.1.min.js"></script>

    <script type="text/javascript">

        $(function () {

            $.getJSON('data.json', function (results) {

                $.publish('app.myevent', results);

            });

            $.subscribe('app.myevent', function(e, results) {

                $('body').html(

                    results.one

                );

            });

        });

jQuery的观察者模式详解

总结:jQuery的观察者模式,实际上是让on方法绑定的自定义事件先不执行,直到使用trigger方法来触发事件。使用jQuery的观察者模式的好处是:一次发布,多次订阅。

Javascript 相关文章推荐
Javascript封装DOMContentLoaded事件实例
Jun 12 Javascript
页面内容排序插件jSort使用方法
Oct 10 Javascript
js+canvas绘制矩形的方法
Jan 28 Javascript
轻松掌握JavaScript中的Math object数学对象
May 26 Javascript
Bootstrap Table使用方法详解
Aug 01 Javascript
微信公众号  提示:Unauthorized API function 问题解决方法
Dec 05 Javascript
SVG描边动画
Feb 23 Javascript
详解axios在node.js中的post使用
Apr 27 Javascript
简述JS浏览器的三种弹窗
Jul 15 Javascript
微信用户访问小程序的登录过程详解
Sep 20 Javascript
《javascript设计模式》学习笔记四:Javascript面向对象程序设计链式调用实例分析
Apr 07 Javascript
vue3自定义dialog、modal组件的方法
Jan 04 Vue.js
使用jQuery和Bootstrap实现多层、自适应模态窗口
Dec 22 #Javascript
sails框架的学习指南
Dec 22 #Javascript
了不起的node.js读书笔记之mongodb数据库交互
Dec 22 #Javascript
javascript动态创建及删除元素的方法
Dec 22 #Javascript
了不起的node.js读书笔记之例程分析
Dec 22 #Javascript
了不起的node.js读书笔记之node的学习总结
Dec 22 #Javascript
了不起的node.js读书笔记之node.js中的特性
Dec 22 #Javascript
You might like
php join函数应用
2011/05/04 PHP
PHP中header和session_start前不能有输出原因分析
2013/01/11 PHP
PHP curl 获取响应的状态码的方法
2014/01/13 PHP
php实现登录tplink WR882N获取IP和重启的方法
2016/07/20 PHP
php实现的三个常用加密解密功能函数示例
2017/11/06 PHP
PHP中in_array的隐式转换的解决方法
2018/03/06 PHP
使用laravel的Eloquent模型如何获取数据库的指定列
2019/10/17 PHP
JS实现浏览器菜单命令
2006/09/05 Javascript
Javascript写了一个清除“logo1_.exe”的杀毒工具(可扫描目录)
2007/02/09 Javascript
jQuery $.extend()用法总结
2014/06/15 Javascript
js简单抽奖代码
2015/01/16 Javascript
JavaScript学习笔记之数组的增、删、改、查
2016/03/23 Javascript
javascript的BOM
2016/05/03 Javascript
Javascript打印局部页面实例
2016/06/21 Javascript
JavaScript中日期函数的相关操作知识
2016/08/03 Javascript
给Easyui-Datebox设置隐藏或者不可用的解决方法
2017/05/26 Javascript
vue axios同步请求解决方案
2017/09/29 Javascript
vue利用better-scroll实现轮播图与页面滚动详解
2017/10/20 Javascript
JS异步错误捕获的一些事小结
2019/04/26 Javascript
layui实现数据表格自定义数据项
2019/10/26 Javascript
原生JS封装拖动验证滑块的实现代码示例
2020/06/01 Javascript
[50:27]Secret vs VG 2018国际邀请赛小组赛BO2 第二场 8.17
2018/08/20 DOTA
Python学习笔记之os模块使用总结
2014/11/03 Python
django rest framework之请求与响应(详解)
2017/11/06 Python
python实现下载pop3邮件保存到本地
2018/06/19 Python
使用Python将字符串转换为格式化的日期时间字符串
2019/09/01 Python
numpy数组做图片拼接的实现(concatenate、vstack、hstack)
2019/11/08 Python
Python greenlet和gevent使用代码示例解析
2020/04/01 Python
服务器端jupyter notebook映射到本地浏览器的操作
2020/04/14 Python
使用python创建生成动态链接库dll的方法
2020/05/09 Python
Python引入多个模块及包的概念过程解析
2020/09/21 Python
批评与自我批评发言稿
2014/10/15 职场文书
党的群众路线教育实践活动批评与自我批评发言稿
2014/10/16 职场文书
2014矛盾纠纷排查调处工作总结
2014/12/09 职场文书
爱晚亭导游词
2015/02/09 职场文书
父母教会我观后感
2015/06/17 职场文书