Yii2開發技巧 使用類似閉包的方式封裝事務
阿新 • • 發佈:2018-11-17
在控制器中執行事務的時候,一般的程式碼如下:
$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