JavaScript面向对象编程(1)

 

Javascript是一个类C的语言,他的面向对象的东西相对于C++/Java比较奇怪,但是其的确相当的强大,在 Todd 同学的“对象的消息模型”一文中我们已经可以看到一些端倪了。这两天有个前同事总在问我Javascript面向对象的东西,所以,索性写篇文章让他看去吧,这里这篇文章主要想从一个整体的角度来说明一下Javascript的面向对象的编程。

AD:


    Javascript是一个类C的语言,他的面向对象的东西相对于C++/Java比较奇怪,但是其的确相当的强大,在 Todd 同学的“对象的消息模型”一文中我们已经可以看到一些端倪了。这两天有个前同事总在问我Javascript面向对象的东西,所以,索性写篇文章让他看去吧,这里这篇文章主要想从一个整体的角度来说明一下Javascript的面向对象的编程。(成文比较仓促,应该有不准确或是有误的地方,请大家批评指正)

    另,这篇文章主要基于 ECMAScript 5, 旨在介绍新技术。关于兼容性的东西,请看最后一节。

    初探

    我们知道Javascript中的变量定义基本如下:

       
       
    1. var name = 'Chen Hao';;
    2. var email = 'haoel(@)hotmail.com';
    3. var website = 'http://coolshell.cn';

    如果要用对象来写的话,就是下面这个样子:

       
       
    1. var chenhao = {
    2. name :'Chen Hao',
    3. email : 'haoel(@)hotmail.com',
    4. website : 'http://coolshell.cn'
    5. };

    于是,我就可以这样访问:

       
       
    1. //以成员的方式
    2. chenhao.name;
    3. chenhao.email;
    4. chenhao.website;
    5. //以hash map的方式
    6. chenhao["name"];
    7. chenhao["email"];
    8. chenhao["website"];

    关于函数,我们知道Javascript的函数是这样的:

       
       
    1. var doSomething = function(){
    2. alert('Hello World.');
    3. };

    于是,我们可以这么干:

       
       
    1. var sayHello = function(){
    2. var hello = "Hello, I'm "+ this.name
    3. + ", my email is: " + this.email
    4. + ", my website is: " + this.website;
    5. alert(hello);
    6. };
    7. //直接赋值,这里很像C/C++的函数指针
    8. chenhao.Hello = sayHello;
    9. chenhao.Hello();

    相信这些东西都比较简单,大家都明白了。 可以看到javascript对象函数是直接声明,直接赋值,直接就用了。runtime的动态语言。

    还有一种比较规范的写法是:

       
       
    1. //我们可以看到, 其用function来做class。
    2. var Person = function(name, email, website){
    3. this.name = name;
    4. this.email = email;
    5. this.website = website;
    6. this.sayHello = function(){
    7. var hello = "Hello, I'm "+ this.name + ", \n" +
    8. "my email is: " + this.email + ", \n" +
    9. "my website is: " + this.website;
    10. alert(hello);
    11. };
    12. };
    13. var chenhao = new Person("Chen Hao", "haoel@hotmail.com",
    14. "http://coolshell.cn");
    15. chenhao.sayHello();

    顺便说一下,要删除对象的属性,很简单:

       
       
    1. delete chenhao['email']

    上面的这些例子,我们可以看到这样几点:

    ◆ Javascript的数据和成员封装很简单。没有类完全是对象操作。纯动态!

    ◆ Javascript function中的this指针很关键,如果没有的话,那就是局部变量或局部函数。

    ◆ Javascript对象成员函数可以在使用时临时声明,并把一个全局函数直接赋过去就好了。

    ◆ Javascript的成员函数可以在实例上进行修改,也就是说不同实例相同函数名的行为不一定一样。

    属性配置 – Object.defineProperty

    先看下面的代码:

       
       
    1. //创建对象
    2. var chenhao = Object.create(null);
    3. //设置一个属性
    4. Object.defineProperty( chenhao,
    5. 'name', { value: 'Chen Hao',
    6. writable: true,
    7. configurable: true,
    8. enumerable: true });
    9. //设置多个属性
    10. Object.defineProperties( chenhao,
    11. {
    12. 'email' : { value: 'haoel@hotmail.com',
    13. writable: true,
    14. configurable: true,
    15. enumerable: true },
    16. 'website': { value: 'http://coolshell.cn',
    17. writable: true,
    18. configurable: true,
    19. enumerable: true }
    20. }
    21. );

    下面就说说这些属性配置是什么意思。

    writable:这个属性的值是否可以改。

    configurable:这个属性的配置是否可以改。

    enumerable:这个属性是否能在for…in循环中遍历出来或在Object.keys中列举出来。

    value:属性值。

    get()/set(_value):get和set访问器。

    Get/Set 访问器

    关于get/set访问器,它的意思就是用get/set来取代value(其不能和value一起使用),示例如下:

       
       
    1. var age = 0;
    2. Object.defineProperty( chenhao,
    3. 'age', {
    4. get: function() {return age+1;},
    5. set: function(value) {age = value;}
    6. enumerable : true,
    7. configurable : true
    8. }
    9. );
    10. chenhao.age = 100; //调用set
    11. alert(chenhao.age); //调用get 输出101(get中+1了);

    我们再看一个更为实用的例子——利用已有的属性(age)通过get和set构造新的属性(birth_year):

       
       
    1. Object.defineProperty( chenhao,
    2. 'birth_year',
    3. {
    4. get: function() {
    5. var d = new Date();
    6. var y = d.getFullYear();
    7. return ( y - this.age );
    8. },
    9. set: function(year) {
    10. var d = new Date();
    11. var y = d.getFullYear();
    12. this.age = y - year;
    13. }
    14. }
    15. );
    16. alert(chenhao.birth_year);
    17. chenhao.birth_year = 2000;
    18. alert(chenhao.age);

    这样做好像有点麻烦,你说,我为什么不写成下面这个样子:

       
       
    1. var chenhao = {
    2. name: "Chen Hao",
    3. email: "haoel@hotmail.com",
    4. website: "http://coolshell.cn",
    5. age: 100,
    6. get birth_year() {
    7. var d = new Date();
    8. var y = d.getFullYear();
    9. return ( y - this.age );
    10. },
    11. set birth_year(year) {
    12. var d = new Date();
    13. var y = d.getFullYear();
    14. this.age = y - year;
    15. }
    16. };
    17. alert(chenhao.birth_year);
    18. chenhao.birth_year = 2000;
    19. alert(chenhao.age);

    是的,你的确可以这样的,不过通过defineProperty()你可以干这些事:

    1)设置如 writable,configurable,enumerable 等这类的属性配置。

    2)动态地为一个对象加属性。比如:一些HTML的DOM对像。

    查看对象属性配置

    如果查看并管理对象的这些配置,下面有个程序可以输出对象的属性和配置等东西:

       
       
    1. //列出对象的属性.
    2. function listProperties(obj)
    3. {
    4. var newLine = "<br />";
    5. var names = Object.getOwnPropertyNames(obj);
    6. for (var i = 0; i < names.length; i++) {
    7. var prop = names[i];
    8. document.write(prop + newLine);
    9. // 列出对象的属性配置(descriptor)动用getOwnPropertyDescriptor函数。
    10. var descriptor = Object.getOwnPropertyDescriptor(obj, prop);
    11. for (var attr in descriptor) {
    12. document.write("..." + attr + ': ' + descriptor[attr]);
    13. document.write(newLine);
    14. }
    15. document.write(newLine);
    16. }
    17. }
    18. listProperties(chenhao);

    call,apply, bind 和 this

    关于Javascript的this指针,和C++/Java很类似。 我们来看个示例:(这个示例很简单了,我就不多说了)

       
       
    1. function print(text){
    2. document.write(this.value + ' - ' + text+ '<br>');
    3. }
    4. var a = {value: 10, print : print};
    5. var b = {value: 20, print : print};
    6. print('hello');// this => global, output "undefined - hello"
    7. a.print('a');// this => a, output "10 - a"
    8. b.print('b'); // this => b, output "20 - b"
    9. a['print']('a'); // this => a, output "10 - a"

    我们再来看看call 和 apply,这两个函数的差别就是参数的样子不一样,另一个就是性能不一样,apply的性能要差很多。(关于性能,可到 JSPerf 上去跑跑看看)

       
       
    1. print.call(a, 'a'); // this => a, output "10 - a"
    2. print.call(b, 'b'); // this => b, output "20 - b"
    3. print.apply(a, ['a']); // this => a, output "10 - a"
    4. print.apply(b, ['b']); // this => b, output "20 - b"

    但是在bind后,this指针,可能会有不一样,但是因为Javascript是动态的。如下面的示例

       
       
    1. var p = print.bind(a);
    2. p('a'); // this => a, output "10 - a"
    3. p.call(b, 'b'); // this => a, output "10 - b"
    4. p.apply(b, ['b']); // this => a, output "10 - b"

    评论
    添加红包

    请填写红包祝福语或标题

    红包个数最小为10个

    红包金额最低5元

    当前余额3.43前往充值 >
    需支付:10.00
    成就一亿技术人!
    领取后你会自动成为博主和红包主的粉丝 规则
    hope_wisdom
    发出的红包
    实付
    使用余额支付
    点击重新获取
    扫码支付
    钱包余额 0

    抵扣说明:

    1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
    2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

    余额充值