1、目的

  由于有多个程序和脚本需要对mysql进行读写数据库,每次在脚本中进行数据库的连接、用cursor进行操作过于麻烦,因此希望可以有一个脚本开放接口,只需要传入sql语句,就可以返回结果回来。因此有需要一个可以支持并发量较大的脚本来进行数据库操作。以上就要求我的接口具有异步非阻塞、在结果返回前保持长连接、并发大。因此单纯的多线程和协程没办法满足要求,就需要用到tornado框架。

2、程序

1)客户端代码(通过requests 调用接口)

import reuqests
POST = requests.post # post请求方式
def db_query(sql, method='query'):
db_api = 'http://127.0.0.1:8000/db/api'
db_base = "dbbase"
db_ret = POST(db_api, data=json_encode({'method': method, 'sql': sql, 'dbbase': db_base, 'pwd': 'password123'}))
if db_ret.status_code == 200:
if json_decode(db_ret.text)['status'] == 'True':
db_ret_data = json_decode(db_ret.text)['data']
return {'status': 'ok', 'data': db_ret_data, 'err': ''}
else:
return {'status': 'False', 'data': [], 'err': json_decode(db_ret.text)['data']}
else:
return {'status': 'False', 'data': [], 'err': 'connect error'}

 2)服务端代码(基于tornado框架)

# coding=utf8
import MySQLdb
import MySQLdb.cursors
import tornado
import tornado.ioloop
import tornado.web
import tornado.gen
from tornado.concurrent import run_on_executor
from concurrent.futures import ThreadPoolExecutor class DB_CONFIG:
config = {
'dbbase': ('authentic', 'password'), # 数据库user、password
} class DB:
def __init__(self, dbbase):
user, password = DB_CONFIG.config.get(dbbase, (None, None))
if user == None or password == None:
raise Exception('KeyError', dbbase)
db_host = '192.168.xx.xx'
db_port = 1234 self.db = MySQLdb.connect(db_host, user, password, dbbase, port=db_port, cursorclass=MySQLdb.cursors.DictCursor) self.cursor = self.db.cursor() def query(self, sql):
self.cursor.execute(sql)
return self.cursor.fetchall() def commit(self, sql):
try:
self.cursor.execute(sql)
self.db.commit()
return {'status': True, 'data': ''}
except Exception as e:
self.db.rollback()
return {'status': False, 'data': e} def close(self):
self.db.close() class ServiceHandler(tornado.web.RequestHandler):
executor = ThreadPoolExecutor(900) # 必须定义一个executor的属性,然后run_on_executor装饰器才会游泳。 @run_on_executor # 线程内运行;query函数被run_on_executor包裹(语法糖),将该函数的执行传递给线程池executor的线程执行,优化了处理耗时性任务,以致达到不阻塞主线程的效果。
def query(self, dbname, method, sql):
db = DB(dbname)
ret = ''
if method == 'query':
ret = db.query(sql)
elif method == 'commit':
ret = db.commit(sql)
db.close()
return ret @tornado.web.asynchronous # 保持长连接,直到处理后返回
@tornado.gen.coroutine # 异步、协程处理;增加并发量
def post(self):
data = tornado.escape.json_decode(self.request.body) # 获取参数,json.loads()解码
r = {'status': '', 'data': ''}
if not data.get('pwd', None):
r['status'], r['data'] = ('False', 'not password')
elif not data.get('dbbase', None):
r['status'], r['data'] = ('False', 'not DB select')
else:
if data['pwd'] != 'password123)': # 接口的密码认证
r = {'status': 'False', 'data': 'password error'}
elif data['method'] == 'query': d = yield self.query(data['dbbase'], 'query', data['sql'])
r = {'status': 'True', 'data': d}
elif data['method'] == 'commit': db_r = yield self.query(data['dbbase'], 'commit', data['sql'])
if db_r['status']:
r = {'status': 'True', 'data': 'commit sucessful'}
else:
r = {'status': 'False', 'data': db_r['data']}
else:
r = {'status': 'False', 'data': 'method Invaild'}
self.write(tornado.escape.json_encode(r)) # 写入返回信息写入response
self.finish() # 结束服务 def get(self):
return self.post() if __name__ == "__main__":
application = tornado.web.Application([
(r"/db/api", ServiceHandler), # 路由映射
])
application.listen(8000) # 监听端口
tornado.ioloop.IOLoop.instance().start() # 启动服务

3、请求举例(post请求)

  1) sql语句

 sql = 'select vid from video where status=1 group by vid ORDER BY num ASC limit 100'

   2)客户端调用db_query函数

db_ret = db_query(sql, 'query') # 其中query是数据库操作的方法 query为查询/commit 为insert/update/delete等数据库修改操作时使用

  通过调用db_query传入参数sql语句和操作方式,返回结果或错误内容

基于tornado---异步并发接口的更多相关文章

  1. 异步并发利器:实际项目中使用CompletionService提升系统性能的一次实践

    场景 随着互联网应用的深入,很多传统行业也都需要接入到互联网.我们公司也是这样,保险核心需要和很多保险中介对接,比如阿里.京东等等.这些公司对于接口服务的性能有些比较高的要求,传统的核心无法满足要求, ...

  2. 基于tornado的爬虫并发问题

    tornado中的coroutine是python中真正意义上的协程,与python3中的asyncio几乎是完全一样的,而且两者之间的future是可以相互转换的,tornado中有与asyncio ...

  3. 深入理解yield(三):yield与基于Tornado的异步回调

    转自:http://beginman.cn/python/2015/04/06/yield-via-Tornado/ 作者:BeginMan 版权声明:本文版权归作者所有,欢迎转载,但未经作者同意必须 ...

  4. Python 开源异步并发框架的未来

    http://segmentfault.com/a/1190000000471602 开源 Python 是开源的,介绍的这几个框架 Twisted.Tornado.Gevent 和 tulip 也都 ...

  5. 5.(基础)tornado异步

    终于到了传说中的异步了,感觉异步这个名字听起来就很酷酷的,以前还不是多擅长Python时,就跑去看twisted的源码,结果给我幼小的心灵留下了创伤.反正包括我在内,都知道异步编程很强大,但是却很少在 ...

  6. Python开源异步并发框架

    Python开源异步并发框架的未来 2014年3月30日,由全球最大的中文IT社区CSDN主办的“开源技术大会·” (Open Source Technology Conference ,简称OSTC ...

  7. tornado 异步

    引言 注:正文中引用的 Tornado 代码除特别说明外,都默认引用自 Tornado 4.0.1. tornado.gen 模块是一个基于 python generator 实现的异步编程接口.通过 ...

  8. Tornado异步(2)

    Tornado异步 因为epoll主要是用来解决网络IO的并发问题,所以Tornado的异步编程也主要体现在网络IO的异步上,即异步Web请求. 1. tornado.httpclient.Async ...

  9. 基于tornado实现web camera

    基于tornado实现web camera 近期在学习python.找了一个框架学习,我选择的是tornado.由于其不仅仅是一个web开发框架,其还是一个server,异步事件库,一举多得. 我一直 ...

随机推荐

  1. PMP 1~3章错题总结

    工作到了一定的年限,都或多或少想了解管理的知识,PMP是国际认证的一项考试,招聘要求上也有提及. 不需要报名培训班,万能的某宝即可解决报名.PDU.学习资料的问题,但3900的考试费还是免不了的,为了 ...

  2. /home/jdk1.8.0_211/jre/bin/java: Permission denied

    在Linux上安装JDK后启动tomcat 日志中提示  /home/jdk1.8.0_211/jre/bin/java: Permission denied 原因是:运行的命令在这个路径下没有权限 ...

  3. light4j一个轻量级的低延时、高吞吐量、内存占用量小的API平台

    1.背景(abstract) 笔者算是一个极客类型的程序员了.喜欢探索一些程序内在的原理.稳定性.自动化运维.健壮性,很多时间也会 去对程序的内存使用率.cpu使用率锱铢必较.尽量克扣掉不必要的cpu ...

  4. Python3使用random生成随机数

    本文介绍使用Python3中的random库生成随机数.随机小数.随机序列.随机字符串以及扑克洗牌等方法. 一.生成随机浮点数或小数 1.#生成0-1之间的浮点数 import random rnd ...

  5. VM无法正常使用桥接模式获取IP上网

    问题: 有时候会遇到VM使用桥接模式时无法正常获取IP的情况 原因: 初步怀疑是因为你的电脑是双网卡 解决方法: 这时候,就需要修改VM的虚拟网络编辑器的配置 解决步骤: 编辑->虚拟网络编辑器 ...

  6. React全家桶+Material-ui构建的后台管理系统

    一.简介 一个使用React全家桶(react-router-dom,redux,redux-actions,redux-saga,reselect)+Material-ui构建的后来管理中心. 二. ...

  7. DockerFile语法【h】

    DockerFile在我理解就是可以将所需要构建镜像的功能.组件都天前配置好,然后直接生成一个Image,而不是先生成镜像,再通过修改容器的方法来生成最终需要的镜像.   镜像的定值实际上就是定值每一 ...

  8. web基础---->script标签的特殊使用

    今天要讲的就是怎样使用<script>去请求一个servlet,加载一些js资源以及额外的逻辑处理: 目录: JS的引入的几种方式 在script的标签中引入Servlet 动态引入JS的 ...

  9. kafka web端管理工具 kafka-manager【转发】

    1,下载与安装 $ git clone https://github.com/yahoo/kafka-manager.git $ cd kafka-manager $ ./sbt clean dist ...

  10. Python之logging.basicConfig函数各参数

    filename: 指定日志文件名 filemode: 和file函数意义相同,指定日志文件的打开模式,'w'或'a' format: 指定输出的格式和内容,format可以输出很多有用信息,如上例所 ...