关于API的设计和需求抽象
一,先来谈抽象吧,因为抽象跟后面的API的设计是息息相关的
有句话说的好(不知道谁说的了):计算机科学中的任何问题都可以抽象出一个中间层就解决了。
抽象是指在思维中对同类事物去除其现象的、次要的方面,抽取其相同的、主要的方面,从而做到从个别中把握一般,从现象中把握本质的认识工程和思维方法。
在计算机学科中,抽象也称为模型化,源于实验科学,主要要素为数据采集方法和假设的形式说明、模型的构造与预测、实验分析。结果分析。在为可能的算法、
数据结构和系统结构等构造模型时使用此过程。抽象的结果为概念、符号和模型。在图论中体现的是抽象与理论形态,欧拉从哥尼斯堡七巧板问题入手,将其抽象为边和点的问题进行研究,成为图论研究的先驱。哈密顿回路、中国邮路等问题都是对现实问题进行的抽象,这些问题的研究和解决形成了一套比较完整的关于图的理论,包括一系列的定义、公理和定理等。
首先,人脑往往不适于长幅记忆或直接面对复杂的二进制底层,人们在面对根本无法控制的事情时,往往把它们转化为另外一件可控的事抽象正是这样一种方法,它可以隐藏低级层面的复杂性,而在另一个层面上提供新的更为强大的能力。再在这里抽象上构建更为高层的抽象,即抽象只是把问题变了个形式,抽象完成了之后,只要不是过度抽象,那么所有后来的事情都是另外一回事了,抽象是解决移殖问题最好的方法,抽象源于一个简单的事实,把事物从逻辑上分开,这样就会解偶他们之间的联系。只有把接口拉高,向高层抽象,那么就可以忽视平台逻辑,其次,从问题到解决不是一步而就的,所有需要建立中间层,先完成这诸多中间层,当中间的逻辑被解决的时候,事情自然就变得简单了(从这个意义上来说,世间一切软件都是抽象品,软件即抽象),抽象的本质在于远离问题,从靠近人的一个高层角度去解决更高级的问题。 但是抽象的优点正是它的限制性,它可能带来再大的复杂性1,一般抽象到了某个程度,为了获得计算机作为底层的冯氏能力,,就不应该再抽象下去了。开发模型不需要再变了,数据抽象到数据结构级就是顶级了,再抽象就不是开发问题了,现在的虚拟机的提出,都是基于已有的模式,直接参照硬件上的机器设计中断,模拟数据类型等,从来没有那个虚拟机,其内部结构不是图灵模型,因为如果那样的话,它上面的开发模型将不再是数据加代码的方式。从来没有人突破过这个创新。仅仅因为大部分人没有想过,或根本无法尝试。其次,抽象就是编程界对事物方方面面的范式的一个界定。但一般的,抽象是抽取对象的可用部分,我们从来都是抽取事情对于我们的可用部分,所以设计时千万不能做大而全的抽象。
抽象都是有唯度的,数据结构就是基于数据化类型的唯度抽象了开发中组织内存的套路,而高级语言机制或范式基于多样化写代码的唯度抽象了开发,设计模式是基于设计抽象了应用,而设计,就是人的事情了,处在前面数据结构,范式,应用的所有实际上软件的设计哲学是可以用来解释一切的,因为它是真正的哲学,而真正的哲学并不仅适用软件开发(软工和计算机是二个完全不同的抽象,虽然没有人提出过计算机抽象到底是什么,软工抽象到底里面有哪些抽象存在,我们仅能站在某个或某些唯度上给出一个描述性的概念而不是有限集,这也就够了,如果能站在一个大全的唯度上说明到软工的全部抽象,虽然这是不可能的,但我们还是给得出的这个结果取个名字,叫范式,范式在意义上是大全而的抽象,然而人类的范式总表现为某些唯度上的产物上面层次。下面详细介绍这个唯度的概念。
我们来问个问题,程序如何分类呢,从算法和数据结构的角度看我们可以发现,数据结构加算法等于程序。因为数据结构源于从一套相似的算法中找出操作对象的共性这个现实,而从复用来看呢,,又可以产生设计和接口就等于程序这种说法,因此这完全是不同事物的不同唯度而已。根本没有可比性。(至少二者都可以产生程序这个概念,于是,程序=机器加电也是正确的)抽象把事物的复杂度换化到另一层面,实际上也是另一唯度。这就是抽象与唯度。其实就语言本身来说,并没有汇编,C,C++和Java,Python,Ruby这几个语言之间哪个语言更强大一点的说法,大凡用其中一方能实现的功能,用一方都完全能够抽象得到,Java所关注的Web编程领域,C++完全可以提供同样的功能实现, 只有抽象完成,整个Windows系统都可以用Java来写,这就是说,在软件的抽象里,任何事情都可以以抽象叠成的方式来完成.但是,“决定用什么语言干什么事”这个前提是“它善于干什么事情”,而不是“能不能”(而这,是由应用本身和人的要求所规定的),所以显然地,在Wintel上装个Jvm,再用Java实现个Windows,这是个傻瓜行为(舍近求远而且有应用的瓶颈问题存在).比如用C进行C++能很好工作的事,这也是个傻行为(仅仅因为C没有显式的OO设计手段虽然它可以抽象得到)。
过程抽象的概念是程序设计语言的设计中最老的概念之一,所有的子程序都是过程抽象。因为它们提供了一种方式,让一个程序说明要完成的某些过程,而不是要提供如何来完成的细节,以使程序设计的过程逐步趋于简单化。例如:当一个程序需要将某种类型的数值数据对象数组排序时,它通常使用一个子程序来进行这种排序过程,在程序中需要进行排序的位置。过程抽象对于程序设计过程十分关键,这种将子程序中的算法的许多的细节抽象出来的能力,使得人们有可能来构造、阅读和理解大程序,现在被认为的大程序必须至少其有好几十万代码。
所有的子程序,包括并发子程序和异常处理程序都是过程抽象。 数据抽象必然跟随着过程抽象的发展而发展,因为每一种数据抽象中的一个不可分割的中心部分都是操作,而操作被定义成过程抽象。
数据抽象的动机之一与过程抽象的类似,它是对抗复杂性的一种武器,是使得大型以及复杂的程序比较容易管理的一种方法。
在程序设计当中,所有内置数据类型都是抽象数据类型,例如:考虑一个浮点的数据类型。
在高级语言中,封装是抽象数据类型的先驱及支持机制。一个封装可以分割编译或者是独立编译,对一组相关的计算机提供了一个抽象系统和一个逻辑组织。
抽象思维方法本身随着人类文明的进化也在不断演化。从大类看,抽象思维方法分为抽象思维的形而上学方法和抽象思维的辩证方法。抽象思维的这两种具体形式不象有些人认为的那样,是对立的和没有共同基础贩。实际上,抽象思维的辩证方法是建立在抽象思维的形而上学方法之上的,在运用分析、综合、归纳、演绎方法来形成概念并确定概念与概念之间演绎的关系、概念外延的数量属性关系这些内容上,抽象思维的辩证方法和抽象思维的形而上学方法完全一样;只是面对发展和变化着的世界,为了提高思维的形而上学方法完全一样;只是面对发展和变化着的世界,为了提高思维的精确度从而减少思维对现实的偏差,抽象思维的辩证方法才增加了对概念内涵的数量属性关系的考察。可以说,抽象思维的辩证方法是对抽象思维的形而上学方法的补充的提高。
二,API的设计
1.为了设计API对你如此重要呢?
----如果你是程序员,那么也可以说你是一个API设计者:一个好的代码,实际就是模块和模块之间有一个API
----有用的模块需要考虑能复用:一旦一个模块有用户了,那么就不能随意改变API了。(因为对于用户可见的是API,如果改变它,用户相当于使用一个全新的产品,需要花费更多的精力去学,即使有详细的文档保证);好的可复用模块之间都是能很好的互相协作的
----在你写代码的时候,思考API如何设计,能提高代码质量
2.一个好的API的特点
2.1易学
2.2即使无文档保证,也能轻易使用。(也就是我看到API的时候一目了然,这涉及到函数命名,参数安排的问题)
2.3难以被错误的使用(同上)
2.4容易读懂,保持原有代码
2.5很好的满足需求
2.6容易扩展
2.7对于用户来说适当
总结:感觉以上很符合Unix设计哲学,KISS原则无处不在啊
3.API设计过程
了解明确需求,思考需求(持有怀疑论的态度反思)
-----更好的解决方案是存在的。(没有最好,只有更好)
-----能使用更简单更有效的方法来构建一件“事物”,让此“事物”变得更普适。(这也说明了软件接口越往上层越普适,不会依赖于平台相关)
3.1先写以小段API的规范说明书开始(保证说明书短小容易修改,API的输入尽量满足多数人所需要的)
3.2提前设计API,不管其实现细节。然后再API的实现过程中,慢慢再完善API的设计,对其做单元测试
3.3插件式的API要保证能够以多种方式来实现。
3.4保持API最现实的异常处理情况,大多数API都要设计成强约束的,因为你不能够迎合每一个用户,在异常处理方面,对于每个用户都必须严格校验。考虑API出错(不理想),经过现实时间的工业检验,这些错误都会慢慢暴露出来的,然后你再去修复。
4.设计API的基本原则
4.1 API的功能必须很简单就能解释清楚,不能复杂。如果对一个API很难命名,那么就说明这是个糟糕的设计,功能抽象做得不好。
4.2 API要尽可能的小,但是不能太小。API需要满足它的需求。当遇到错误,需要及时处理跳出。API的概念比API实现体积重要。你能添加API,但是不能删除它。(保持兼容)。API的参数不能过多,否则依赖过度,说明抽象不好
4.3 API的实现完全不影响API本身
4.4减小API之间的依赖性
4.5从API的名字就可以看出功能,阅读代码像读散文一样。(就是容易理解)
4.6一个好的文档也是重要的。
关于API的设计和需求抽象的更多相关文章
- 优秀的API接口设计原则及方法(转)
一旦API发生变化,就可能对相关的调用者带来巨大的代价,用户需要排查所有调用的代码,需要调整所有与之相关的部分,这些工作对他们来说都是额外的.如果辛辛苦苦完成这些以后,还发现了相关的bug,那对用户的 ...
- 好RESTful API的设计原则
说在前面,这篇文章是无意中发现的,因为感觉写的很好,所以翻译了一下.由于英文水平有限,难免有出错的地方,请看官理解一下.翻译和校正文章花了我大约2周的业余时间,如有人愿意转载请注明出处,谢谢^_^ P ...
- 从涂鸦到发布——理解API的设计过程(转)
英文原文:From Doodles to Delivery: An API Design Process 要想设计出可以正常运行的Web API,对基于web的应用的基本理解是一个良好的基础.但如果你 ...
- RESTful API的设计原则
好RESTful API的设计原则 说在前面,这篇文章是无意中发现的,因为感觉写的很好,所以翻译了一下.由于英文水平有限,难免有出错的地方,请看官理解一下.翻译和校正文章花了我大约2周的业余时间, ...
- 好的RESTful API的设计原则
转载自一位大佬 英文原版 Principles of good RESTful API Design Good API design is hard! An API represents a cont ...
- RESTful API URI 设计: 查询(Query)和标识(Identify)
相关文章:RESTful API URI 设计的一些总结. 问题场景:删除一个资源(Resources),URI 该如何设计? 应用示例:删除名称为 iPhone 6 的产品. 是不是感觉很简单呢?根 ...
- atitit.基于http json api 接口设计 最佳实践 总结o7
atitit.基于http json api 接口设计 最佳实践 总结o7 1. 需求:::服务器and android 端接口通讯 2 2. 接口开发的要点 2 2.1. 普通参数 meth,p ...
- 关于API的设计与实现
http://blog.csdn.net/horkychen/article/details/46612899 API的设计是软件开发中一个独特的领域.最主要的特殊点在于API是供开发者使用的界面,即 ...
- RESTful Web Services中API的设计原则(转)
当下前后端分离的设计已经是web app开发的标配,但是如何设计一个强壮,扩展性好,又规范的API呢 参考以下link,可以得到需要有益的启示.同时个人推荐一本书<web API的设计和开发&g ...
随机推荐
- Delphi 调试Dll报错 通过GetLastError显示错误信息。
LibHandle := LoadLibrary('c:\windows\system32\SpcClass.dll'); ShowMessage(SysErrorMessage(GetLastEr ...
- word-wrap: break-word 和 word-break: break-all 到底有啥区别?
做项目改bug的时候,遇到过好多次,要么是文本超出文本区域,或者单词太长(一般是url链接中的一些鬼),把装它的标签强制撑大,导致一些响应式问题.除此之外,还有很多问题,每次都是恍然醒悟,然后又在网上 ...
- js 等待刷新技术
- Android中设置文本颜色的三种方法
最近刚开始学web,发现好的颜色搭配可以让自己的网页更加美观, 中午不想做事,就无聊滴花了两个小时测试了所有颜色的编码,总结如下 新手没有什么吊炸天的技术,仅仅是一份辅助的文档,有兴趣的朋友可以收藏下 ...
- android 滚动视图(ScrollView)
为了可以让内嵌布局管理器之中加入多个显示的组件,而且又保证程序不这么冗余,所以可以通过 Activity程序进行控制,向内嵌布局管理器中添加多个组件. ScrollView提供一个显示的容器,可以包含 ...
- UVA 10651 Pebble Solitaire(bfs + 哈希判重(记忆化搜索?))
Problem A Pebble Solitaire Input: standard input Output: standard output Time Limit: 1 second Pebble ...
- ViutualBox虚拟机里添加磁盘
1.首先在VirtualBox界面给虚拟机添加一块磁盘 2. 启动系统,查看当前磁盘空间 Last login: Tue Mar 15 22:24:47 2016 from 192.168.1.100 ...
- 使用F#开发量化模型都缺什么?
量化模型多数是基于统计的,因此,统计运算库应该是必备的.在Matlab.R中包含了大量的统计和概率运算,可以说拿来就用,非常方便,相比之下,F#的资源就很少了,这里给大家提供几个链接,可以解决一部分问 ...
- .Net HttpClient 模拟登录微信公众平台发送消息
1.模拟登录 public WeiXinRetInfo ExecLogin(string name, string pass) { CookieContainer cc = new CookieCon ...
- sealed 密封类,不能被其他类继承,但可以继承其他类
public sealed class Person:继承类名 { }