1. 程式人生 > >Yii2開發技巧 使用類似閉包的方式封裝事務

Yii2開發技巧 使用類似閉包的方式封裝事務

在控制器中執行事務的時候,一般的程式碼如下:


$transaction = Yii::$app->db->beginTransaction();
try {
    //一些業務程式碼
    $transaction->commit();
} catch (\Exception $e) {
    $transaction->rollBack();
    throw $e;
}

於是我在想,這個程式碼結構,只有//一些業務程式碼 這一部分是不一樣,卻要重複很多遍,這一不是很冗餘嗎? 而且 不!好!看!,於是我試著尋找解決方法,一開始在stackflow找到一個類似的提問,有方案是在model裡做封裝,但是這樣做有一定問題,如產生巢狀事務等,有興趣的可以點選這裡

檢視該問答

我們的Yii框架給出了一個方法transaction,乍一看好像不能解決傳參的問題,我們先不管,往下看,該方法呼叫方式如下:


Yii::$app->db->transaction(function() {
    //一些業務程式碼
});

我們來看一下這個方法的原始碼


/**
 * Executes callback provided in a transaction.
 *
 * @param callable $callback a valid PHP callback that performs the job. Accepts connection instance as parameter.
 * @param string|null $isolationLevel The isolation level to use for this transaction.
 * See [[Transaction::begin()]] for details.
 * @throws \Exception|\Throwable if there is any exception during query. In this case the transaction will be rolled back.
 * @return mixed result of callback function
 */
public function transaction(callable $callback, $isolationLevel = null)
{
    $transaction = $this->beginTransaction($isolationLevel);
    $level = $transaction->level;

    try {
        $result = call_user_func($callback, $this);
        if ($transaction->isActive && $transaction->level === $level) {
            $transaction->commit();
        }
    } catch (\Exception $e) {
        $this->rollbackTransactionOnLevel($transaction, $level);
        throw $e;
    } catch (\Throwable $e) {
        $this->rollbackTransactionOnLevel($transaction, $level);
        throw $e;
    }

    return $result;
}

這個方法接受一個回撥函式和事務的隔離級別,
從這裡我們看出,這個方法雖然解決重複程式碼,卻還有幾個問題沒有解決:
第一,這個方法丟擲的異常我們需要在接收外面處理,我們不可能直接丟擲,這樣對客戶端很不友好。
第二:沒有記錄日誌的行為,即使出了問題也不容易排除。
第三:其實還是第一個問題,如果我們需要對每個異常做處理,在transaction方法外再巢狀一層try...catch...,那麼和沒有封裝好像沒什麼區別?

根據方法可擴充套件不可修改的原則,我們應該在自己公共方法裡對這個方法進行過載,過載程式碼如下:


public static function TransactionExecute(callable $function,$level=null)
{
    try{
        \Yii::$app->db->transaction($function,$level);
}catch (\Exception $e){
        //記錄日誌
        \Yii::error($e->getMessage());
        //這裡可以理解成丟擲自定義的異常類。
        (new self())->returnWayTip(1004, 'trans異常錯誤');
    }
}

然後回到如何傳參的問題,我們可以使用閉包,貼一段虛擬碼,如下:


//執行事務
PublicFunction::TransactionExecute(function () use ($token_reward, $reward_info) {
        //業務程式碼
        $token_reward->save(0);
    MsgHelper::send($reward_info['post_id'], MsgHelper::SOMEONE_FINISH_REWARD, $reward_info);

    });




大功告成,程式碼看起來有沒有更好看呢?

如有問題,歡迎指教。

原文地址:https://segmentfault.com/a/1190000015998299