Skip to content

Latest commit

 

History

History
450 lines (329 loc) · 10.7 KB

README.md

File metadata and controls

450 lines (329 loc) · 10.7 KB

javascript best practices TODO

  • 三目运算符

    常规代码:

      var authorized = false;
      var active;
      
      if(authorized) {
        active = true;
      } else {
        active = false;
      }  
    

    好的实践:

      var active = authorized ? true : false;
    

    注: 三目运算符始终会返回一个结果。条件判断,除了false, 0, undefined, NaN, "", null,其它的都为true。

    延伸:

      //执行函数
      authorized ? function(){
                     console.log('user is active...');
                   }()
                 : function(){
                     console.log('user is not active...');
                   }();
                   
      //多个表达式             
      var a,b;
      authorized ? (a = 1, b = 2) : (a = b = 0);
      
      //嵌套
      var c; 
      authorized ? (a = 1, b = 2) : authorized ? c = 3 : c = 0;
    
  • 使用条件||和&&

    常规代码:

      var class = {
        _addMember: function(member){
          this.members = this.members ? this.members : [];
          this.members.push(member);
        }
      };
    

    好的实践:

      var class = {
        _addMember: function(member){
          this.members = this.members || [];
          this.members.push(member);
        }
      }
    

    注: 这里主要注意||操作符前面表达式得出的结果是true还是false, true的话,执行短路操作,也就是碰到第一个表达式true的就返回。如果是&&的话,一直要判断到最后才能确定最终表达式的值。

  • switch语句使用

    常规代码:

      function findKindVal(kind){
        if(kind == 'a') {
          return 0;
        } else if(kind == 'b') {
          return 1;
        } else if(kind == 'c') {
        	return 2;
        }...
      }
    

    好的实践:

      function findKindVal(kind) {
        switch(kind) {
        	case 'a':
        	  return 0;
        	case 'b':
        	  return 1;
        	case 'c':
        	  return 2;
        	case 'd':
        	case 'e':
        	  return 3;
        	default:
        	  return '-1';
        }
      }
    

    注: 这里是直接return了,如果不是的话,需要加上break中断条件判断。这里可以把一些命中率高的选项放在前面。

  • 优化循环

    常规代码:

      var tasks = {
        numbers: [1,2,3,4,5,6,7,8,9]
      };
      
      for(var i = 0; i < tasks.numbers.length; i++) {
        console.log(tasks.numbers[i]);
      }
    

    好的实践:

      var numbers = tasks.numbers,
      	len = numbers.length;
      	
      for(var i = 0; i < len; i++){
        console.log(numbers[i])
      }
      
      //或者
      var i = len;
      while(i--){
        console.log(numbers[len - (i + 1)]);
      }
    

    注: 这里要访问i, numbers, length, 数组索引[i],效率会低一些,和0的比较效率是最高的。

  • 执行脚本

    常规代码:

      <head>
        <script src="..."></script>
        ...
      </head>
      <body>
        <script src="..."></script>
        //显示页面内容
      </body>
    

    问题: 如果脚本执行的业务逻辑过多,会导致页面内容无法立刻呈现在用户面前,这里应该将页面在显示给用户之前需要处理的业务逻辑,放在页面加载前处理,之后的放在底部。或者使用HTML5的async属性允许页面显示在脚本运行前。

    好的实践 1:

      <body>
      //显示页面内容
      <script src="..."><script>
      </body>
    

    好的实践 2:

      <head>
        <script src="..." async></script>
      </head>
      //...
    
  • 使用继承高效使用内存

    使一类对象共有的方法都使用构造器原型。每个对象都共用方法,而各自特有的属性,在实例化之后可以自定义。

    常规代码:

      function ObjectA () {
        this.methodA = function(){}
        this.methodB = function(){}
      }
      
      var objecta = new ObjectA();
    

    好的实践:

      ObjectA.prototype = {
        methodA: function(){},
        methodB: function(){}
      };
    
  • 使用createDocumentFragment添加DOM

    页面添加DOM常规做法是获取页面DOM节点,然后添加新的DOM。每一次添加DOM都会刷新页面,消耗时间而且降低效率。使用createDocumentFragment方法将新节点添加到一个fragment state上,最后将fragment上的所有节点一次性添加到页面DOM上。

    常规代码:

    var list = document.getElementById('book_list');
    for(var i = 0; i < 10; i++) {
      var element = document.createElement('li');
      element.appendChild(document.createTextNode('book_' + i));
      list.appendChild(element);
    }
    

    好的实践:

    var list = document.getElementById('book_list');
    var fragment = document.createDocumentFragment();
    
    for(var i = 0; i < 10; i++) {
      var element = document.createElement('li');
      element.appendChild(document.createTextNode('book_' + i));
      fragment.appendChild(element);
    }
    
    list.appendChild(fragment);
    
  • 减少var定义变量

    每使用一个var定义变量都会给JavaScript解析器添加一条查找路径,可以使用,将多个变量定义使用一个var定义。

    常规实践:

      var a = 1;
      var b = 2;
      var foo = 'bar';
    

    好的实践:

      var a = 1,
          b = 2,
        foo = 'bar';
    
  • 在循环外定义变量

    JavaScript作用域是函数级别的,不是代码块,所以在循环块外定义变量,避免在循环内一次又一次定义变量。

    常规实践:

      for(var i = 0; i < 10; i++) {
        var element = document.createElement('li');
      }
    

    好的实践:

      var element;
      for(var i = 0; i < 10; i++) {
        element = document.createElement('li');
      }
    
  • 使用join方法链接字符

    如果链接字符数组,使用join方法

    常规实践:

      var strs = ['a', 'b', 'c'],
          len = strs.length,
          result = '';
          
      for(var i = 0; i < len; i++) {
        result += strs[i];
      }
    

    好的实践:

      var result = strs.join('')
    
  • 使用console.time评估代码执行时间

    语法:

      console.time(label);
      console.timeEnd(label);
    

    好的实践:

      var list = [1, 2, 3, 4],
           len = list.length,
           arr = [];
      
      console.time("Iterate the list...");
      for(var i = 0; i < len; i++) {
        arr.push(list[i]);
      }
      console.timeEnd("Iterate the list...");
    

    console.time可以嵌套

      console.time("Total time ...");
      console.time("Part 1 need time...");
      //...
      console.timeEnd("Part 1 need time...");
      console.time("Part 2 need time...");
      //...
      console.timeEnd("Part 2 need time...");
      console.timeEnd("Total time ...");
    

    注: 其中两个部分加起来的时间和不一定等于总的时间,因为浏览器解析器要预解析一些指令。针对给出的时间,对代码进行优化。

  • 精确获取执行时间

    利用代码执行时间差,获取millseconds

      var before = +new Date();
      var after = +new Date();
      var elapsedTime = after - before;
      console.log(elapsedTime)
    

    好的实践:

      function PerfTest(implement, params, times) {
        this.implement = implement;
        this.params = params;
        this.times = times || 10000;
        this.average = 0;    	  
      }
      
      PerfTest.prototype = {
        run: function(){
        	var beginAt, endAt, sumTime = 0;
        	for(var i = 0, times = this.times; i < times; i++) {
        	  beginAt = +new Date();
        	  this.implement(this.params); //调用测试函数
        	  endAt = +new Date();
        	  sumTime += endAt - beginAt;
        	}
        	this.average = sumTime / this.times;
        	return console.log("Average executated " + this.times + " times and time is : " + this.average );
        }
      }
      
      var list = [1,2,3,4,5,6,7,8,9,10];
      var testFunc = function(list){
        var arr = [];
        for(var i = 0; i < list.length; i++) {
          arr.push(list[i]);
        }
      }
      
      var test = new PerfTest(testFunc, list);
      //var test = new PerfTest(testFunc, list, 100000); //指定执行次数
      test.run();
    
  • 使用比较操作符

好的实践: 使用 === 进行比较

  • JS文档化

现在主流的JavaScript使用的模块模式,而CommonJS比较主流一些,当然也可以使用AMD或者UMD规范。

如果使用CommonJS规范,可以使用jsdoc工具帮助我们生成可视化以及可读性更好的文档。

    /**
* User module
* @module User
* @desc 用户模型
*/
'use strict';

var Const = require('../common/const');
var Util = require('../common/util');
//...

module.exports = {
        /**
     * 获取用户详细信息
     *
     * @function showUser
     * @param {String} userId - 用户ID
     */
        showUser: function () {
            //...
        }
    };

gulpfile.js内容如下:

    {
  "name": "xxx",
  "version": "0.0.1",
  "description": "xxxx",
  "main": "app.js",
  "scripts": {
    "test": "jest",
    "generate-docs": "node_modules/.bin/jsdoc --configure .jsdoc.json --verbose"
  },
  "author": "dongtong",
  "license": "MIT",
  "devDependencies": {
    "browser-sync": "^2.11.0",
    "gulp": "^3.9.0",
    "gulp-autoprefixer": "^3.1.0",
    "gulp-base64": "^0.1.3",
    "gulp-concat": "^2.6.0",
    "gulp-cssnano": "^2.1.0",
    "gulp-html-minifier": "^0.1.8",
    "gulp-jscs": "^3.0.2",
    "gulp-jscs-stylish": "^1.3.0",
    "gulp-jshint": "^2.0.0",
    "gulp-minify-css": "^1.2.3",
    "gulp-notify": "^2.2.0",
    "gulp-plumber": "^1.0.1",
    "gulp-rename": "^1.2.2",
    "gulp-uglify": "^1.5.1",
    "jest-cli": "^0.8.2",
    "jsdoc": "^3.4.0",
    "jshint": "^2.8.0",
    "jshint-stylish": "^2.1.0",
    "minami": "^1.1.1",
    "vinyl-source-stream": "^1.1.0"
  },
  "dependencies": {
    "browserify": "^13.0.0",
    "glob": "^6.0.4",
    "vinyl-buffer": "^1.0.0"
  }
}

.jsdoc.json内容如下:

    {
    "tags": {
        "allowUnknownTags": true,
        "dictionaries": ["jsdoc"]
    },
    "source": {
        "include": ["src/js/modules/", "src/js/common/", "package.json"],
        "includePattern": ".js$",
        "excludePattern": "(node_modules/|docs)"
    },
    "plugins": [
        "plugins/markdown"
    ],
    "templates": {
        "cleverLinks": false,
        "monospaceLinks": true
    },
    "opts": {
        "destination": "./docs/",
        "encoding": "utf8",
        "private": true,
        "recurse": true,
        "template": "./node_modules/minami"
    }
}

最后在gulp工程下(这里使用的是Gulp前端工作自动化流程)运行:

  npm run generate-docs

会在项目工程跟目录下面生成一个docs目录,打开里面的index.html文件即可预览文档。