prototype.js学习(1)

prototype.js学习(1) 1、关于javascript的apply和call函数prototype.js中用了大量的apply和call函数,不注意会造成理解偏差。官方解释:应用某一对象的一个方法,用另一个对象替换当前对象。apply与call的区别是第二个参数不同。apply是 数组或者arguments对象。而call是逗号隔开的任何类型。apply,call方法最让人混淆的地方也是apply,call的

大家好,又见面了,我是你们的朋友全栈君。 
1、关于javascript的apply和call函数

prototype.js中用了大量的apply和call函数,不注意会造成理解偏差。
官方解释:应用某一对象的一个方法,用另一个对象替换当前对象。
apply与call的区别是第二个参数不同。apply是  数组或者arguments 对象。而call是逗号隔开的任何类型。

apply,call方法最让人混淆的地方也是apply,call的特色。但最好不要滥用。
能改变调用函数的对象。如下例,函数中用到this关键字,这时候this代表的是apply,call函数的第一个参数。

<script src=”prototype1.3.1.js”></script>
<input type=”text” id=”myText”  value=”input text”>
<script>
       function Obj(){

           this.value=”对象!”;
       }
       var value=”global 变量”;
       function Fun1(){

           alert(this.value);
       }
       window.Fun1();      //global 变量
       Fun1.apply(window); //global 变量,相当于执行window.Fun1();(把Fun1函数应用在Window对象上,即Window.Fun1())
       Fun1.apply($(‘myText’)); //input text,相当于执行 $(‘myText’).Fun1();
       Fun1.apply(new Obj());   //对象!
</script>

2、关于闭包
prototype.js在Class.create,bind等中用到javascript的闭包特色。但整体上prototype.js对于强大的闭包特性用的不多。大家可以参阅我翻译的篇文章了解闭包
3、让我比较反感的两个方法
(1)
var Class = {

  create: function() {

    return function() {
      this.initialize.apply(this, arguments);
    }
  }
}
很讨厌用别的语言的风格来写javascript。用这个方法构造自定义类  并没有觉得有多方便,减少代码行数,只会让人难理解,多定义一个initialize方法。
其实讨厌这条有些牵强,不过修改Object的原型对象就有点过分了。
(2)Object.prototype.extend
  先不过你取个extend的名字会让熟悉java的人引起的歧义。修改Object的prototype就说不过去了。不知道作者是怎么考虑的。当你for in循环对象是,麻烦就来了。可能有人会问你for in干吗。 我一个项目中既用了DWR,也用了prototype.js,dwr返回的javascript对象都多了个exetend属性,还得特殊处理。
  以前我比较过dojo和prototype.js中继承的实现,现在我明白个道理。对于javascript这种没有静态类型检查,语法宽松的语言来讲,如果你选择了某个js类库,那你也必须适应作者写javascript的风格。prototype.js的作者对extend的使用炉火纯青,如果我们不当它只是个属性拷贝的函数的话,多读读prototype.js的代码是好的。
4、关于函数的绑定
  类库提供了Function.prototype.bind  Function.prototype.bindAsEventListener两个方法。首先我们从概念上解释一个这两个方法。
任何一个函数都可以调用这两个方法;参数的是javascript对象或网页上元素对象;返回类型是个函数对象。
本来我就是个函数,返回还是函数,到这两个函数有什么不同呢。看实现代码,关键还是apply/call函数的代码。其实这里只是转化了一下方法调用的对象。

<script src=”prototype1.3.1.js”></script>
<input type=checkbox id=myChk name=”asf” value=1> Test
<script>
    var CheckboxWatcher = Class.create();
    CheckboxWatcher.prototype = {

       initialize: function(chkBox, message) {

            this.chkBox = $(chkBox);
            this.message = message;
            this.chkBox.onclick = this.showMessage.bindAsEventListener(this);
       },
       showMessage: function(evt) {

          alert(this.message + ‘ (‘ + evt.type + ‘)’);
       }
    };
new CheckboxWatcher(‘myChk’,’message!!!!’);
//$(‘myChk’).οnclick=function(){};
</script>
这是 https://compdoc2cn.dev.java.net/ 上举的例子,个人感觉没什么意思,反而让我对bind,bindAsEventListener有些反感。(javascript就是这样,明明大家都知道的语法,但写出来的代码差别确很大)
看下面代码:

<script src=”prototype1.3.1.js”></script>
<input type=checkbox id=myChk name=”chk” value=1> Test
<script>
function Class(){

    this.name=”class”;
}
Class.prototype.getName=function(){

    alert(this.name);
}
var obj=new Class();
//$(‘myChk’).οnclick=obj.getName;
$(‘myChk’).οnclick=obj.getName.bind(obj);
//$(‘myChk’).οnclick=obj.getName.bind($(‘myChk’));
</script>

从上面代码可以看出bind/bindAsEventListener只是包装了一下apply/call方法,改变方法的调用对象。如例子,你可以把obj.getName方法转化成任何对象调用,并且把方法让表单元素触发。(bind和bindAsEventListener之间只是返回函数的参数不同)
这两个方法也可以用在对象之间的方法重用,实现类似继承方法的概念。看以下代码,其实是比较无聊的。

<script src=”prototype1.3.1.js”></script>
<script>
function Class1(name){

    this.name=name;
}
Class1.prototype.getName=function(){

    alert(this.name);
}
function Class2(name){

    this.name=name;
  this.getName=Class1.prototype.getName.bind(this);
}
var obj1=new Class2(“yql”);
obj1.getName();
var obj2=new Object();
obj2.name=”zkj”;
obj2.fun=Class1.prototype.getName.bind(obj2);
obj2.fun();
</script>

我从来没读过prototype.js的扩展项目代码,也不知道bind..的最佳实践,一起挖掘吧。但你绝对不要把bind/bindAsEventListener从绑定的词义上来理解,可能会让你更加迷惑。从apply/call理解本质。应用某一对象的一个方法,用另一个对象替换当前对象。

5、关于事件的注册

<script src=”prototype1.3.1.js”></script>
<input type=checkbox id=myChk name=”chk” value=1> Test
<script>
Event.observe(myChk, ‘click’, showMessage, false);
//$(‘myChk’).οnclick=showMessage;
//$(‘myChk’).οnclick=showMessage.bind();
$(‘myChk’).οnclick=showMessage.bind($(‘myChk’));
function showMessage() {

      alert(this.value);
}
</script>

执行上面代码,你就能明白Event.observe与bind/bindAsEventListener之间的区别:
(1) 显然Event.observe有限制,只能处理简单的函数,并函数中不能有this之类的东西。
(2)Event.observe内部用到addEventListener/attachEvent。能把多个函数加到一个触发事件(window.onload)。bind是覆盖。

6、关于事件监听最佳实践
很显然prototype.js提供的事件注册方法不是很完善。那看看dojo的时间注册吧(中文版),更加复杂,估计很多人像我一样,对于dojo暂时持观望态度。
如果你看过的前篇关于闭包的介绍,可能见过以下代码。
看以下代码前我想表述一个观点,任何网页中元素,浏览器都会为你创建一个对象()。(我觉得)这些对象与你建立javascript对象区别是它们有事件监听,会响应鼠标键盘的事件。如果你用了以下代码,那么把事件监听代码很好的转化到你的javascript代码中。

function associateObjWithEvent(obj, methodName){

    return (function(e){

        e = e||window.event;
        return obj[methodName](e, this);
    });
}
function DhtmlObject(elementId){

    var el = getElementWithId(elementId);
    if(el){

        el.onclick = associateObjWithEvent(this, “doOnClick”);
        el.onmouseover = associateObjWithEvent(this, “doMouseOver”);
        el.onmouseout = associateObjWithEvent(this, “doMouseOut”);
    }
}
DhtmlObject.prototype.doOnClick = function(event, element){

    … // doOnClick method body.
}
DhtmlObject.prototype.doMouseOver = function(event, element){

    … // doMouseOver method body.
}
DhtmlObject.prototype.doMouseOut = function(event, element){

    … // doMouseOut method body.
}

有时间我想用以上思想实现一个网页浮动框拖拉的代码(其实已经有很多了),待续……..

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/163408.html原文链接:https://javaforall.net

(0)
上一篇 2022年7月22日 下午11:16
下一篇 2022年7月22日 下午11:16


相关推荐

  • 如何解决Linux 系统下 ifconfig 命令无网络接口 ens33[通俗易懂]

    如何解决Linux 系统下 ifconfig 命令无网络接口 ens33

    2022年3月13日
    438
  • FFprobe使用指南

    FFprobe使用指南FFprobe 使用指南 FFprobe 是什么 ffprobe 是一个多媒体流分析工具 它从多媒体流中收集信息 并且以人类和机器可读的形式打印出来 它可以用来检测多媒体流的容器类型 以及每一个多媒体流的格式和类型 它可以作为一个独立的应用来使用 也可以结合文本过滤器执行更复杂的处理 FFprobe 使用示例最简单的使用方式 shellffprobe mp4 不显示欢迎信息

    2026年3月18日
    2
  • php二维码分享到朋友圈,php实现的微信分享到朋友圈并记录分享次数功能

    php二维码分享到朋友圈,php实现的微信分享到朋友圈并记录分享次数功能本文实例讲述了php实现的微信分享到朋友圈并记录分享次数功能。分享给大家供大家参考,具体如下:1.引入JS文件2.通过config接口注入权限验证配置3.通过ready接口处理成功验证4.通过error接口处理失败验证JSDK档说明:https://mp.weixin.qq.com/wiki/7/aaa137b55fb2e0456bf8dd9148dd613f.html(1)(2)页面加入获取we…

    2022年6月6日
    31
  • 什么是深浅拷贝?如何实现深浅拷贝

    什么是深浅拷贝?如何实现深浅拷贝什么是深浅拷贝 如何实现深浅拷贝首先 先了解深浅拷贝的含义 浅拷贝就是通过赋值的方式进行拷贝 那为什么说这是浅拷贝呢 就是因为赋值的方式只会把对象的表层赋值给一个新的对象 如果里面有属性值为数组或者对象的属性 那么就只会拷贝到该属性在栈空间的指针地址 新对象的这些属性数据就会跟旧对象公用一份 也就是说两个地址指向同一份数据 一个改变就会都改变 深拷贝则不会出现上述问题 引用数据类型 地址跟数据都会拷贝出来 浅拷贝让我们先来看看浅拷贝 代码走你 letperson uname 张

    2026年3月17日
    2
  • 鼠标滚轮编码器工作原理_速度编码器工作原理

    鼠标滚轮编码器工作原理_速度编码器工作原理鼠标滚轮一旦出现滚动跳动,不连贯,基本都要换,修鼠标会经常遇到,好奇之下想了解一下这个小东西的原理。滚轮一端插在这个转盘里面,我们滚动滚轮时候,转盘被带动旋转,产生脉冲信号,电脑依靠这个信号判断滚轮的旋转方向和速度。我们拆一个机械编码器来看看。就是这个小东西,特别简单有没有,一共就4个零件最左边是铁壳,上面一般会有厂家信息,安装高度,和寿命等比如这个,安装高度10毫米,寿命500万圈。PS:一般普通的鼠标,都是选用安装高度为11mm,但还是要自己量清楚。这里需要注意的是,安装

    2026年4月19日
    2
  • 查询oracle视图创建语句及如何向视图中插入数据[通俗易懂]

    今天由于要测试接口,需要在数据库中插入一些测试数据。但当我在向数据库插入数据的时候,发现接口查询的是视图并不是表,所以将遇到的问题在这里记录一下。1.向视图插入数据的时候分两种情况1.1对于简单视图(视图建立在一张表上),跟表一样直接插入数据就好;1.2对于复杂视图(视图建立时包含多表关联、分组、聚合函数),这个时候不能直接插入数据,应该创建一个INSTEAD类型的触发器来操作,将要插入…

    2022年4月12日
    653

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号