本篇文章給大家帶來的內容是關於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); });