<<Javascript Patterns>>阅读笔记 -- 第2章 基本技巧(一)

第一次写这种东西, 有些生涩和蹩脚, 也是为了自己在表达或是总结方面有所提高, 同时为看过的东西留个痕迹, 以便日后查阅.

有错误或是不妥的地方, 还望各位指正, 谢谢!

第1章 简介

本章主要介绍了本书要讨论的内容, Javascript的一些基本概念, 面向对象, 原型, ECMASCript5, JSLine和console的相关内容.

不做过多阐述, 给出模式的定义

广义上: 模式是一个可以用来产生其他事物的模板或模型.

在软件开发中, 模式是指一个通用问题的解决方案.

书中主要讨论三种类型的模式: 设计模式, 编码模式, 反模式.

所谓反模式, 不是bug, 但使用它时, 引发的问题往往比解决的问题多...

第2章 基本技巧

编写可维护的代码

不用多说, 大家都有过阅读别人写的代码的历经, 确实阅读代码比写代码需要更多的时间.

每个人的想法是一样的, 甚至同一个人现在的想法与一周前的也会不一样, 写出的代码也就有所不同.

如果这些代码不易于维护, 可读性差, 那就更消耗时间.

易于维护的代码应具备如下特性:

  • 可读性好
  • 具有一致性
  • 预见性好
  • 看起来如同一个人编写
  • 有文档

尽量少用全局变量

全局变量意味着在整个js应用或是页面中, 它们被共享. 有极大的可能出现命名冲突.

例如在一个脚本里定义了全局变量result, 后来又在某函数中定义了一个也叫result的全局变量,

那么就会导致脚本中的result变量被覆盖, 脚本无法正常工作.

首要的规则就是用var定义变量, 如下:

function sum(x, y) {
    var result = x + y;
    // 反模式: 使用了全局变量
    // result = x + y;
    return result;
}

另外, 在使用var定义变量时, 不要使用链式赋值. 例如:

// 反模式, 不要使用
function fun() {
    var a = b = 7;
    // a为局部变量, b为全部变量
    // 原因: 从右至左的操作符优先级, var a = (b = 0);
    // ...
}
// 如果对链接赋值的变量都进行了声明, 则不会出现全局变量, 如:
function fun() {
    var a, b;
    a = b = 8;
}

明确定义的全局变量与隐式定义的全局变量有微小的不同, 不同之处在于能否用delete撤消变量

  • 使用var定义的全局变量(在函数外创建)不能删除.
  • 不使用var定义的全局变量(在函数内创建)可以删除.

这说明隐式定义的全局变量严格来说不是真正的变量, 真正的变量不能被删除的, 它只是全局对象的一个属性, 只有属性可以被delete的掉

借用书中的例子:

// 定义三个变量
var global_var = 1;
global_novar = 2; // 反模式
(function(){
    global_fromfun = 3; // 反模式
}());
// 试着执行delete
delete global_var;
delete global_novar;
delete global_fromfun;
// 查看操作结果
console.info(typeof global_var); // number
console.info(typeof global_novar); // undefined
console.info(typeof global_fromfun); // undefined

注: ES5的strict模式中, 为没有声明的变量赋值会抛出错误.

单一var定义变量

可提高代码可读性, 同时在声明变量时也可同时赋值或做一些其他的操作. 例如:

function func() {
    var x = 2,
        y = 3,
        sum = x + y,
        obj = {

        };
    // ...
}

提升: 凌散变量问题

所谓提升, 就是在Javascipt函数里, 允许在任意地方声明多个变量, 无论在哪里声明, 效果都跟在函数的顶部声明一样.

问题举例:

name = ‘global‘; // 全局变量
(function func() {
    console.info(name); // undefined
    var name = ‘local‘;
    console.info(name); // local
}());
// 局部变量name的声明被提升了, 等同于下面这样:
name = ‘global‘; // 全局变量
(function func() {
    var name; // 等同于 ==> var name = undefined;
    console.info(name); // undefined
    name = ‘local‘;
    console.info(name); // local
}());

所以, 建议, 最好在函数开始就声明好要用的所有变量.

关于for循环

重点是缓存数组或类数组的长度

for(var i=0; i<arr.length; i++) {
    // ...
}

像这种访问方式, 每次循环都会访问数据的长度, 这样会使得代码变慢,

尤其是当arr不是真正的数组, 而是HTML集合对象时更为严重, 比如:

document.getElementsByName();

document.getElementsByTagName();

document.getElementsByClassName();

集合对象是DOM下的活动对象, 每次访问在它们的长度也就是在访问活动的DOM, DOM操作是相当耗时的.

缓存数据长度的for循环如下:

for(var i=0, len=arr.length; i<len; i++) {
    // ...
}

另外, 书中提到两个更有效率的循环方式, 有兴趣的可以学习一下.

主要就是使用最少的变量, 且逐步减至0, 因为与0的比较会更有效率, 代码如下:

// 第一个仍然是for循环
for(var i=arr.length; i--;) {
    // ...
}
// 第二个是while
var i = arr.length;
while(i--) {
    // ...
}
时间: 2024-12-13 08:42:38

<<Javascript Patterns>>阅读笔记 -- 第2章 基本技巧(一)的相关文章

&lt;&lt;Javascript Patterns&gt;&gt;阅读笔记 -- 第2章 基本技巧(二)

关于for-in循环 循环数据时, 强烈不推荐使用for-in循环.因为当Array对象被扩展后, 再用for-in循环遍历数据会导致逻辑上的错误, 举例说明: var arr = ['a', 'b', 'c']; // 下标遍历 for(var i=0, len=arr.length; i<len; i++) { console.info(typeof i); // number console.info(i); } // for-in遍历 for(var i in arr) { consol

&lt;&lt;Javascript Patterns&gt;&gt;阅读笔记 – 第3章 字面量和构造函数

对象字面量 首先给出对象字面量的定义语法: 1. 将对象定义在一对括号中(左大括号“{”和右大括号”}”) 2. 对象中以逗号分隔属性和方法. 每个属性或方法以key-value的形式出现, key和value之间以冒号分割. 3. 当给变量赋值时, 不要忘记或大括号后面的分号 空对象 var obj = {}; 这样就定义了一个空的对象, 但它并非什么也没有, 至少它具有从Object.prototype中继承下来的属性和方法. 来自构造函数的对象 语法: // 反模式, 不推荐这么用 Var

C++PRIMER 阅读笔记 第三章

本章主要介绍 string vector 和 bitset, 不能贪多,现在本文主要介绍 string 与 vector 头文件中最好不要使用namespace std, 因为头文件会直接被预处理器放置到C中 std::string 的构造方式: string s1; string s2(s1); string s3("value");string s4(n,'c');           第三种构造方式可有"\0"? cout 与 cin 可以直接操作string

人月神话阅读笔记—第三章

人月神话阅读笔记-第三章 一个由一流人才组成的小型的精干的队伍比由一群平庸的程序员组成的大型团队更有效率,但是这里面有一个重要的问题:如何在有意义的进度安排内创建大型的系统. 优秀的程序员和较差的程序员之间生产效率的差距是巨大的,当一个10人的精干团队进行开发,和一个100个人的平庸程序员进行开发,前者的效率更高.但是对于效率和概念的完成性来说,最好由少数干练人员开发,而大型系统需要大量人员进行开发,但是这两者是有矛盾的,需要进行平衡. 在开发过程中,可以使用一种崭新的开发方案,类似于外科医生做

构建之法阅读笔记05-第六章

阅读笔记 第六章:敏捷流程 第六章敏捷流程主要介绍了什么是敏捷流程及其原则,还有什么时候可以选择敏捷的开发方法,什么时候选择其他方法. 敏捷的流程是指一系列价值观和方法论的集合.介绍了一些敏捷开发原则,比如,经常发布可用的软件,业务人员和开发人员在项目开发过程中应该每天共同工作,面对面的交流始终是最有效的沟通方式,不断关注技术和设计,保持简明,团队要学会自我管理,时时总结如何提高团队效率,并付诸行动. 敏捷流程的方法论---Scrum方法论.首先第一步需要找出完成产品需要做的事情,然后决定当前的

构建之法阅读笔记08-第十一章

阅读笔记 第十一章:软件设计与实现 在第十一章的软件设计与实现方面,介绍了一些关于典型的开发流程和开发阶段的一些管理方法. 在拿到设计文档之后,还需要做一些其他事情,比如估计任务所需要的时间,写一些原型代码,看看效果:做代码的自我复审,进行重构:写单元测试等等.最后还要把修改集集成到代码库中. 开发人员有一个标准的工作流程:进行功能需求分析,复审设计文档,详细设计,实现设计来编写代码,同伴复审,源代码的合并.构建等等,其中的每一步都有可能出现bug,要随时发现并且修改bug,最后是测试完成,发布

人月神话阅读笔记—第四章

人月神话阅读笔记-第四章 ------2016.6.14 概念的完整性是很重要的,为了反应一系列连贯的设计思路,可以省略一些不规则的特性和改进,不提倡独立和无法整合的系统,最需要的是在整体概念上的完整性要求. 获得概念的完整性时,会出现一种情况,编程系统使计算机更加好用,但是功能比较多的时候,软件外部描述就会比系统本身大很多:但是功能太少,不能满足需求,但是都需要满足概念上的完整性. 在进行概念的完整性时,产品设计需要由一个人或者少数几个人来实现,但是对于大型的系统,需要将设计方法.体系结构的工

《领域驱动设计》阅读笔记 第1章 消化知识

ddd小白,一篇章节便能激起了心中涟漪,感慨之初,记于笔下. 第1章  消化知识 用醍醐灌顶.茅塞顿开来形容此章短短的文字,实不为过. 简单介绍背景:旅游互联网,B2B,初创公司.产品设计-代码开发的衔接有过两种明显形式: 1. 项目的推进由产品部起头,收集.分析.过滤需求,形成原型文档(word,excel,visio,axure),提交CTO.CEO评审(整个产品90%的原型.流程.字段),交付开发.测试工程师. 开发工程师花一两天理解.讨论原型文档,而后建立数据库表,开撸代码,按模块交付测

深入理解 C 指针阅读笔记 -- 第六章

Chapter6.h #ifndef __CHAPTER_6_ #define __CHAPTER_6_ /*<深入理解C指针>学习笔记 -- 第六章*/ typedef struct __person { char* name; char* title; unsigned int age; }person; /*结构体内存的释放问题*/ void __struct_memory_test(); #endif Chapter6.cpp #include "Chapter6.h&quo