API(接口)是什么
API(接口)是什么?举个常见的例子,在京东上下单付款之后,商家选用顺丰发货,然后你就可以在京东上实时查看当前的物流信息。京东和顺丰作为两家独立的公司,为什么会在京东上实时看到顺丰的快递信息,这就要用到API,当查看自己的快递信息时,京东利用顺丰提供的API接口,可以实时调取信息呈现在自己的网站上。除此,你也可以在快递100上输入订单号查取到快递信息。只要有合作,或是有允许,别的公司都可以通过顺丰提供的API接口调取到快递信息。既然有多方调用,那提供一个统一的调用规范会方便很多。
我们看下百度是怎么给出API的定义的:
API(Application Programming Interface, 应用程序 编程接口)是一些预先定义的 函数 ,目的是提供 应用程序 与开发人员基于某 软件 或硬件得以访问一组 例程 的能力,而又无需访问源码,或理解内部工作 机制 的细节。
从百度定义中,我们先划下重点:函数、 提供 应用程序 与开发人员 、无需访问源码、例程 。注意这四个短语。其中,例程是某个系统对外提供的功能接口或服务的集合,本文着重说下 API的数据服务功能接口 。
首先,下面的文章我会先演示怎么通过浏览器,不需要编程、不需要访问源码,来调用免费的API接口,达到让你怎么使用最简单的接口;
然后,会演示怎么通过编程去调用接口;
最后,演示开启WEB服务,写一个简单的API接口,以体现出API的函数 (y=f(x)) 功能。
你也可以复制代码试下。通过你自己的尝试,你会了解到些后端、前段的东西,并加深你对API的理解。那么我们就开始吧,先上一个图:以API提供信息(数据)的功能,看下数据是怎么流动的。如果给API一个具体的位置的话,就位于下图中央。

为了更改的理解,我穿插一个故事角色。假如,世界太大,身处非洲大陆的你想去看看,成为了一个水手,被一位著名的航海家带去去远航(你掌握着船上唯一一台卫星电脑),途中不免航海家会让确定下城市的经纬度。这时航海家让你去确定下深圳市的经纬度。
1、你是个人肉API
这时,你能想到简单的办法就是去百度搜索一下。 除了航海家,船上的高级水手、大副,估计都有可能向你问,这个时候,其实你就是一个接口,大家都来向你获取地理位置信息,作为接口,你本身不产生信息,你只是信息的搜集者、传递者,提供人肉数据服务。

2、使用免费的经纬度API接口
在百度搜索的过程中,众多网友给出的经纬度不一致,你要反复对比,去辨真伪。时间长了,你可能会觉得大家都问位置,太烦了,我这个人肉API效率太慢,有没有更快的方式。
这时你发现了一个阿里云的一个免费的API接口: http://gc.ditu.aliyun.com/geocoding?a= 深圳 。你在浏览器地址栏输入了这个URL,果然得到了正确的经纬度,你又把URL中的深圳改为了别的城市,发现也可以查到。这时,通个这个接口,你给大伙确定位置的效率大大提升。

3、你不满足于此
你觉得这个接口太好了,刚好你会点Python,何不写点代码,注意这个时候你的角色发生了变化:你由一个人肉API转变了一个开发人员(注意这个是百度API定义中的关键词),你现在的目标是写代码,能够自动获取到这个某个地方的经纬度。
#python
import requests
r = requests.get("http://gc.ditu.aliyun.com/geocoding?a=深圳")
loc = r.json()
print loc
#打印出的结果:{u'alevel': 4, u'lon': 114.05786, u'level': 2, u'cityName': u'', u'address': u'', u'lat': 22.54309}
# 与在浏览器输入地址的结果一致
print "深圳的经度是%s,纬度是%s" % (loc["lon"], loc["lat"])
#打印出的结果是:深圳的经度是114.05786,纬度是22.54309
到这里,我们再来回顾下API的百度定义。目前,你作为一个开发人员, 具备通过写了几行代码访问“阿里云经纬度接口”的能力, 你无需访问这个API的源码、不需要理解这个API是怎么弄成的 。也就是,不管是通过浏览器,还是编程语言,目前你已经会用简单的API了。
4、在会用简单API的接口后,你进一步想知道它的内在机制是什么
建立在会用的基础之上,那么API的源码、其内在机制是怎么回事?目前你能调用经纬度的API,简单的获取到位置还不够,你可能自己想写一个API接口,不紧紧提供位置信息,同时附加上对这个城市的介绍。
利用Python的tornado模块,搭建一个web服务。我们建一个脚本空文件 web_server.py,把以下代码复制进去
# -*- coding: utf-8 -*-
import sys
reload(sys)
sys.setdefaultencoding('utf-8')
import tornado.ioloop
import tornado.web
import json
import requests
lists = {u"深圳": "是经济特区,紧邻广州,接壤香港,人口约1200万",
u"青岛": "旅游城市,濒临黄海,特产啤酒,人口约920万"
def get_loc(city):
r = requests.get("http://gc.ditu.aliyun.com/geocoding?a=%s" % city.encode('UTF-8'))
loc = r.json()
return "经度是%s,纬度是%s" % (loc["lon"], loc["lat"])
class BaseHandler(tornado.web.RequestHandler):
def get_current_user(self):
return self.get_secure_cookie("user")
class MainHandler(BaseHandler):
def get(self):
if self.request.arguments.has_key("id"):
greeting = self.get_argument('id', 'Hello')
if greeting in lists:
self.write(greeting + ": " + str(get_loc(greeting)) + "," + str(lists[greeting]))
else:
self.write("none")
settings = dict(cookie_secret="P1/V61oETzdkLmGeJJFuYh7Eo5KXQAGaYgEQnp2XdTo=", debug=True)
application = tornado.web.Application([(r"/", MainHandler), ], **settings)
if __name__ == "__main__":
application.listen(8888)
tornado.ioloop.IOLoop.current().start()
执行web_server.py,执行后就会开启一个web服务

接着,我们在浏览器输入 http:// 192.168.199.204:8888/? id= 深圳
注意,我自己的电脑局域网IP是192.168.199.204,需要替换成你自己电脑(或虚拟机)的ip

从上图看,我们实现了,和”阿里云经纬接口“的相同功能,并且获取到了除经纬度,还有对城市的简单描述。随着航行越来越多,你让船上的大伙不仅了解了地理位置信息,也了解了每个城市的经济人文情况。随着信息数据的增多,特别需要一个东西能先把数据先保存起来,用到的时候,API再直接调用。
5、数据储存,调我所需
上面我们模仿了阿里云的API,自己搭建了一个web服务,提供额一个更为详细的API地理位置信息服务。现在想一下,我们的代码里只有深圳、青岛两个城市,还有很多没提到的城市,另外,城市还有经济发展情况、饮食文化等。我们把深圳、青岛两个城市信息存入到mysql里。

存入之后,我们从mysql里直接拿数据,通过API直接展现在到浏览器中。改动 web_server.py脚本中的
# -*- coding: utf-8 -*-
import sys
reload(sys)
sys.setdefaultencoding('utf-8')
import tornado.ioloop
import tornado.web
import json
import requests
import MySQLdb.constants
def get_mysql(sql_string):
conn = MySQLdb.connect(host='localhost', port = 3306, user = 'root', passwd = '123456', db = 'test', charset = 'utf8')
cursor = conn.cursor()
cursor.execute(sql_string)
resultList = {}
for data in cursor.fetchall():
city_name = data[0]
city_introduce = data[1]
resultList[city_name] = city_introduce
return resultList
cursor.close()
conn1.close()
def get_loc(city):
r = requests.get("http://gc.ditu.aliyun.com/geocoding?a=%s" % city.encode('UTF-8'))
loc = r.json()
return "经度是%s,纬度是%s" % (loc["lon"], loc["lat"])
class BaseHandler(tornado.web.RequestHandler):
def get_current_user(self):
return self.get_secure_cookie("user")
class MainHandler(BaseHandler):
def get(self):
if self.request.arguments.has_key("id"):
greeting = self.get_argument('id', 'Hello')
lists = get_mysql("select name,introduce from test.city where name ='%s' " % greeting.encode('UTF-8')) #直接从mysql里读取城市信息,特别注意这里的SQL
if greeting in lists:
self.write(greeting + ": " + str(get_loc(greeting)) + "," + str(lists[greeting]))
else:
self.write("none")
settings = dict(cookie_secret="P1/V61oETzdkLmGeJJFuYh7Eo5KXQAGaYgEQnp2XdTo=", debug=True)