对于熟悉C#和Java的兄弟们,面向对象的三大思想(封装,继承,多态)肯定是了解的,那么如何在Javascript中利用封装这个特性呢?

我们会把现实中的一些事物抽象成一个Class并且把事物的属性(名词)作为Class的Property把事物的动作(动词)作为Class的methods。在面向对象的语言中(C#等)都会有一些关键字来修饰类或者属性(Private,public,protect),这些关键词描述了访问的权限,不多做解释。

我们来看看Javascript的易变的特性(我们还用上一次的例子):

01 var Man = function (name, age) {
02           this.Name = name;
03           this.Age = age;
04     }
05    var Person = new Interface("Person", ["GetName""GetAge"]);
06          Man.prototype = { GetName: function () { return this.Name; },
07          GetAge: function () { return this.Age; }
08      }    
09    var Gonn = new Man("Gonn", 25);
10    alert(Gonn.GetAge());
11  Gonn.DisplayAll = function () { return "Name: "+this.GetName() + "; Age: " this.GetAge() }
12 alert(Gonn.DisplayAll());

先创建了一个Class(Javascript的匿名方法)拥有2个公共的(public)的字段(本篇blog会详细讲解,继续往下看)和2个public的方法,我们创建了一个Instance--Gonn,但是我可以为这个Instance动态的添加一个DisplayAll的方法,我想任何面向对象的语言是做不到这一点的,Javascript的灵活体现之一。

现在假设一个场景,如果有很多的程序员要用这段代码,由于Javascript的易变性,程序员就可以在实例化后改变Name的值,那初始化的动作就没有意义了:

1 var Gonn = new Man("Gonn", 25);
2 Gonn.Name = "Alice";
3 alert(Gonn.GetName());

所以我们不能让外部的人去任意的修改这个字段,在Java或C#中我们只需要个这个字段改为Private,就万事OK了,但是Javascript没有这个关键词,那我们需要这么做呢。我们可以想下在C#除了设置Private之外我们还可以怎么做?我们可以设置Setter和Getter方法。

我们来修改下上面的代码:我们称方法一:

01 var Person = new Interface("Person", ["SetName""SetAge""GetName","GetAge"]);
02        var Man = function (name, age) {
03            this.SetAge(age);
04            this.SetName(name);
05        }
06        Man.prototype = {
07            SetName: function (name) { this.Name = name; },
08            SetAge: function (age) { this.Age = age; },
09            GetName: function () { return this.Name; },
10            GetAge: function () { return this.Age; }
11        }
12       var Alan = new Man("Alan", 25);
13       Alan.Name = "Alice"//悲剧了,我alert的时候变成Alice了
14       Alan.SetAge(10);//悲剧,被别人把我的年龄给这么小
15       alert(Alan.GetName());
16       Alan.DisplayAll = function () { return "Name: "+this.GetName() + "; Age: " this.GetAge() }
17       alert(Alan.DisplayAll());

貌似样子很像C#中的Setter和Getter,但是还是可以被外部修改。但是从约束上来看,貌似比上面的code要好看些,通过方法来设置初始值。但是问题还是没有解决,我们来看看下面一种方法:闭包。解释一下,在Javascript中是通过This关键字来开发权限的(Public)。在讲闭包之前,我们需要了解下闭包的本质: 在Javascript中,只有方法是有作用域的,如果在方法中声明的变量在外部是无法访问的,那Private的概念就出来了。

01 var Person = new Interface("Person", ["SetName""SetAge""GetName","GetAge"]);
02        var Man = function (newname, newage) {
03                var name, age;
04                this.SetName = function (newname) { name = newname; }
05                this.SetAge = function (newage) { age = newage; }
06                this.GetName = function () { return name; }
07                this.GetAge = function () { return age; }
08                this.SetAge(newage);
09                this.SetName(newname);    
10        }
11        var Alan = new Man("Alan", 25);
12        Alan.name="Alice"//现在name是private了,我是无法去修改的
13        Alan.SetAge(10);
14       alert(Alan.GetAge());

现在私有的功能就实现了,我们只是用Var来代替了This而已。//我们把公共(Public)并且可以访问Private的方法称为特权方法,比如上面的this.SetName, this.SetAge.

如果我们的公共方法不涉及到访问Private的字段,那我们可以把他们放到Prototype中。//好处是多个实例的时候内存中也只有一分拷贝

1 Man.prototype.DisplayAll = function () { return "Name: " this.GetName() + "; Age: " this.GetAge() }

哈哈~我们来看下稍微有点难度的东西:静态变量和方法。

我们都是知道静态的东西属于类(Class),我们来修改下上面的代码:

01 var Person = new Interface("Person", ["SetName""SetAge""GetName","GetAge","GetCount"]);
02       var Man = (function () {
03           var count = 0;
04           return function (newname, newage) {
05               var name, age;
06               this.SetName = function (newname) { name = newname; }
07               this.SetAge = function (newage) { age = newage; }
08               this.GetName = function () { return name; }
09               this.GetAge = function () { return age; }
10               this.GetCount = function () { return count; }
11               this.SetAge(newage);
12               this.SetName(newname);
13               count++;
14           }
15       })();
16           Man.prototype.DisplayAll = function () { return "Name: " +this.GetName() + "; Age: " this.GetAge() }
17           var Alan1 = new Man("Alan", 25);
18           var Alan2 = new Man("Alan", 25);
19      alert("There are "+Alan2.GetCount()+" instances of Man" );

不管我们是通过Alan1或Alan2去GetCount,结果都一样都是2. 这里count就是一个私有的静态变量。

如何在Javascript中利用封装这个特性的更多相关文章

  1. 深入理解JavaScript中的属性和特性

    深入理解JavaScript中的属性和特性 JavaScript中属性和特性是完全不同的两个概念,这里我将根据自己所学,来深入理解JavaScript中的属性和特性. 主要内容如下: 理解JavaSc ...

  2. 2018.2.2 JavaScript中的封装

    JavaScript中的封装 1.封装的概念 通过将一个方法或者属性声明为私用的,可以让对象的实现细节对其他对象保密以降低对象之间的耦合程度,可以保持数据的完整性并对其修改方式加以约束,这样可以使代码 ...

  3. JavaScript中面向对象的三大特性(一个菜鸟的不正经日常)

    经过几天的学习,把jQuery给啃会了,但是运用的还不算特别熟练,总感觉自己在JavaScript方面的基础十分欠缺,所以继续拾起JavaScript,开始更好的编程之旅~ 今天学的是JavaScri ...

  4. 如何在 JavaScript 中检查字符串是否包含子字符串?

    如何在 JavaScript 中检查字符串是否包含子字符串? // var test4 = _.includes(string, substring); 该方法需要此文件 <script src ...

  5. 第一百四十三节,JavaScript,利用封装库做百度分享

    JavaScript,利用封装库做百度分享 效果图 html代码 <div id="share"> <h2>分享到</h2> <ul> ...

  6. JavaScript中利用Ajax 实现客户端与服务器端通信(九)

    一:Ajax (Asynchronous JavaScript and XML)不是一个新的技术,事实上,它是一些旧有的成熟的技术以一种全新的更加强大的方式整合在一起 Ajax的关键技术: 1.使用X ...

  7. 如何在JavaScript中使用高阶函数

    将另一个函数作为参数的函数,或者定义一个函数作为返回值的函数,被称为高阶函数. JavaScript可以接受高阶函数.这种处理高阶函数的能力以及其他特点,使JavaScript成为非常适合函数式编程的 ...

  8. 如何在JavaScript中正确引用某个方法(bind方法的应用)

    在JavaScript中,方法往往涉及到上下文,也就是this,因此往往不能直接引用,就拿最常见的console.log("info…")来说,避免书写冗长的console,直接用 ...

  9. javaScript中利用ActiveXObject来创建FileSystemObject操作文件

    注:如果用javascript读本地文件,遇到安全问题. 需在浏览器中进行设置,如下:     工具—> Internet选项->安全->自定义级别->启用“没有标识为安全的A ...

随机推荐

  1. python mac下安装虚拟环境

    Mac 下 Flask 框架 workon命令找不到 ---- 最终解决方案(详解具体实现操作过程中遇到的坑) Mac 下 Flask 的 全网最详细搭建 1.安装virtualenv和virtual ...

  2. Flask初学者:视图函数和类视图

    当一个url请求进入后台时,一般有两种方式来进行处理:视图函数和类视图.视图函数直接使用一个函数来进行处理并返回数据给浏览器,类视图则是使用类来进行处理并返回的,所以当需要进行的处理比较简单,则可以考 ...

  3. ecshop 全系列版本网站漏洞 远程代码执行sql注入漏洞

    ecshop漏洞于2018年9月12日被某安全组织披露爆出,该漏洞受影响范围较广,ecshop2.73版本以及目前最新的3.0.3.6.4.0版本都受此次ecshop漏洞的影响,主要漏洞是利用远程代码 ...

  4. IO复用——select系统调用

    1.select函数 此函数用于在一段时间内,监听用户感兴趣的文件描述符上的可读.可写和异常等事件. #include<sys/select.h> int select(int nfds, ...

  5. Redis 在springBoot中的一个使用示例

    在现系统中使用了一个字典表,更新或插入字典表需要做Redis缓存 @Override @Cache(name = Constants.REDIS_PREFIX_DIC, desc = "变更 ...

  6. hdu畅通工程(并查集)

    Problem Description 某省调查城镇交通状况,得到现有城镇道路统计表,表中列出了每条道路直接连通的城镇.省政府“畅通工程”的目标是使全省任何两个城镇间都可以实现交通(但不一定有直接的道 ...

  7. VS中的一些标记

    1.//ToDO:此标记运行时会显示在任务列表窗口中.

  8. leetcode笔记--2 reverse string

    my answer: 出错点:new_list[s] = list_s[u-1-s] 这样会出错, 重点:(1) map(str, s) 函数的使用,例:ls = [1,2,3]rs = map(st ...

  9. vue3.0 部署的基础流程

    1.创建vue.config.js 主要是负责做设置的 2.修改vue.config.js 参考官方说明: 注意:对于本地开发的同学要注意,你之前在处理网络请求时是在8080端口下请求,现在如果换成了 ...

  10. 使用JDK自带的keytool工具生成证书

    一.keytool 简介 keytool 是java用于管理密钥和证书的工具,它使用户能够管理自己的公钥/私钥对及相关证书,用于(通过数字签名)自我认证(用户向别的用户/服务认证自己)或数据完整性以及 ...