JavaScript的API设计原则

开发 前端
本篇博文来自一次公司内部的前端分享,从多个方面讨论了在设计接口时的原则,总共包含了七个大块。系卤煮自己总结的一些经验教训。同时也参考了一些文章,地址会在后面贴出来。很难做到详尽充实,如果有好的建议或者不对的地方,还望不吝赐教斧正。

前言

本篇博文来自一次公司内部的前端分享,从多个方面讨论了在设计接口时的原则,总共包含了七个大块。系卤煮自己总结的一些经验教训。同时也参考了一些文章,地址会在后面贴出来。很难做到详尽充实,如果有好的建议或者不对的地方,还望不吝赐教斧正。

一、接口的流畅性

好的接口是流畅易懂的,他主要体现如下几个方面:

1.简单

操作某个元素的css属性,下面是原生的方法:

  1. document.querySelectorAll('#id').style.color = 'red'

封装之后

  1. function a(selector, color) { 
  2.  
  3. document.querySelectorAll(selector)[0].style.color = color 
  4.  
  5. a('#a', 'red');

从几十个字母长长的一行到简简单单的一个函数调用,体现了api简单易用

2.可阅读性

a(‘#a’, ‘red’)是个好函数,帮助我们简单实用地改变某个元素,但问题来了,如果***次使用改函数的人来说会比较困惑,a函数是啥函数,没有人告诉他。开发接口有必要知道一点,人都是懒惰的,从颜色赋值这个函数来说,虽然少写了代码,但是增加了记忆成本。每次做这件事情的时候都需要有映射关系。 a—->color. 如果是简单的几个无所谓,但是通常一套框架都有几十甚至上百的api,映射成本增加会使得程序员哥哥崩溃。 我们需要的就是使得接口有意义,下面我们改写一下a函数: 

  1. function letSomeElementChangeColor(selector, color) { 
  2.  
  3.   document.querySelectorAll(selector, color); 
  4.  
  5.  

letSomeElementChangeColor相对于a来说被赋予了语言意义,任何人都会知道它的意义

3.减少记忆成本

我们刚刚的函数也是这样的它太长了letSomeElementChangeColor虽然减少了映射成本,但是增加了记忆成本。要知道,包括学霸在内,任何人都不喜欢被单词。原生获取dom的api也同样有这个问题 document.getElementsByClassName; document.getElementsByName; document.querySelectorAll;这些api给人的感觉就是单词太长了,虽然他给出的意义是很清晰,然而这种做法是建立在牺牲简易性的基础上进行的。于是我们又再次改写这个之前函数

  1. function setColor(selector, color) { 
  2.  
  3.   xxxxxxxxxxxx 
  4.  
  5.  

在意义不做大的变化前提下,缩减函数名称。使得它易读易记易用;

4.可延伸

所谓延伸就是指函数的使用像流水一样按照书写的顺序执行形成执行链条:

  1. document.getElementById('id').style.color = 'red'
  2.  
  3. document.getElementById('id').style.fontSize = '12px'
  4.  
  5. document.getElementById('id').style.backgourdColor = 'pink' 

用我们之前的之前的方法是再次封装两个函数 setFontSize, setbackgroundColor; 然后执行它们 setColor(‘id’, ‘red’);setFontSiez(‘id’, ’12px’); setbackgroundColor(‘id’, ‘pink’); 显然,这样的做法没有懒出境界来;id元素每次都需要重新获取,影响性能,失败;每次都需要添加新的方法 失败 每次还要调用这些方法,还是失败。下面我们将其改写为可以延伸的函数 首先将获取id方法封装成对象,然后再对象的每个方法中返回这个对象:

  1. function getElement(selector) { 
  2.  
  3.   this.style = document.querySelecotrAll(selector).style; 
  4.  
  5.  
  6.   
  7.  
  8. getElement.prototype.color = function(color) { 
  9.  
  10.   this.style.color = color; 
  11.  
  12.   return this; 
  13.  
  14.  
  15. getElement.prototype.background = function(bg) { 
  16.  
  17.   this.style.backgroundColor = color; 
  18.  
  19.   return this; 
  20.  
  21.  
  22. getElement.prototype.fontSize = function(size) { 
  23.  
  24.   this.style.fontSize = size
  25.  
  26.   return this; 
  27.  
  28.  
  29.   
  30.  
  31. //调用 
  32.  
  33. var el = new getElement('#id'
  34.  
  35. el.color('red').background('pink').fontSize('12px');  

简单、流畅、易读后面我们会在参数里面讲到如何继续优化。所以,大家都比较喜欢用jquery的api,虽然一个$符号并不代表任何现实意义,但简单的符号有利于我们的使用。它体现了以上的多种原则,简单,易读,易记,链式写法,多参处理。

nightmare:

  1. document.getElementById('id').style.color = 'red'
  2.  
  3. document.getElementById('id').style.fontSize = '12px'
  4.  
  5. document.getElementById('id').style.backgourdColor = 'pink' 

dream:

  1. $('id').css({color:'red', fontSize:'12px', backgroundColor:'pink'}) 

二、一致性

1.接口的一致性

相关的接口保持一致的风格,一整套 API 如果传递一种熟悉和舒适的感觉,会大大减轻开发者对新工具的适应性。 命名这点事:既要短,又要自描述,最重要的是保持一致性 “在计算机科学界只有两件头疼的事:缓存失效和命名问题” — Phil Karlton 选择一个你喜欢的措辞,然后持续使用。选择一种风格,然后保持这种风格。

Nightware:

  1. setColor, 
  2.  
  3. letBackGround 
  4.  
  5. changefontSize 
  6.  
  7. makedisplay  

dream:

  1. setColor; 
  2.  
  3. setBackground; 
  4.  
  5. setFontSize 
  6.  
  7. set.........  

尽量地保持代码风格和命名风格,使人读你的代码像是阅读同一个人写的文章一样。

三、参数的处理

1.参数的类型

判断参数的类型为你的程序提供稳定的保障

  1. //我们规定,color接受字符串类型 
  2.  
  3. function setColor(color) { 
  4.  
  5.   if(typeof color !== 'string'return
  6.  
  7. dosomething 
  8.  
  9.  

2.使用json方式传参

使用json的方式传值很多好处,它可以给参数命名,可以忽略参数的具体位置,可以给参数默认值等等 比如下面这种糟糕的情况:

  1. function fn(param1, param2...............paramN) 

你必须对应地把每一个参数按照顺序传入,否则你的方法就会偏离你预期去执行,正确的方法是下面的做法。

  1. function fn(json) { 
  2.  
  3. //为必须的参数设置默认值 
  4.  
  5.    var default = extend({ 
  6.  
  7. param: 'default'
  8.  
  9. param1: 'default' 
  10.  
  11. ...... 
  12.  
  13.    },json) 
  14.  
  15.  

这段函数代码,即便你不传任何参数进来,他也会预期运行。因为在声明的时候,你会根据具体的业务决定参数的缺省值。

四、可扩展性

软件设计最重要的原则之一:永远不修改接口,指扩展它!可扩展性同时会要求接口的职责单一,多职责的接口很难扩展。 举个栗子:

  1. //需要同时改变某个元素的字体和背景 
  2.  
  3. // Nightmare: 
  4.  
  5. function set(selector, color) { 
  6.  
  7.   document.querySelectroAll(selector).style.color = color; 
  8.  
  9.   document.querySelectroAll(selector).style.backgroundColor = color; 
  10.  
  11.  
  12.   
  13.  
  14. //无法扩展改函数,如果需要再次改变字体的大小的话,只能修改此函数,在函数后面填加改变字体大小的代码 
  15.  
  16.   
  17.  
  18. //Dream 
  19.  
  20. function set(selector, color) { 
  21.  
  22.   var el = document.querySelectroAll(selector); 
  23.  
  24.   el.style.color = color; 
  25.  
  26.   el.style.backgroundColor = color; 
  27.  
  28.   return el; 
  29.  
  30.  
  31.   
  32.  
  33. //需要设置字体、背景颜色和大小 
  34.  
  35. function setAgain (selector, color, px) { 
  36.  
  37.   var el = set(selector, color) 
  38.  
  39.   el.style.fontSize = px; 
  40.  
  41.   return el; 
  42.  
  43.  

以上只是简单的添加颜色,业务复杂而代码又不是你写的时候,你就必须去阅读之前的代码再修改它,显然是不符合开放-封闭原则的。修改后的function是返回了元素对象,使得下次需要改变时再次得到返回值做处理。

2.this的运用

可扩展性还包括对this的以及call和apply方法的灵活运用:

  1. function sayBonjour() { 
  2.  
  3.   alert(this.a) 
  4.  
  5.  
  6.   
  7.  
  8. obj.a = 1; 
  9.  
  10. obj.say = sayBonjour; 
  11.  
  12. obj.say();//1 
  13.  
  14. //or 
  15.  
  16. sayBonjour.call||apply(obj);//1  

五、对错误的处理

1.预见错误

可以用 类型检测 typeof 或者try…catch。 typeof 会强制检测对象不抛出错误,对于未定义的变量尤其有用。

2.抛出错误

大多数开发者不希望出错了还需要自己去找带对应得代码,***方式是直接在console中输出,告诉用户发生了什么事情。我们可以用到浏览器的输出api:console.log/warn/error。你还可以为自己的程序留些后路: try…catch。

  1. function error (a) { 
  2.  
  3.   if(typeof a !== 'string') { 
  4.  
  5.     console.error('param a must be type of string'
  6.  
  7.   } 
  8.  
  9.  
  10.   
  11.  
  12. function error() { 
  13.  
  14.   try { 
  15.  
  16.     // some code excucete here maybe throw wrong 
  17.  
  18.   }catch(ex) { 
  19.  
  20.     console.wran(ex); 
  21.  
  22.   } 
  23.  
  24.  

六、可预见性

可预见性味程序接口提供健壮性,为保证你的代码顺利执行,必须为它考虑到非正常预期的情况。我们看下不可以预见的代码和可预见的代码的区别用之前的setColor

  1. //nighware 
  2.  
  3. function set(selector, color) { 
  4.  
  5.   document.getElementById(selector).style.color = color; 
  6.  
  7.  
  8. //dream 
  9.  
  10. zepto.init = function(selector, context) { 
  11.  
  12.   var dom 
  13.  
  14.   // If nothing given, return an empty Zepto collection 
  15.  
  16.   if (!selector) return zepto.Z() 
  17.  
  18.   // Optimize for string selectors 
  19.  
  20.   else if (typeof selector == 'string') { 
  21.  
  22.     selector = selector.trim() 
  23.  
  24.     // If it's a html fragment, create nodes from it 
  25.  
  26.     // Note: In both Chrome 21 and Firefox 15, DOM error 12 
  27.  
  28.     // is thrown if the fragment doesn't begin with < 
  29.  
  30.     if (selector[0] == '<' && fragmentRE.test(selector)) 
  31.  
  32.       dom = zepto.fragment(selector, RegExp.$1, context), selector = null 
  33.  
  34.     // If there's a context, create a collection on that context firstand select 
  35.  
  36.     // nodes from there 
  37.  
  38.     else if (context !== undefined) return $(context).find(selector) 
  39.  
  40.     // If it's a CSS selector, use it to select nodes. 
  41.  
  42.     else dom = zepto.qsa(document, selector) 
  43.  
  44.   } 
  45.  
  46.   // If a function is given, call it when the DOM is ready 
  47.  
  48.   else if (isFunction(selector)) return $(document).ready(selector) 
  49.  
  50.   // If a Zepto collection is given, just return it 
  51.  
  52.   else if (zepto.isZ(selector)) return selector 
  53.  
  54.   else { 
  55.  
  56.     // normalize array if an array of nodes is given 
  57.  
  58.     if (isArray(selector)) dom = compact(selector) 
  59.  
  60.     // Wrap DOM nodes. 
  61.  
  62.     else if (isObject(selector)) 
  63.  
  64.       dom = [selector], selector = null 
  65.  
  66.     // If it's a html fragment, create nodes from it 
  67.  
  68.     else if (fragmentRE.test(selector)) 
  69.  
  70.       dom = zepto.fragment(selector.trim(), RegExp.$1, context), selector = null 
  71.  
  72.     // If there's a context, create a collection on that context firstand select 
  73.  
  74.     // nodes from there 
  75.  
  76.     else if (context !== undefined) return $(context).find(selector) 
  77.  
  78.     // And last but no least, if it's a CSS selector, use it to select nodes. 
  79.  
  80.     else dom = zepto.qsa(document, selector) 
  81.  
  82.   } 
  83.  
  84.   // create a new Zepto collection from the nodes found 
  85.  
  86.   return zepto.Z(dom, selector) 
  87.  
  88.  

以上是zepto的源码,可以看见,作者在预见传入的参数时做了很多的处理。其实可预见性是为程序提供了若干的入口,无非是一些逻辑判断而已。zepto在这里使用了很多的是非判断,同时导致了代码的冗长,不适合阅读。总之,可预见性真正需要你做的事多写一些对位置实物的参数。把外部的检测改为内部检测。是的使用的人用起来舒心放心开心。呐!做人嘛最重要的就是海森啦。

七、注释和文档的可读性

一个***的接口是不需要文档我们也会使用它,但是往往接口量一多和业务增加,接口使用起来也会有些费劲。所以接口文档和注释是需要认真书写的。注释遵循简单扼要地原则,给多年后的自己也给后来者看:

  1. //注释接口,为了演示PPT用 
  2.  
  3. function commentary() { 
  4.  
  5.   //如果你定义一个没有字面意义的变量时,***为它写上注释:a:没用的变量,可以删除 
  6.  
  7.   var a; 
  8.  
  9.   
  10.  
  11.   //在关键和有歧义的地方写上注释,犹如画龙点睛:路由到hash界面后将所有的数据清空结束函数 
  12.  
  13.   return go.Navigate('hash'function(){ 
  14.  
  15.     data.clear(); 
  16.  
  17.   }); 
  18.  
  19.  

***

推荐markdown语法书写API文档,github御用文档编写语法。简单、快速,代码高亮、话不多说上图 

 

 

 

卤煮在此也推荐几个在线编辑的网站。诸君可自行前往练习使用。

https://www.zybuluo.com/mdeditor

http://mahua.jser.me/

参考博文

  • 前端头条-javascript的api设计原则(http://top.css88.com/archives/814)
责任编辑:庞桂玉 来源: 前端大全
相关推荐

2016-03-29 09:59:11

JavaScriptAPI设计

2022-02-10 23:38:23

API架构设计

2014-07-21 09:22:40

GoAPI

2023-09-21 11:20:46

2024-08-26 15:35:40

2015-09-23 17:12:18

API设计原则

2015-09-24 08:52:53

API设计原则

2024-09-19 08:46:46

SPIAPI接口

2010-10-11 11:25:26

MySQL主键

2024-03-13 15:21:24

APIJava原则

2020-01-08 14:45:38

Kubernetes存储架构

2010-10-19 17:21:35

SQL SERVER主

2012-05-08 10:14:45

设计原则

2013-04-17 10:46:54

面向对象

2020-06-18 09:16:20

JavaScript缓存API

2020-07-29 09:13:28

JavaScript开发技术

2010-01-28 10:01:28

C++的设计原则

2014-09-10 10:35:11

Material De设计原则

2013-06-09 11:04:07

设计扁平化设计平面化设计

2017-10-20 14:35:38

数据库设计原则
点赞
收藏

51CTO技术栈公众号