優化你的PHP程式碼,從現在做起
- 2019 年 11 月 7 日
- 筆記

前言
我一生的文章都會放在這裡,我的部落格,我希望每一行程式碼,每一段文字都能幫助你。https://github.com/CrazyCodes/Blog
大家好,我是CrazyCodes ,今天我們不聊工具、規範等等等等的輔助,就聊一下該如何寫一段「好」的程式碼,本文以我的職業生涯碰到的程式碼為例,如有出入請在評論區提出異議,謝謝。
搜索功能
搜索很常見,複雜的搜索大多出行在後台,舉個栗子,大概需求是這樣的
這是一個後台用戶列表的搜索功能
搜索條件 |
可否並行 |
是否必填 |
---|---|---|
用戶名 |
可以 |
否 |
手機號碼 |
可以 |
否 |
是否已認證 |
可以 |
是 |
用戶性別 |
可以 |
否 |
最近登錄時間 |
可以 |
否 |
賬戶餘額 |
可以 |
否 |
初學者程式碼
看到這些例子你是否不由的一顫,又要開始造輪子的是不是? 以原生的例子為例,開始你可能會這樣寫(以下為偽程式碼)
if (IS_POST) { $like = ''; if (isset($_POST['username'])) { $username = $_POST['username']; $like .= "username like '%" . $username . "%' and "; } if (isset($_POST['phone'])) { $phone = $_POST['phone']; $like .= "phone like '%" . $phone . "%' and"; } if ($_POST['is_auth']) { $isAuth = $_POST['is_auth']; $like .= "is_auth like '%" . $isAuth . "%' and"; } if ($_POST['sex']) { $sex = $_POST['sex']; $like .= "sex like '%" . $sex . "%' and"; } if ($_POST['time']) { $time = $_POST['time']; $like .= "time like '%" . $time . "%' and"; } if ($_POST['wallet']) { $wallet = $_POST['wallet']; $like .= "wallet like '%" . $wallet . "%' and"; } $like = rtrim($like, 'and'); $sql = "SELECT * FROM `user` WHERE {$like}"; } else { return view('user'); }
封裝
恩…,還不錯,結構清晰,傳統的初學者條型程式碼,接下來我們先封裝一下幾塊程式碼。
function post($param) { return isset($_POST[$param]) ? $_POST[$param] : null; } if (IS_POST) { $like = ''; if (post('username')) { $username = post('username'); $like .= "username like '%" . $username . "%' and "; } if (post('phone')) { $phone = post('phone'); $like .= "phone like '%" . $phone . "%' and"; } if (post('is_auth')) { $isAuth = post('is_auth'); $like .= "is_auth like '%" . $isAuth . "%' and"; } if (post('sex')) { $sex = post('sex'); $like .= "sex like '%" . $sex . "%' and"; } if (post('time')) { $time = post('time'); $like .= "time like '%" . $time . "%' and"; } if (post('wallet')) { $wallet = post('wallet'); $like .= "wallet like '%" . $wallet . "%' and"; } $like = rtrim($like, 'and'); $sql = "SELECT * FROM `user` WHERE {$like}"; } else { return view('user'); }
適當使用迭代
恩,至少我們可以自由控制post方法了,但是這類過程化程式碼維護性太低,我們再改進下
function post($param) { return isset($_POST[$param]) ? $_POST[$param] : false; } function postAll() { return $_POST; } if (IS_POST) { $like = ''; foreach (postAll() as $key => $value) { if (post($key)) { $like .= "{$key} like '%{$value}%' and"; } } $like = rtrim($like, 'and'); $sql = "SELECT * FROM `user` WHERE {$like}"; } else { return view('user'); }
面向對象
加個迭代程式碼看起來還算是整潔了點,作為PHP程式設計師,寫程式碼不面向對象不靠譜,把class加上
function request($param = null) { return new Request($param); } class Request { public function __construct(string $param = null) { return isset($_POST[$param]) ? $_POST[$param] : false; } public function all() { return $_POST; } } class User { public function index() { if (IS_POST) { $like = ''; foreach (request()->all() as $key => $value) { if (request($key)) { $like .= "{$key} like '%{$value}%' and"; } } $like = rtrim($like, 'and'); $sql = "SELECT * FROM `user` WHERE {$like}"; } else { return view('user'); } } }
對User的改造
我們在對User的類進行改造,做一些判斷及篩選
function request($param = null) { return new Request($param); } class Request { public function __construct(string $param = null) { return isset($_POST[$param]) ? $_POST[$param] : false; } public function all() { return $_POST; } } class User { public $request = [ 'username', 'phone', 'is_auth', 'sex', 'time', 'wallet' ]; public function index() { if (IS_POST) { $like = ''; foreach (request()->all() as $key => $value) { if (in_array($key, $this->request) && request($key)) { $like .= sprintf("%s like %s and", $key, $value); } } $like = rtrim($like, 'and'); $sql = "SELECT * FROM `user` WHERE {$like}"; } else { return view('user'); } } }
這就差不多了,對比真是的程式碼可能還相差甚遠,我寫這篇文章的目的不是教會你如何寫程式碼,是想說明編碼不是一次性的,應經過多次修改,使程式碼具有可維護性,擴展性等等的,各種「性」
致謝
感謝你看到這裡,希望本篇文章可以幫到你,謝謝