django框架事务处理小结【ORM 事务及raw sql,customize sql 事务处理】
本文实例讲述了django框架事务处理。分享给大家供大家参考,具体如下:
django中要求事务处理的情况有两种:
1.基于djangoorM的transaction处理
2.是基于自定义SQL语句的transaction的处理,通常是比较复杂的SQL,用ORM处理不方便的时候用的。或者是大批量SQL语句执行,比较在意效率的情况下用。
首先说一下第二种情况,因为这种情况相对简单一点,没ORM那么多东西,用我写的一个方法来解释
fromdjango.dbimportconnection,transaction ..... defbatch_execsql(sqlarray): cursor=connection.cursor()#得到处理的游标对象 ret="" try: forsqlinsqlarray: cursor.execute(sql) transaction.commit_unless_managed()#这是重点,没有这条语句,就不会commit。 exceptException,e:#简单的异常处理,可以忽略 ret=str(e) cursor.close() returnret#有异常则返回异常,否则返回为空字符串
由上面可以看出transaction.commit_unless_managed()的重要性,这是自定义SQL语句情况下处理事务的方法.上面的例子中的sqlarray表示一个list,里面有很多自己写的SQL语句,而这些语句要求在一个事务中完成。
再来看看第一种情况,用ORM的时候,事务的处理.这在django的官方文档中有说明,下面简单翻译介绍下
1.django默认的事务,是自动处理的,当你在调用orM的model.save(),model.delete()的时候,所有改动会被立即提交的,相当于数据库设置了autocommit,没有隐藏的rollback.
2.对http请求的事务拦截,这是推荐的方式,使用了transaction中间件来完成,这是比较好的方法,但必须在settings.py中配置.
MIDDLEWARE_CLASSES=( 'django.middleware.cache.UpdateCacheMiddleware', 'django.contrib.sessions.middleware.SessionMiddleware', 'django.middleware.common.CommonMiddleware', 'django.middleware.transaction.TransactionMiddleware', 'django.middleware.cache.FetchFromCacheMiddleware', )
但需要注意的是,这样配置之后,与你中间件的配置顺序是有很大关系的。在TransactionMiddleware之后的所有中间件都会受到事务的控制。所以当你把session中间件放到Transaction之后,同样会受到影响。但 CacheMiddleware,UpdateCacheMiddleware,andFetchFromCacheMiddleware 不会受到影响,cache机制有自己的处理方式,用了内部的connection来处理
另外TransactionMiddleware只对default的数据库配置有效,如果要对另外的数据连接用这种方式,必须自己实现中间件。
3.自己来控制事务,这种情况下,你自己灵活控制事务.在settings.py中不用配置TransactionMiddleware中间件了,基本采用装饰模式来实现。
a)@transaction.autocommit,django默认的事务处理,采用此装饰模式会忽略掉全局的transaction设置
fromdjango.dbimporttransaction @transaction.autocommit defviewfunc(request): .... @transaction.autocommit(using="my_other_database") defviewfunc2(request): ....
b)@transaction.commit_on_success在一个方法中,所有工作完成后,提交事务。
fromdjango.dbimporttransaction @transaction.commit_on_success defviewfunc(request): .... @transaction.commit_on_success(using="my_other_database") defviewfunc2(request): ....
c)commit_manually(),完全自己处理,但如果你没有调用commit()或者rollback(),将会抛出TransactionManagementError异常.
fromdjango.dbimporttransaction @transaction.commit_manually defviewfunc(request): ... #Youcancommit/rollbackhoweverandwheneveryouwant transaction.commit() ... #Butyou'vegottoremembertodoityourself! try: ... except: transaction.rollback() else: transaction.commit() @transaction.commit_manually(using="my_other_database") defviewfunc2(request): ....
希望本文所述对大家基于Django框架的Python程序设计有所帮助。