如何让函数调用支持事务0715

一个项目,由两个人组成,小王和小明。小王为小明提供函数,将数据保存到数据库当中,小明不关心入库细节。下面小明是调用小王的代码。


InsertHouse(house);
InsertFridge(fridge);
InsertTelevision(television);

每次小明调用三个函数,创建一个房子,以及为这个房子添加一个冰箱和电视机。随着开发不断继续,小明发现一个很重要的问题,就是小王的函数接口并没有支持事务。在连续调用三个函数的中间,如果其中某次调用异常了,没有合适的方式做到回滚。如果由小明在调用测采用try catch来处理,代码会非常的丑陋。

try{InsertHouse(house);}catch{DeleteHouse(house);}
try{InsertFridge(fridge);}catch{DeleteHouse(house);DeleteFridge(fridge);}

try{InsertTelevision(television);}catch{DeleteHouse(house);DeleteFridge(fridge);DeleteTelevision(television);}


如果在删除的时候再次出异常那该怎么办?这么写下去就无穷尽了。

在函数的基础上支持事务的方法很多,本文既不可能穷举所有方案,也没有能力对所有方案一较高低,给出最优方案,本文只能给出一个有点意思的方案。

小王和小明坐了下来,他们首先想到的是将函数压缩成一个,在函数内测支持事务,保证这次调用要么成功,要么就完全失败,不会产生垃圾数据。
InsertAllThings(house,fridge,television);

但是这样一来,如果将来需要新增插入沙发该怎么办?这就需要修改函数的参数个数,而且不断新增下去,会越来越长,而且有的时候,并不是所有内容需要同时插入,一个家里面,可能先只有电视机,后来才买了冰箱和沙发。小王和小明又想到了另外一个方案,就是定义一个结构,来表达要插入的数据。
obj.house=house;
obj.fridge=fridge;
obj.television=television;
InsertAllThings(obj);
这样函数参数固定下来了,但是随着支持新增沙发,那么obj的参数要不断新增。

还有没有更好的方法呢?采用' 柯里化'的风格来支持,保证单个函数的简单整洁,又具备扩展性。
Begin().InsertHouse(house).InsertFridge(fridge).InsertTelevision(television).Commit();

在函数Begin中返回一个内部对象obj,后面每次调用Insert只是将参数保存在obj中,并且再次将obj返回,最后的Commit才是真正执行插入数据库操作。为了方便理解我给出部分函数的实现。
def Begin()
{return obj;}
def obj.InsertHouse()
{obj.house=house;return obj;}
def obj.Commit()
{//根据obj的参数来生成SQL语句,然后入库}

采用这种方式,使用者小明不用显式关心obj的属性,实现者小王可以不断增加函数来新增支持其他家具或者电器,并且每个新增的函数参数简单,都支持强类型检查,容易理解。

你可能感兴趣的:(如何让函数调用支持事务0715)