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

一个项目,由两个人组成,小王和小明。小王为小明提供函数,将数据保存到数据库当中,小明不关心入库细节。下面小明是调用小王的代码。
<pre><code>
InsertHouse(house);
InsertFridge(fridge);
InsertTelevision(television);
</pre></code>
每次小明调用三个函数,创建一个房子,以及为这个房子添加一个冰箱和电视机。随着开发不断继续,小明发现一个很重要的问题,就是小王的函数接口并没有支持事务。在连续调用三个函数的中间,如果其中某次调用异常了,没有合适的方式做到回滚。如果由小明在调用测采用try catch来处理,代码会非常的丑陋。
<pre><code>
try{InsertHouse(house);}catch{DeleteHouse(house);}
try{InsertFridge(fridge);}catch{DeleteHouse(house);DeleteFridge(fridge);}

try{InsertTelevision(television);}catch{DeleteHouse(house);DeleteFridge(fridge);DeleteTelevision(television);}
</pre></code>
如果在删除的时候再次出异常那该怎么办?这么写下去就无穷尽了。

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

小王和小明坐了下来,他们首先想到的是将函数压缩成一个,在函数内测支持事务,保证这次调用要么成功,要么就完全失败,不会产生垃圾数据。
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的属性,实现者小王可以不断增加函数来新增支持其他家具或者电器,并且每个新增的函数参数简单,都支持强类型检查,容易理解。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,779评论 18 399
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,789评论 0 9
  • 师傅说,医者可以挽救天下人的性命,却独独救不了自己。这就是医者的宿命。 永远,不要爱上一个以剑为生的男人。
    岚风的叶子阅读 438评论 0 0
  • 你有多久没有清理你的“购物车”了?你收藏的文章有多久没看了?你购买的课程是否已经看完?你是否觉得你拥有了很多...
    米十七阅读 393评论 0 2
  • 既然是搭档 为什么对我不苟言笑 哪怕多说一句话都觉得是 浪费 是我的失败 还是你的不坦...
    彧或阅读 195评论 0 0