Python SQLite3数据库操作类分享
接触Python时间也不是很长的,最近有个项目需要分析数据,于是选用Python为编程语言,除了语言特性外主要还是看重Python对于SQLite3数据库良好的支持能力了,因为需要灵活处理大量的中间数据。
刚开始一些模块我还乐此不疲的写SQL语句,后来渐渐厌倦了,回想到以前捣鼓C#的时候利用反射初步构建了个SQL查询构造器,直到发现linq,于是放弃了这个计划,当然微软后来又推出了EntityFramework,这些都是后话了,而且现在我对微软的东西兴趣不是很大的,好了,扯多了,下面继续正文。
对了,再扯一句,优秀的博客程序Drupal也使用了类似的查询构造器进行数据库查询,避免直接写SQL语句,另外这样做的一点点好处就是,可以一定程度的屏蔽平台相关性,对于数据库迁移还是有帮助的。
不过我今天介绍的数据库辅助类查询构造器是个很简单的东东,甚至只限于SQLite数据库,如果有童鞋感兴趣可以完善下,我目前只要操作SQLite顺手就可以了,对于比较大的数据库应用就直接上ORM吧。
先看代码:
importsqlite3
#***************************************************
#*
#*Description:Python操作SQLite3数据库辅助类(查询构造器)
#*Author:wangye
#*
#***************************************************
def_wrap_value(value):
   returnrepr(value)
def_wrap_values(values):
   returnlist(map(_wrap_value,values))
def_wrap_fields(fields):
   forkey,valueinfields.items():
       fields[key]=_wrap_value(value)
   returnfields
def_concat_keys(keys):
   return"["+"],[".join(keys)+"]"
def_concat_values(values):
   return",".join(values)
def_concat_fields(fields,operator=(None,",")):
   ifoperator:
       unit_operator,group_operator=operator
   #fields=_wrap_fields(fields)
   compiled=[]
   forkey,valueinfields.items():
       compiled.append("["+key+"]")
       ifunit_operator:
           compiled.append(unit_operator)
           compiled.append(value)
       compiled.append(group_operator)
   compiled.pop()#poplastgroup_operator
   return"".join(compiled)
classDataCondition(object):
   """
       本类用于操作SQL构造器辅助类的条件语句部分
       例如:
       DataCondition(("=","AND"),id=26)
       DataCondition(("=","AND"),True,id=26)
   """
   def__init__(self,operator=("=","AND"),ingroup=True,**kwargs):
       """
           构造方法
           参数:
               operator操作符,分为(表达式操作符,条件运算符)
               ingroup 是否分组,如果分组,将以括号包含
               kwargs  键值元组,包含数据库表的列名以及值
                        注意这里的等于号不等于实际生成SQL语句符号
                        实际符号是由operator[0]控制的
           例如:
           DataCondition(("=","AND"),id=26)
           (id=26)
           DataCondition((">","OR"),id=26,age=35)
           (id>26ORage>35)
           DataCondition(("LIKE","OR"),False,name="John",company="Google")
           nameLIKE'John'ORcompanyLIKE"Google"
       """
       self.ingroup=ingroup
       self.fields=kwargs
       self.operator=operator
   def__unicode__(self):
       self.fields=_wrap_fields(self.fields)
       result=_concat_fields(self.fields,self.operator)
       ifself.ingroup:
           return"("+result+")"
       returnresult
   def__str__(self):
       returnself.__unicode__()
   deftoString(self):
       returnself.__unicode__()
classDataHelper(object):
   """
       SQLite3数据查询辅助类
   """
   def__init__(self,filename):
       """
           构造方法
           参数:filename为SQLite3数据库文件名
       """
       self.file_name=filename
   defopen(self):
       """
           打开数据库并设置游标
       """
       self.connection=sqlite3.connect(self.file_name)
       self.cursor=self.connection.cursor()
       returnself
   defclose(self):
       """
           关闭数据库,注意若不显式调用此方法,
           在类被回收时也会尝试调用
       """
       ifhasattr(self,"connection")andself.connection:
           self.connection.close()
   def__del__(self):
       """
           析构方法,做一些清理工作
       """
       self.close()
   defcommit(self):
       """
           提交事务
           SELECT语句不需要此操作,默认的execute方法的
           commit_at_once设为True会隐式调用此方法,
           否则就需要显示调用本方法。
       """
       self.connection.commit()
   defexecute(self,sql=None,commit_at_once=True):
       """
           执行SQL语句
           参数:
               sql 要执行的SQL语句,若为None,则调用构造器生成的SQL语句。
               commit_at_once是否立即提交事务,如果不立即提交,
               对于非查询操作,则需要调用commit显式提交。
       """
       ifnotsql:
           sql=self.sql
       self.cursor.execute(sql)
       ifcommit_at_once:
           self.commit()
   deffetchone(self,sql=None):
       """
           取一条记录
       """
       self.execute(sql,False)
       returnself.cursor.fetchone()
   deffetchall(self,sql=None):
       """
           取所有记录
       """
       self.execute(sql,False)
       returnself.cursor.fetchall()
   def__concat_keys(self,keys):
       return_concat_keys(keys)
   def__concat_values(self,values):
       return_concat_values(values)
   deftable(self,*args):
       """
           设置查询的表,多个表名用逗号分隔
       """
       self.tables=args
       self.tables_snippet=self.__concat_keys(self.tables)
       returnself
   def__wrap_value(self,value):
       return_wrap_value(value)
   def__wrap_values(self,values):
       return_wrap_values(values)
   def__wrap_fields(self,fields):
       return_wrap_fields(fields)
   def__where(self):
       #self.condition_snippet
       ifhasattr(self,"condition_snippet"):
           self.where_snippet="WHERE"+self.condition_snippet
   def__select(self):
       template="SELECT%(keys)sFROM%(tables)s"
       body_snippet_fields={
           "tables":self.tables_snippet,
           "keys":self.__concat_keys(self.body_keys),
       }
       self.sql=template%body_snippet_fields
   def__insert(self):
       template="INSERTINTO%(tables)s(%(keys)s)VALUES(%(values)s)"
       body_snippet_fields={
           "tables":self.tables_snippet,
           "keys":self.__concat_keys(list(self.body_fields.keys())),
           "values":self.__concat_values(list(self.body_fields.values()))
       }
       self.sql=template%body_snippet_fields
   def__update(self):
       template="UPDATE%(tables)sSET%(fields)s"
       body_snippet_fields={
           "tables":self.tables_snippet,
           "fields":_concat_fields(self.body_fields,("=",","))
       }
       self.sql=template%body_snippet_fields
   def__delete(self):
       template="DELETEFROM%(tables)s"
       body_snippet_fields={
           "tables":self.tables_snippet
       }
       self.sql=template%body_snippet_fields
   def__build(self):
       {
           "SELECT":self.__select,
           "INSERT":self.__insert,
           "UPDATE":self.__update,
           "DELETE":self.__delete
       }[self.current_token]()
   def__unicode__(self):
       returnself.sql
   def__str__(self):
       returnself.__unicode__()
   defselect(self,*args):
       self.current_token="SELECT"
       self.body_keys=args
       self.__build()
       returnself
   definsert(self,**kwargs):
       self.current_token="INSERT"
       self.body_fields=self.__wrap_fields(kwargs)
       self.__build()
       returnself
   defupdate(self,**kwargs):
       self.current_token="UPDATE"
       self.body_fields=self.__wrap_fields(kwargs)
       self.__build()
       returnself
   defdelete(self,*conditions):
       self.current_token="DELETE"
       self.__build()
       #if*conditions:
       self.where(*conditions)
       returnself
   defwhere(self,*conditions):
       conditions=list(map(str,conditions))
       self.condition_snippet="AND".join(conditions)
       self.__where()
       ifhasattr(self,"where_snippet"):
           self.sql+=self.where_snippet
       returnself
下面举几个例子供大家参考吧:
db=DataHelper("/home/wangye/sample.db3")
db.open()#打开数据库
db.execute("""
   CREATETABLE[staffs](
     [staff_id]INTEGERPRIMARYKEYAUTOINCREMENT,
     [staff_name]TEXTNOTNULL,
     [staff_cardnum]TEXTNOTNULL,
     [staff_reserved]INTEGERNOTNULL
)
""")#直接执行SQL语句,注意这里commit_at_once默认为True
db.table("staffs").insert(staff_name="John",staff_cardnum="1001",staff_reserved=0)
#插入一条记录
rs=db.table("staffs").select("staff_id","staff_name").fetchall()
#直接取出所有staff_id和staff_name
rs=db.table("staffs").select("staff_name").where(DataCondition(("=","AND"),id=1)).fetchone()
#取一条staff_id为1的staff_name
rs=db.table("staffs").select("staff_name").where(DataCondition(("<","AND"),id=100),DataCondition(("=","AND"),staff_reserved=1)).fetchone()
#取一条id小于100并且staff_reserved为1的staff_name记录
db.close()#关闭数据库
目前还没有让其支持星号(*)操作符,另外在多表同名列操作方面处理得也不是很好,这个只用于日常简单的脚本操作,最好不要用于生产环境,因为可能有未知问题。
