zoukankan      html  css  js  c++  java
  • 如何编写可维护的面向对象JavaScript代码

    如何编写可维护的面向对象JavaScript代码

    能够写出可维护的面向对象JavaScript代码不仅可以节约金钱,还能让你很受欢迎。不信?有可能你自己或者其他什么人有一天会回来重用你的代码。如果能尽量让这个经历不那么痛苦,就可以节省不少时间。地球人都知道,时间就是金钱。同样的,你也会因为帮某人省去了头疼的过程而获得他的偏爱。但是,在开始探索如何编写可维护的面向对象JavaScript代码之前,我们先来快速看看什么是面向对象。如果已经了解面向对象的概念了,就可以直接跳过下一节。

     

    什么是面向对象?

     

    面向对象编程主要通过代码代表现实世界中的实质对象。要创建对象,首先需要写一个“类”来定义。 类几乎可以代表所有的东西:账户,员工,导航菜单,汽车,植物,广告,饮料,等等。而每次要创建对象的时候,就从类实例化一个对象。换句话说,就是创建类的实例做为对象。事实上,通常处理一个以上的同类事物时就会使用到对象。另外,只需要简单的函数式程序就可以做的很好。对象实质上是数据的容器。因此在一个employee对象中,你可能要储存员工号,姓名,入职日期,职称,工资,资历,等等。对象也包括处理数据的函数(也叫做“方法”)。方法被用作媒介来确保数据的完整性,以及在储存之前对数据进行转换。例如,方法可以接收任意格式的日期然后在储存之前将其转化成标准化格式。最后,类还可以继承其他的类。继承可以让你在不同类中重复使用相同代码。例如,银行账户和音像店账户都可以继承一个基本的账户类,里面包括个人信息,开户日期,分部信息,等等。然后每个都可以定义自己的交易或者借款处理等数据结构和方法。

     

    警告:JavaScript面向对象是不一样的

     

    在上一节中,概述了经典的面向对象编程的基本知识。说经典是因为JavaScript并不遵循这些规则。相反地,JavaScript的类是写成函数的样子,而继承则是通过原型实现的。原型继承基本上意味着使用原型属性来实现对象的继承,而不是从类继承类。
     
    • 对象的实例化
    以下是JavaScript中对象实例化的例子:
     
     1   // 定义Employee类
    2   function Employee(num, fname, lname) {
    3     this.getFullName = function () {
    4   return fname + " " + lname;
    5   }
    6   };
    7
    8   // 实例化Employee对象
    9   var john = new Employee("4815162342", "John", "Doe");
    10   alert("The employee's full name is " + john.getFullName());
     
    在这里,有三个重点需要注意:
     
    class”函数名的第一个字母要大写。这表明该函数的目的是被实例化而不是像一般函数一样被调用。
    在实例化的时候使用了new操作符。如果省略掉new仅仅调用函数则会产生很多问题。
    因为getFullName指定给this操作符了,所以是公共可用的,但是fnamelname则不是。由Employee函数产生的闭包给了getFullNamefnamelname入口,但同时对于其他类仍然是私有的。 
     
    • 原型继承
    下面是JavaScript中原型继承的例子:
     
    1 // 定义Human类
    2 function Human() {
    3 this.setName = function (fname, lname) {
    4 this.fname = fname;
    5 this.lname = lname;
    6 }
    7 this.getFullName = function () {
    8 return this.fname + " " + this.lname;
    9 }
    10 }
    11
    12 // 定义Employee类
    13 function Employee(num) {
    14 this.getNum = function () {
    15 return num;
    16 }
    17 };
    18 //让Employee继承Human类
    19 Employee.prototype = new Human();
    20
    21 // 实例化Employee对象
    22 var john = new Employee("4815162342");
    23 john.setName("John", "Doe");
    24 alert(john.getFullName() + "'s employee number is " + john.getNum());
     
    这一次,创建的Human类包含人类的一切共有属性——我也将fnamelname放进去了,因为不仅仅是员工才有名字,所有人都有名字。然后将Human对象赋值给它的prototype属性。

     

    通过继承实现代码重用

     

    在前面的例子中,原来的Employee类被分解成两个部分。所有的人类通用属性被移到了Human类中,然后让Employee继承Human。这样的话,Human里面的属性就可以被其他的对象使用,例如Student(学生),Client(顾客),Citizen(公民),Visitor(游客),等等。现在你可能注意到了,这是分割和重用代码很好的方式。处理Human对象时,只需要继承Human来使用已存在的属性,而不需要对每种不同的对象都重新一一创建。除此以外,如果要添加一个“中间名字”的属性,只需要加一次,那些继承了 Human 类的就可以立马使用了。反而言之,如果我们只是想要给一个对象加“中间名字”的属性,我们就直接加在那个对象里面,而不需要在Human 类里面加。
     

    1. Public(公有的)和Private(私有的)

     

    接下来的主题,我想谈谈类中的公有和私有变量。根据对象中处理数据的方式不同,数据会被处理为私有的或者公有的。私有属性并不一定意味着其他人无法访问。可能只是某个方法需要用到。 
     
    • 只读
    有时,你只是想要在创建对象的时候能有一个值。一旦创建,就不想要其他人再改变这个值。为了做到这点,可以创建一个私有变量,在实例化的时候给它赋值。
     
    1 function Animal(type) {
    2 var data = [];
    3 data['type'] = type;
    4 this.getType = function () {
    5 return data['type'];
    6 }
    7 }
    8
    9 var fluffy = new Animal('dog');
    10 fluffy.getType(); // 返回 'dog'
     
    在这个例子中,Animal类中创建了一个本地数组data。当 Animal对象被实例化时,传递了一个type的值并将该值放置在data数组中。因为它是私有的,所以该值无法被覆盖(Animal函数定义了它的范围)。一旦对象被实例化了,读取type值的唯一方式是调用getType方法。因为getType是在Animal中定义的,因此凭借Animal产生的闭包,getType可以进到data中。这样的话,虽可以读到对象的类型却无法改变。
     
    有一点非常重要,就是当对象被继承时,“只读”技术就无法运用。在执行继承后,每个实例化的对象都会共享那些只读变量并覆盖其值。最简单的解决办法是将类中的只读变量转换成公共变量。但是你必须保持它们是私有的,你可以使用Philippe在评论中提到的技术。 
     
    • Public(公有) 
    当然也有些时候你想要任意读写某个属性的值。要实现这一点,需要使用this操作符。
     
    1 function Animal() {
    2 this.mood = '';
    3 }
    4
    5 var fluffy = new Animal();
    6 fluffy.mood = 'happy';
    7 fluffy.mood; // 返回 'happy'
     
    这次Animal类公开了一个叫mood的属性,可以被随意读写。同样地,你还可以将函数指定给公有的属性,例如之前例子中的getType函数。只是要注意不要给getType赋值,不然的话你会毁了它的。 
     
    • 完全私有 
    最后,可能你发现你需要一个完全私有化的本地变量。这样的话,你可以使用与第一个例子中一样的模式而不需要创建公有方法。
     
    1 function Animal() {
    2 var secret = "You'll never know!"
    3 }
    4
    5 var fluffy = new Animal();

     

    2. 写灵活的API

    既然我们已经谈到类的创建,为了保持与产品需求变化同步,我们需要保持代码不过时。如果你已经做过某些项目或者是长期维护过某个产品,那么你就应该知道需求是变化的。这是一个不争的事实。如果你不是这么想的话,那么你的代码在还没有写之前就将注定荒废。可能你突然就需要将选项卡中的内容弄成动画形式,或是需要通过Ajax调用来获取数据。尽管准确预测未来是不大可能,但是却完全可以将代码写灵活以备将来不时之需。 
     
    • Saner参数列表
    在设计参数列表的时候可以让代码有前瞻性。参数列表是让别人实现你代码的主要接触点,如果没有设计好的话,是会很有问题的。你应该避免下面这样的参数列表:
     
    1 function Person(employeeId, fname, lname, tel, fax, email, email2, dob) {
    2 };
     
    这个类十分脆弱。如果在你发布代码后想要添加一个中间名参数,因为顺序问题,你不得不在列表的最后往上加。这让工作变得尴尬。如果你没有为每个参数赋值的话,将会十分困难。例如:
     
    1 var ara = new Person(1234, "Ara", "Pehlivanian", "514-555-1234", null, null, null, "1976-05-17");
     
    操作参数列表更整洁也更灵活的方式是使用这个模式:
     
    1 function Person(employeeId, data) {
    2 };
     
    有第一个参数因为这是必需的。剩下的就混在对象的里面,这样才可以灵活运用。
     
    1 var ara = new Person(1234, {
    2 fname: "Ara",
    3 lname: "Pehlivanian",
    4 tel: "514-555-1234",
    5 dob: "1976-05-17"
    6 });
     
    这个模式的漂亮之处在于它即方便阅读又高度灵活。注意到fax, emailemail2完全被忽略了。不仅如此,对象是没有特定顺序的,因此哪里方便就在哪里添加一个中间名参数是非常容易的:
     
    1 var ara = new Person(1234, {
    2 fname: "Ara",
    3 mname: "Chris",
    4 lname: "Pehlivanian",
    5 tel: "514-555-1234",
    6 dob: "1976-05-17"
    7 });
     
    类里面的代码不重要,因为里面的值可以通过索引来访问:
     
    1 function Person(employeeId, data) {
    2 this.fname = data['fname'];
    3 };
     
    如果data['fname'] 返回一个值,那么他就被设定好了。否则的话,没被设定好,也没有什么损失。
     
    • 让代码可嵌入
    随着时间流逝,产品需求可能对你类的行为有更多的要求。而该行为却与你类的核心功能没有半毛钱关系。也有可能是类的唯一一种实现,好比在一个选项卡的面板获取另一个选项卡的外部数据时,将这个选项卡面板中的内容变灰。你可能想把这些功能放在类的里面,但是它们不属于那里。选项卡条的责任在于管理选项卡。动画和获取数据是完全不同的两码事,也必须与选项卡条的代码分开。唯一一个让你的选项卡条不过时而又将那些额外的功能排除在外的方法是,允许将行为嵌入到代码当中。换句话说,通过创建事件,让它们在你的代码中与关键时刻挂钩,例如onTabChange, afterTabChange, onShowPanel, afterShowPanel等等。那样的话,他们可以轻易地与你的onShowPanel事件挂钩,写一个将面板内容变灰的处理器,这样就皆大欢喜了。JavaScript库让你可以足够容易地做到这一点,但是你自己写也不那么难。下面是使用YUI 3的一个例子。
     
    1 <script type="text/javascript" src="http://yui.yahooapis.com/combo?3.2.0/build/yui/yui-min.js"></script>
    2 <script type="text/javascript">
    3 YUI().use('event', function (Y) {
    4
    5 function TabStrip() {
    6 this.showPanel = function () {
    7 this.fire('onShowPanel');
    8
    9 // 展现面板的代码
    10
    11 this.fire('afterShowPanel');
    12 };
    13 };
    14
    15 // 让TabStrip有能力激发常用事件
    16 Y.augment(TabStrip, Y.EventTarget);
    17
    18 var ts = new TabStrip();
    19
    20 // 给TabStrip的这个实例创建常用时间处理器
    21 ts.on('onShowPanel', function () {
    22 //在展示面板之前要做的事
    23 });
    24 ts.on('onShowPanel', function () {
    25 //在展示面板之前要做的其他事
    26 });
    27 ts.on('afterShowPanel', function () {
    28 //在展示面板之后要做的事
    29 });
    30
    31 ts.showPanel();
    32 });
    33 </script>
     
    这个例子有一个简单的 TabStrip 类,其中有个showPanel方法。这个方法激发两个事件,onShowPanelafterShowPanel。这个能力是通过用Y.EventTarget扩大类来实现的。一旦做成,我们就实例化了一个TabStrip对象,并将一堆处理器都分配给它。这是用来处理实例的唯一行为而又能避免混乱当前类的常用代码。
     

    总结

    如果你打算重用代码,无论是在同一网页,同一网站还是跨项目操作,考虑一下在类里面将其打包和组织起来。面向对象JavaScript很自然地帮助实现更好的代码组织以及代码重用。除此以外,有点远见的你可以确保代码具有足够的灵活性,可以在你写完代码后持续使用很长时间。编写可重用的不过时JavaScript代码可以节省你,你的团队还有你公司的时间和金钱。这绝对能让你大受欢迎。
     
    大家有空移步到我的google blogger转转,那里有的不只是技术:www.yan-life.eu

    最近的项目中,一直要使用到WebService,为了测试自己编写的WebService,就找了一个叫Storm的小工具。它帮了我很大的忙,所以我和大家分享一下。

    Storm

     

    是一款用于测试Web服务的免费并且开源的工具。是用F#编写的,如果感兴趣的话,你可以在codeplex上找到它的源码。

     

    Storm特性:

     

    1. 可以测试.NET、Java等多种编程语言所写的Web服务;
    2. 动态调用Web服务方法,即使需要输入和复杂数据类型;
    3. 节约开发时间和花费(仅仅为测试Web服务而开发throw-away测试客户端太过浪费);
    4. 在一个UI中测试多个Web服务;
    5. 编辑、控制原始SOAP请求;
    6. 其它(自己探索,乐趣更多!)。

    运行环境 : .NET 2.0及以上,F# 1.9.3.14 (可选)

    Storm快速上手: 

     

    1. 添加一个WebService(如下图一所示):点击添加,输入wsdl的endpoint url,比如

      - http://www.deeptraining.com/webservices/weather.asmx
      - http://api.google.com/GoogleSearch.wsdl

    2. 选择一个Web方法:让storm分析这个web方法,然后给出调用这个方法所需的参数。
    3. 选择一个Web方法参数,并且输入必要的值。
    4. 点击GO(绿色箭头按钮):调用web方法。(结果如下图二所示)

    图一: 添加WebServie

    图二 : 调用WebServe方法返回结果

    Storm快捷操作:

     

    • 加载WebService并且测试它的方法
      1. 在工具栏中点击ADD
      2. 提供Wsdl的endpoint Url
      3. 点击GO(绿色箭头按钮)
      4. 选择一个Web方法或点击右键,并在新标签页中打开它
      5. 点击SEND
    • 保存SOAP请求作为测试用例
      1. 打开WebService
      2. 在树视图中(闪电图标),选择一个Web方法
      3. 点击SEND
      4. 在SOAP输入框中,切换到“原始视图”或“XML视图”
      5. 编辑数据
      6. 点击SAVE。选择“Test Data”选项,填写测试用例的标签,附注等。点击OK
      7. 现在,原始的SOAP请求消息将是该项目的一部分
      8. 同样,你可以在SOAP响应输出框中用步骤4-7保存SOAP响应作为预期响应。
    • 保存当前会话为storm项目
      1. 假设你已经打开了一个带或不带测试用例的WebService
      2. 点击工具栏中的SAVE按钮
      3. 填写项目名称,单击OK:将创建下列文件结构

        {File} StormProject.stormproj
        - {Folder} ServiceName
        - {Folder} WebMethod
        - {Folder} TestCaseName
        - {File} TestInputLabel_testinput.xml
        - {File} WebMethod_test.config
        - {File} ServiceName_ws.config

    • 载入已保存的Storm项目
      1. 创建一个storm项目
      2. 在工具栏中单击OPEN,选择已创建的storm项目文件
      3. 点击OK:加载WebService和所有测试用例。
    • 执行已保存的测试用例
      1. 打开一个storm项目
      2. 右击一个测试用例节点。选择Run/EDIT:这将打开一个新标签页
      3. 根据需要编辑测试输入(点击Edit图标)
      4. 点击RUN来调用Web方法
      5. 如果是绿色,就意味着实际响应和预期响应一样
    • 发送多个数组项
      1. 加载WebService
      2. 选择一个接受数组为输入参数的Web方法
      3. 点击“树视图”,编辑
      4. 点击RUN来调用Web方法
      5. 切换到“原始视图”。根据需要编辑SOAP请求,只要你喜欢,你可以添加任意多的数组项。但是要确保,你编辑后,SOAP请求还是正确的XML格式。(可以使用XML视图确认。)
    • 发送多个成员数组项
      1. 加载WebService
      2. 选择一个接受类作为输入参数的Web方法(这个类含有数组成员)
      3. 点击“树视图”(闪电图标)
      4. 选择类:弹出对象编辑框
      5. 在对象编辑框中,选择数组成员
      6. 添加数组项

    Un ciel nappé de nuages!

    大家有空移步到我的google blogger转转,那里有的不只是技术:www.yan-life.eu
  • 相关阅读:
    手游页游和端游的服务端的架构与区别
    TiKV 源码解析系列——如何使用 Raft
    TiKV 源码解析系列
    三篇文章了解 TiDB 技术内幕 —— 谈调度
    三篇文章了解 TiDB 技术内幕——说计算
    三篇文章了解 TiDB 技术内幕——说存储
    TiDB 源码阅读系列文章(一)序
    【合集】TiDB 源码阅读系列文章
    9个offer,12家公司,35场面试,从微软到谷歌,应届计算机毕业生的2012求职之路
    python datetime和unix时间戳之间相互转换
  • 原文地址:https://www.cnblogs.com/Leo_wl/p/2438266.html
Copyright © 2011-2022 走看看