初识ext了解js的潮流
开篇,原本想大量引用网上对Ext-base.js文件的分析。但是经过大量的阅读国外的书籍发现,原来那人根本不懂什么是js面向对象程序设计。
参考书:
Apress出版的Pro.JavaScript.Design.Patterns
Professional.JavaScript.for.Web.Developers.2nd.Edition.2009
本文想通过一步一步的对javascript语法的进化来最后达到最ext-base.js的解析过程。
1. Javascript中的类
利用对javascript的常识,我认为js的设计应该如下:
定义了一个新的类Anim,委派了两个方法给prototype属性。所谓的prototype,我的理解是指针。实际上它和c++,c中指针的运行方法有微小的差别,但大致一样。详情请参考上面的两本书。
如果你喜欢把所有的属性和方法都装在类中。如下:
通过以上的操作,产生一个新的概念:Encapsulation。我的理解:装箱。就是把所有的属性和方法放到类中。
值得谈到的是Javascript中的函数。在javascript中函数就是类。
函数的大部分概念和C++没有什么区别。值得注意的是匿名函数.
如果你使用装箱的写法,可以试试下面一种:
在javascript中,所有的东西都是对象,所有的对象都是可变的。这就意味着你可以在类定义好,并实例化以后改变它。
这种对象的可变性叫:自我检查introspection.你能实时的检查任何的属性和方法。这正因为对象可以实时改变,javascript才很少进行类型检查。
在javascript中,还有个根深蒂固的概念。闭包Closures。闭包就是在类中定义的函数。由于对象具有可变性,因此我们可以在类体意外去调用和改变类中定义的函数。
在javascript中,能够想到的创建对象的样式有三种。完全显示的对象(最简单但只提供公有成员)。使用括号标识方法和属性来区别私有成员(只是伪代码),详情请参考上面的书籍。最后是通过闭包的方法建立真正的私有成员。通过闭包实现私有成员。
有了闭包的特性,是可以实现真实的私有成员和公有成员的。在javascript中没有私有,公有的特性的。但是在实际的程序当中,我们往往要用到这些特性。这也是面向对象的重要部分之一。
但是这么写的类,虽然是完成了类成员的私有,公有特性。但是这样的方法很难被继承,同时也造成大量的内存冗余。因为javascript中没有资源回收的机制。在类使用完毕之后,存储空间不会随之消失。我们还需要静态方法和真正的动态空间存储的方法。
静态方法的实现,实际上是利用匿名函数,在内存的堆中申请一片空间。类中只记录这片存储区域的地址。类调用完成,匿名函数随之消失。同时类中申请的静态变量保留了下来。
有了上面面向对象的公有私有成员和方法,对象可变性的概念。来看看js中继承的实现
Javascript的语法提供使用基于对象的继承。我们可以用它来模拟基于类的继承。
尽管代码相当简单,利用prototype这个指针进行地址的链接。但是它却是个很复杂的话题。javascript没有继承关键字。与之替代的,每个object对象都有prototype属性。这个属性指向另一个对象或者null。当对象成员被读取,javascript通过prototype读取该对象。如果它在现在的对象中不存在,javascript会读取每个对象的prototype。直到读到该对象或者null,这和其他的面向对象语言完全不一样。这种方法使得多个子类被实例化的同时,父类也同样被实例化很多次。浪费了内存。在子类实例化的情况下,实例化改变父类,子类会受到很大影响。
基于以上原因,必须要改变继承的方式。
继承方式已经实现。但是离ext-base.js的目标还差得远。在此,先提出一个概念:Prototypal Inheritance,我的理解为:指针型继承。指针型继承非常不同。我们发现最好的去思考它的方法是忘掉一切关于类和实例化的方法。只考虑对象本身,典型创建对象有两种方法
a.定义对象结构,使用类声明
b.实例化类,并创建对象(网上把它叫做对象直接量)
通过实例化类来创建的对象是拷贝所有实例化的属性,连接所有实例化方法的地址在指针型继承中,这样做的优点是你能很简单的建一个类。这个类可以被其他类重用。
通过Prototypal Inheritance的指针继承,我们体会到一种设计模式:骨架模式。骨架模式就是基类设置好所有的属性和方法,并赋默认值。子类需要自己的方法可以任意的添加。这里引出一个概念:异步读写父类成员。为了有效的使用指针型继承,我们必须忘掉所有传统的继承方式。这就是所要忘记的方式之一。在传统的继承里,每个Author类都拷贝了books数组。你可以通过author[1].books.push('New Book Title')添加到数组。这种方式不能用于指针型继承。因为prototype链起了作用。 clone函数并不是object对象的prototype完全独立的拷贝。当你读取author[1].name,你只是得到从prototype那里保存的连接。并没有得到实例的值。当你写一个author[1].name,实际是为author[1]对象添加了一个新的属性。通俗的讲,父类给出了骨架,子类去加肉。
但是有时候,指针对象会有子对象包含其中。如果你想重写子对象某一个单一的方法,你必须重建整个对象。这可以通过将子对象复制给空对象实现。但是这要求被克隆的对象必须明确子对象的结构和默认值。为了保持各对象之间的低耦合度,任何复杂的子对象都应该用固定的方法创建。工厂模式因此而产生。
到此,ext-base.js的所有概念都有了。下面开始具体分析extend类
参考书:
Apress出版的Pro.JavaScript.Design.Patterns
Professional.JavaScript.for.Web.Developers.2nd.Edition.2009
本文想通过一步一步的对javascript语法的进化来最后达到最ext-base.js的解析过程。
1. Javascript中的类
利用对javascript的常识,我认为js的设计应该如下:
/* Anim class. */ var Anim = function() { ... }; Anim.prototype.start = function() { ... }; Anim.prototype.stop = function() { ... }; /* Usage. */ var myAnim = new Anim(); myAnim.start(); ... myAnim.stop();
定义了一个新的类Anim,委派了两个方法给prototype属性。所谓的prototype,我的理解是指针。实际上它和c++,c中指针的运行方法有微小的差别,但大致一样。详情请参考上面的两本书。
如果你喜欢把所有的属性和方法都装在类中。如下:
/* Anim class, with a slightly different syntax for declaring methods. */ var Anim = function() { ... }; Anim.prototype = { start: function() { ... }, stop: function() { ... } };
通过以上的操作,产生一个新的概念:Encapsulation。我的理解:装箱。就是把所有的属性和方法放到类中。
值得谈到的是Javascript中的函数。在javascript中函数就是类。
函数的大部分概念和C++没有什么区别。值得注意的是匿名函数.
<html> <head> <title>String Example</title> <script type="text/javascript"> //在javascript中,函数都最好的对象 //函数的大部分概念和C++没有什么区别。值得注意的是匿名函数 //示例如下: /* An anonymous function, executed immediately. */ (function(){ var foo = 10; var bar = 2; alert(foo * bar); })(); //这个函数的定义和执行并没有分配给任何变量。最后的()执行了这个函数。 //他们是空的,但是并不是这种情况 /* An anonymous function with arguments. */ (function(foo, bar){ alert(foo * bar); })(10, 2); //下例同第一个例子等价。不过将值给了一个内在的变量 /* An anonymous function that returns a value. */ var baz = (function(foo, bar){ return foo * bar; })(10, 2); alert(baz); //匿名函数最有趣的应用时建立闭包。 //闭包是一个受保护的变量空间,这就是说闭包函数可以在他们所定义的任意地方执行。不受函数体影响 /* An anonymous function used as a closure. */ var baz; (function(){ var foo = 10; var bar = 2; baz = function(){ return foo * bar; }; })(); baz(); // baz can access foo and bar, even though it is executed outside of the class </script> </head> <body> <!-- Nothing in the body --> //在javascript中,函数都最好的对象 //函数的大部分概念和C++没有什么区别。值得注意的是匿名函数 //示例如下: /* An anonymous function, executed immediately. */ (function(){ var foo = 10; var bar = 2; alert(foo * bar); })(); //这个函数的定义和执行并没有分配给任何变量。最后的()执行了这个函数。 //他们是空的,但是并不是这种情况 /* An anonymous function with arguments. */ (function(foo, bar){ alert(foo * bar); })(10, 2); //下例同第一个例子等价。不过将值给了一个内在的变量 /* An anonymous function that returns a value. */ var baz = (function(foo, bar){ return foo * bar; })(10, 2); alert(baz); //匿名函数最有趣的应用时建立闭包。 //闭包是一个受保护的变量空间,这就是说闭包函数可以在他们所定义的任意地方执行。不受函数体影响 /* An anonymous function used as a closure. */ var baz; (function(){ var foo = 10; var bar = 2; baz = function(){ return foo * bar; }; })(); baz(); // baz can access foo and bar, even though it is executed outside of the class </body> </html>
如果你使用装箱的写法,可以试试下面一种:
/* Add a method to the Function object that can be used to declare methods. */ Function.prototype.method = function(name, fn) { this.prototype[name] = fn; }; /* Anim class, with methods created using a convenience method. */ var Anim = function() { ... }; Anim.method('start', function() { ... }); Anim.method('stop', function() { ... }); 并修改method,让指针只记住类中各种方法的地址。即只给出链接。 /* This version allows the calls to be chained. */ Function.prototype.method = function(name, fn) { this.prototype[name] = fn; return this; }; /* Anim class, with methods created using a convenience method and chaining. */ var Anim = function() { ... }; Anim.method('start', function() { ... }). method('stop', function() { ... });
在javascript中,所有的东西都是对象,所有的对象都是可变的。这就意味着你可以在类定义好,并实例化以后改变它。
<html> <head> <title>String Example</title> <script type="text/javascript"> //在javascript中,所有的东西都是对象 //所有的对象都是可变的。 //这就意味着你可以在类定义好,并实例化以后改变它。 /* Class Person. */ function Person(name, age){ this.name = name; this.age = age; } Person.prototype = { getName: function(){ return this.name; }, getAge: function(){ return this.age; } } /* Instantiate the class. */ var alice = new Person('Alice', 93); var bill = new Person('Bill', 30); document.write("/* Instantiate the class. */<br/>"); document.write("name: "+alice.name+";her age: "+alice.age+"<br/>"); document.write("name: "+bill.name+";his age: "+bill.age+"<br/>"); /* Modify the class. */ Person.prototype.getGreeting = function(){ return 'Hi ' + this.getName() + '!'; }; document.write("/* Modify the class. */<br/>"); document.write("alice.getGreeting:"+alice.getGreeting()+"<br/>"); /* Modify a specific instance. */ alice.displayGreeting = function(){ alert(this.getGreeting()); } document.write("/* Modify a specific instance. */<br/>alert(getGreeting)"); alice.displayGreeting(); </script> </head> <body> <!-- Nothing in the body --> </body> </html>
这种对象的可变性叫:自我检查introspection.你能实时的检查任何的属性和方法。这正因为对象可以实时改变,javascript才很少进行类型检查。
在javascript中,还有个根深蒂固的概念。闭包Closures。闭包就是在类中定义的函数。由于对象具有可变性,因此我们可以在类体意外去调用和改变类中定义的函数。
<html> <head> <title>String Example</title> <script type="text/javascript"> function foo(){ var a = 10; function bar(){ a *= 2; } bar(); return a; } document.writeln(foo()); </script> </head> <body> <!-- Nothing in the body --> </body> </html>
在javascript中,能够想到的创建对象的样式有三种。完全显示的对象(最简单但只提供公有成员)。使用括号标识方法和属性来区别私有成员(只是伪代码),详情请参考上面的书籍。最后是通过闭包的方法建立真正的私有成员。通过闭包实现私有成员。
有了闭包的特性,是可以实现真实的私有成员和公有成员的。在javascript中没有私有,公有的特性的。但是在实际的程序当中,我们往往要用到这些特性。这也是面向对象的重要部分之一。
<html> <head> <title>String Example</title> <script type="text/javascript"> var Book = function(newIsbn, newTitle, newAuthor){ // implements Publication // Private attributes. var isbn, title, author; // Privileged methods. this.getIsbn = function(){ return isbn; }; this.setIsbn = function(newIsbn){ isbn = newIsbn; }; this.getTitle = function(){ return title; }; this.setTitle = function(newTitle){ title = newTitle || 'No title specified'; }; this.getAuthor = function(){ return author; }; this.setAuthor = function(newAuthor){ author = newAuthor || 'No author specified'; }; // Constructor code. this.setIsbn(newIsbn); this.setTitle(newTitle); this.setAuthor(newAuthor); }; // Public, non-privileged methods. Book.prototype = { display: function(){ return "ISBN:"+this.getIsbn() + "<br/>" +"Title:" + this.getTitle() + "<br/>" +"Author:" + this.getAuthor(); } }; var book = new Book('978-0261103283','pro javascript design pattern','Apress'); document.write(book.display()); //使用了更多的内存 //很难被继承 </script> </head> <body> <!-- Nothing in the body --> </body> </html>
但是这么写的类,虽然是完成了类成员的私有,公有特性。但是这样的方法很难被继承,同时也造成大量的内存冗余。因为javascript中没有资源回收的机制。在类使用完毕之后,存储空间不会随之消失。我们还需要静态方法和真正的动态空间存储的方法。
<html> <head> <title>String Example</title> <script type="text/javascript"> var Book = (function(){ // Private static attributes. var numOfBooks = 0; // Return the constructor. return function(newIsbn, newTitle, newAuthor){ // implements Publication // Private attributes. var isbn, title, author; // Private static method. function checkIsbn(isbn){ if (isbn) { return true; } else { return false; } } // Privileged methods. this.getIsbn = function(){ return isbn; }; this.setIsbn = function(newIsbn){ if (!checkIsbn(newIsbn)) throw new Error('Book: Invalid ISBN.'); isbn = newIsbn; }; this.getTitle = function(){ return title; }; this.setTitle = function(newTitle){ title = newTitle || 'No title specified'; }; this.getAuthor = function(){ return author; }; this.setAuthor = function(newAuthor){ author = newAuthor || 'No author specified'; }; // Constructor code. numOfBooks++; // Keep track of how many Books have been instantiated // with the private static attribute. if (numOfBooks > 50) throw new Error('Book: Only 50 instances of Book can be ' + 'created.'); this.setIsbn(newIsbn); this.setTitle(newTitle); this.setAuthor(newAuthor); } })(); // Public static method. Book.convertToTitle = function(inputString){ return inputString + "[us]"; }; // Public, non-privileged methods. Book.prototype = { display: function(){ return "ISBN:" + this.getIsbn() + "<br/>" + "Title:" + this.getTitle() + "<br/>" + "Author:" + this.getAuthor(); } }; var book = new Book('978-0261103283', Book.convertToTitle('pro javascript design pattern'), 'Apress'); document.write(book.display()); </script> </head> <body> <!-- Nothing in the body --> </body> </html>
静态方法的实现,实际上是利用匿名函数,在内存的堆中申请一片空间。类中只记录这片存储区域的地址。类调用完成,匿名函数随之消失。同时类中申请的静态变量保留了下来。
有了上面面向对象的公有私有成员和方法,对象可变性的概念。来看看js中继承的实现
Javascript的语法提供使用基于对象的继承。我们可以用它来模拟基于类的继承。
<html> <head> <title>String Example</title> <script type="text/javascript"> /* Class Person. */ function Person(name){ this.name = name; } Person.prototype.getName = function(){ return this.name; } /* Class Author. */ //建立一个类,继承于其他类 function Author(name, books){ //首先,建立构造器函数,call父类的构造函数,传递参数 Person.call(this, name); // Call the superclass's constructor in the scope of this. this.books = books; // Add an attribute to Author. } //当你调用new操作符,很多事已经为你准备好。首先一个空的object对象已经建立。 //构造器函数调用空对象的scope chain Author.prototype = new Person(); // Set up the prototype chain. Author.prototype.constructor = Author; // Set the constructor attribute to Author. Author.prototype.getBooks = function(){ // Add a method to Author. return this.books; }; //实例测试 var author = []; author[0] = new Author('Dustin Diaz', ['JavaScript Design Patterns']); author[1] = new Author('Ross Harmes', ['JavaScript Design Patterns']); document.write(author[0].getName()+' ' +author[0].getBooks()+"<br/>"); document.write(author[1].getName()+' ' +author[1].getBooks()); </script> </head> <body> <!-- Nothing in the body --> </body> </html>
尽管代码相当简单,利用prototype这个指针进行地址的链接。但是它却是个很复杂的话题。javascript没有继承关键字。与之替代的,每个object对象都有prototype属性。这个属性指向另一个对象或者null。当对象成员被读取,javascript通过prototype读取该对象。如果它在现在的对象中不存在,javascript会读取每个对象的prototype。直到读到该对象或者null,这和其他的面向对象语言完全不一样。这种方法使得多个子类被实例化的同时,父类也同样被实例化很多次。浪费了内存。在子类实例化的情况下,实例化改变父类,子类会受到很大影响。
基于以上原因,必须要改变继承的方式。
<html> <head> <title>String Example</title> <script type="text/javascript"> /* Extend function, improved. */ function extend(subClass, superClass){ var F = function(){ }; F.prototype = superClass.prototype; subClass.prototype = new F(); subClass.prototype.constructor = subClass; subClass.superclass = superClass.prototype; if (superClass.prototype.constructor == Object.prototype.constructor) { superClass.prototype.constructor = superClass; } } /* Class Person. */ function Person(name){ this.name = name; } Person.prototype.getName = function(){ return this.name; } /* Class Author. */ function Author(name, books){ Author.superclass.constructor.call(this, name); this.books = books; } extend(Author, Person); Author.prototype.getBooks = function(){ return this.books; }; //改变Author类 Author.prototype.getName = function(){ var name = Author.superclass.getName.call(this); return name + ', Author of ' + this.getBooks().join(', '); }; //实例测试 var author = []; author[0] = new Author('Dustin Diaz', ['JavaScript Design Patterns']); author[1] = new Author('Ross Harmes', ['JavaScript Design Patterns']); document.write(author[0].getName() + "<br/>"); document.write(author[1].getName()); </script> </head> <body> <!-- Nothing in the body --> </body> </html>
继承方式已经实现。但是离ext-base.js的目标还差得远。在此,先提出一个概念:Prototypal Inheritance,我的理解为:指针型继承。指针型继承非常不同。我们发现最好的去思考它的方法是忘掉一切关于类和实例化的方法。只考虑对象本身,典型创建对象有两种方法
a.定义对象结构,使用类声明
b.实例化类,并创建对象(网上把它叫做对象直接量)
通过实例化类来创建的对象是拷贝所有实例化的属性,连接所有实例化方法的地址在指针型继承中,这样做的优点是你能很简单的建一个类。这个类可以被其他类重用。
<html> <head> <title>String Example</title> <script type="text/javascript"> //函数与函数间指针的传递 /* Clone function. */ function clone(object){ function F(){ } F.prototype = object; return new F; } /* * 指针型继承Prototypal Inheritance * 指针型继承非常不同。我们发现最好的去思考它的方法是忘掉一切关于类和实例化的方法。只考虑对象本身 * 典型创建对象有两种方法 * a.定义对象结构,使用类声明 * b.实例化类,并创建对象 * 通过实例化类来创建的对象是拷贝所有实例化的属性,连接所有实例化方法的地址 * 在指针型继承中,你能很简单的建一个类。这个类可以被其他类重用 */ /* Person Prototype Object. */ var Person = { name: 'default name', getName: function(){ return this.name; } }; //简单对象的继承和测试 var reader = clone(Person); document.write("<h5 style='color:red'>simple object inherited and testing</h5>"); document.write(reader.getName() + "<br/>"); // This will output 'default name'. reader.name = 'John Smith'; document.write(reader.getName() + "<br/>"); // This will now output 'John Smith'. /* Author Prototype Object. */ var Author = clone(Person); Author.books = []; // Default value. Author.getBooks = function(){ return this.books; } var author = []; author[0] = clone(Author); author[0].name = 'Dustin Diaz'; author[0].books = ['JavaScript Design Patterns']; author[1] = clone(Author); author[1].name = 'Ross Harmes'; author[1].books = ['JavaScript Design Patterns']; document.write("<h5 style='color:red'>Author object inherited and testing</h5>"); document.write(author[0].getName() + ' ' + author[0].getBooks() + "<br/>"); document.write(author[1].getName() + ' ' + author[1].getBooks() + "<br/>"); /* * 异步读写父类成员 * 为了有效的使用指针型继承,我们必须忘掉所有传统的继承方式。这就是所要忘记的方式之一。 * 在传统的继承里,每个Author类都拷贝了books数组。你可以通过author[1].books.push('New Book Title')添加到数组 * 这种方式不能用于指针型继承。因为prototype链起了作用 * clone函数并不是object对象的prototype完全独立的拷贝。 * 当你读取author[1].name,你只是得到从prototype那里保存的连接。并没有得到实例的值 * 当你写一个author[1].name,实际是为author[1]对象添加了一个新的属性 */ var authorClone = clone(Author); document.write("<h5 style='color:red'>Symmetrical Reading and Writing of Inherited Members</h5>"); document.write("Name="+authorClone.getName() + "<br/>"); // Linked to the primative Person.name, which is the string 'default name'. document.write("Books="+authorClone.getBooks() + "<br/>"); authorClone.name = 'new name'; // A new primative is created and added to the // authorClone object itself. document.write("Name="+authorClone.getName() + "<br/>"); // Now linked to the primative authorClone.name, which // is the string 'new name'. authorClone.books.push('new book'); // authorClone.books is linked to the array document.write("Books="+authorClone.getBooks() + "<br/>"); // Author.books. We just modified the // prototype object's default value, and all // other objects that link to it will now // have a new default value there. authorClone.books = []; // A new array is created and added to the authorClone // object itself. authorClone.books.push('a new book to array'); // We are now modifying that new array. document.write("<h5 style='color:red'>Asymmetrical Reading and Writing of Inherited Members</h5>"); document.write(authorClone.getName() + ' ' + authorClone.getBooks() + "<br/>"); </script> </head> <body> <!-- Nothing in the body --> </body> </html>
通过Prototypal Inheritance的指针继承,我们体会到一种设计模式:骨架模式。骨架模式就是基类设置好所有的属性和方法,并赋默认值。子类需要自己的方法可以任意的添加。这里引出一个概念:异步读写父类成员。为了有效的使用指针型继承,我们必须忘掉所有传统的继承方式。这就是所要忘记的方式之一。在传统的继承里,每个Author类都拷贝了books数组。你可以通过author[1].books.push('New Book Title')添加到数组。这种方式不能用于指针型继承。因为prototype链起了作用。 clone函数并不是object对象的prototype完全独立的拷贝。当你读取author[1].name,你只是得到从prototype那里保存的连接。并没有得到实例的值。当你写一个author[1].name,实际是为author[1]对象添加了一个新的属性。通俗的讲,父类给出了骨架,子类去加肉。
但是有时候,指针对象会有子对象包含其中。如果你想重写子对象某一个单一的方法,你必须重建整个对象。这可以通过将子对象复制给空对象实现。但是这要求被克隆的对象必须明确子对象的结构和默认值。为了保持各对象之间的低耦合度,任何复杂的子对象都应该用固定的方法创建。工厂模式因此而产生。
<html> <head> <title>String Example</title> <script type="text/javascript"> /* Clone function. */ function clone(object){ function F(){ } F.prototype = object; return new F; } /* * 有时候,指针对象会有子对象包含其中。 * 如果你想重写子对象某一个单一的方法,你必须重建整个对象 * 这可以通过将子对象复制给空对象实现 * 但是这要求被克隆的对象必须明确子对象的结构和默认值 * 为了保持各对象之间的低耦合度,任何复杂的子对象都应该用固定的方法创建 * 工厂模式因此而产生 */ var CompoundObject = {}; CompoundObject.string1 = 'default value'; CompoundObject.createChildObject = function(){ return { bool: true, num: 10 } }; CompoundObject.childObject = CompoundObject.createChildObject(); var compoundObjectClone = clone(CompoundObject); compoundObjectClone.childObject = CompoundObject.createChildObject(); compoundObjectClone.childObject.num = 5; document.write(compoundObjectClone.string1 + "<br/>"); document.write(compoundObjectClone.childObject.num); </script> </head> <body> <!-- Nothing in the body --> </body> </html>
到此,ext-base.js的所有概念都有了。下面开始具体分析extend类
<html> <head> <title>Extend Example</title> <script type="text/javascript"> //Ext类的第一个细胞原型 //骨架模式。采用Prototypal Inheritance的指针继承方式 //ext类是用隐私的实例化类开始进行构造的。 Ext = { version: '3.0' }; //Ext类之对象拷贝函数。实现对象与对象之间的直接复制。 /* * in这个操作符在IE中无法识别非枚举属性 * 以下非枚举属性toString(),hasOwnProperty() ,propertyIsEnumerable() , toLocaleString() , valueOf() * Ext的解决方法是用overrides重新写了apply这个函数,其中所有toString的方法全部由object类的方法覆盖。 * apply的另一变形是applyif。但是如果某属性存在,则不会被覆盖。 */ Ext.apply = function(o, c, defaults){ // no "this" reference for friendly out of scope calls if (defaults) { Ext.apply(o, defaults); } if (o && c && typeof c == 'object') { for (var p in c) { o[p] = c[p]; } } return o; }; Ext.apply(Ext, { //判断类型v是否是object类 isObject: function(v){ return v && typeof v == "object"; }, //Ext类之继承属性 /* * 闭包: * 1、作为一个函数变量的一个引用 - 当函数返回时,其处于激活状态。 * 2、一个闭包就是当一个函数返回时,一个没有释放资源的栈区。 */ //Ext是个匿名函数 extend: function(){ // inline overrides //这句是将所有的supperclass,subclass,overrides方法拷贝到io类备用。 var io = function(o){ for (var m in o) { this[m] = o[m]; } }; var oc = Object.prototype.constructor; return function(sb, sp, overrides){ //如果传递进来的第二个参数是object。 //拷贝子类到overrides类,如果overrides里面有个constructor属性, //就用overrides的constructor当作子类的构造函数。 //否则,创建个新的function,里面包含一句话,就是"sp.apply(this, arguments);", //这个又是闭包的一个应用,在退出extend方法之后并没有释放局部变量sp的内存空间。 //子类拷贝完成 if (Ext.isObject(sp)) { overrides = sp; sp = sb; sb = overrides.constructor != oc ? overrides.constructor : function(){ sp.apply(this, arguments); }; } //var F=function(){},定义一个空函数,里面没有属性。 //F.prototype=sp.prototype //sbp=new F()把F.prototype也就是sp.prototype里面的东西拷贝到sb.prototype //同时,因为F是个没有任何属性的函数,所以不需要再delete任何东西。 //将超类拷贝到了sbp。 //以父类为骨架,并将更多的方法和属性添加到子类中,代码值得借鉴 var F = function(){ }, sbp, spp = sp.prototype; F.prototype = spp; sbp = sb.prototype = new F(); //sb.prototype.constructor是F(),所以sbp.constructor=sb sbp.constructor = sb; sb.superclass = spp; if (spp.constructor == oc) { spp.constructor = sp; } sb.override = function(o){ Ext.override(sb, o); }; sbp.superclass = sbp.supr = (function(){ return spp; }); sbp.override = io; Ext.override(sb, overrides); sb.extend = function(o){ Ext.extend(sb, o); }; return sb; }; }(), //和apply一样的功能,复制对象。如果子类重写了toString,则覆盖父类。 override: function(origclass, overrides){ if (overrides) { var p = origclass.prototype; Ext.apply(p, overrides); if (Ext.isIE && overrides.toString != origclass.toString) { p.toString = overrides.toString; } } } }); //实例测试 //申请一个新的类S function S(){ } //S中两个变量s,s1 S.prototype.s = "s"; S.prototype.s1 = "s1"; //申请一个类C //C中两个变量c,c1 function C(){ this.c = "c"; this.c1 = "c1"; } //C继承S基类,并重写s1 Ext.extend(C, S, { s1: "by c overload" }); //实例化C var c = new C(); alert(c.s); //s alert(c.s1); //by c overload </script> </head> <body> <!-- Nothing in the body --> </body> </html>
相关推荐
初识JavaScript(源代码)初识JavaScript(源代码)初识JavaScript(源代码)初识JavaScript(源代码)初识JavaScript(源代码)初识JavaScript(源代码)初识JavaScript(源代码)初识JavaScript(源代码)初识...
标题 "第一篇 Ext初识" 暗示我们即将探讨的是ExtJS库的入门知识,这是一个基于JavaScript的前端开发框架,用于构建富客户端应用程序。在本文中,我们将深入理解ExtJS的基本概念、核心组件以及如何开始使用这个强大的...
初识C++ 初识C++ 初识C++初识C++初识C++初识C++初识C++
诗仙女炸鱼塘--js
### 01-初识 Node.js 与...通过以上介绍,我们可以了解到 Node.js 不仅扩展了 JavaScript 的应用场景,还极大地提高了开发效率。对于前端开发者而言,学习 Node.js 将有助于提升综合能力,为职业发展提供更多可能性。
接触javascript的时间说起来,长不长,短不短,其实真正开始学习它的时候应该是在去年大四11月份的时候,反正写网上的一些简单的特效,基本都是用jquery写得比较多,以致于对原生的js了解也是一知半解
这个"初识 Node.js-3课时 课件 源码.zip"文件显然是一个关于 Node.js 入门教程的压缩包,包含了三课时的课程内容以及源代码,适合初学者学习理解和实践。 ### 第一课时:Node.js 简介 在第一课时中,你可能会学习...
python 列表初识,通过此代码,你能够了解到python的列表操作
#### 一、初识Ext Ext是一个功能强大的JavaScript框架,用于构建桌面级Web应用程序。它简化了DOM操作、事件处理和Ajax交互,提供了丰富的UI组件,使得开发者能够轻松创建高度交互性的网页。本教程旨在引导读者从零...
### 05初识 Node.js-ev-api-server:构建API服务器详解 #### 一、项目概述与背景 在当今互联网时代,API(Application Programming Interface)已成为软件开发中的重要组成部分,它允许不同的应用程序之间相互通信...
### 03 初识 Node.js - Express #### 1.1 Express 简介 - **Express** 是一个基于 **Node.js** 平台,快速、开放且极简的 Web 开发框架。 - **通俗理解**: Express 类似于 Node.js 内置的 http 模块,主要用于创建...
初识云计算初识云计算初识云计算初识云计算初识云计算初识云计算初识云计算初识云计算
### 02-初识 Node.js 模块化 #### 模块化基本概念 - **模块化的定义**: - 模块化是一种软件设计技术,它将一个复杂的系统分解为更小、更易管理的部分,即“模块”。 - 在现实世界中,模块化的例子比比皆是,...
04 第四章 JavaScript对象及初识面向对象.md
初识js变量类型.md
### 04初识 Node.js-数据库与身份认证 #### 数据库的基本概念 数据库是用于组织、存储和管理数据的一种重要工具,在当前信息化社会中扮演着核心角色。无论是出行记录、消费历史还是浏览过的网页等内容,都可以被视...
对JavaScript的简单认识
jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识及JIT优化jvm初识...
三年级信息技术课程《初识画图》课件内容 因为要参加比赛,所属机房 装不上