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

作者: 红尾黄花鱼 | 来源:发表于2016-07-15 21:56 被阅读37次

    一个项目,由两个人组成,小王和小明。小王为小明提供函数,将数据保存到数据库当中,小明不关心入库细节。下面小明是调用小王的代码。
    <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的属性,实现者小王可以不断增加函数来新增支持其他家具或者电器,并且每个新增的函数参数简单,都支持强类型检查,容易理解。

    相关文章

      网友评论

        本文标题:如何让函数调用支持事务0715

        本文链接:https://www.haomeiwen.com/subject/tpnyjttx.html