## 搜索關鍵字的顯示處理

> 其實很多人都忽視了這個問題,不信你到一些網站上搜索 `'`、`"`、`\`試一下,看看會不會出現驚喜和意外。
頁面中顯示搜索詞有兩種方式:
1. 用js設置填充數據。
2. 直接渲染在標簽中。
但這兩種方式都要面臨一個問題,**那就是用戶輸入的特殊詞不能引起我們頁面錯誤,html和JavaScript代碼不能夠受其影響而出錯。**
如何在不限制用戶輸入特殊詞的情況下,滿足這個要求呢?
下面我們就來探討一下這個問題。
如何解決用戶搜索特殊詞時,顯示沒有問題,搜索框中顯示也沒有問題,**不能引起html結構和js代碼出錯**。搜什么顯示什么,**并且再次點擊搜索按鈕時,搜索的還是當前的詞。**
* * * * *
最簡單的模糊搜索就是這樣了
```php
// 不用過濾,不用進行任何處理(只是清除左右的空格符)
$keyword = trim($request->param('keyword/s'));
// 畢竟不用拼接sql,所以不會有任何安全問題
$where[$field] = ['like', '%' . $keyword . '%'];
```
得益于現代ORM和PDO的完善和普及,讓我們不再有拼接sql字符串的風險問題了,因為我們不需要直接拼接sql了,所以不用在進行任何過濾等處理就可以直接查詢,但是用戶搜索的特殊字符在頁面上顯示時怎么處理呢?比如搜索框中,頁面上等,很多人都忽略了這個問題,但凡細心一點的人,稍加測試就能發現端倪,下面就來看一下:
```html
<!-- 搜索框中顯示搜索詞 -->
<input name="q" type="text" placeholder="搜索:課程" value="" autocomplete="on">
<!-- 頁面顯示搜索詞 -->
<span class="search-keyword">"{:getQ(1)}"</span> 的搜索結果
```
```javascript
// 用js來設置搜索框的value
$('.input-text').val("{:getQ(2)}");
```
js設置時就不會出現引號和html沖突了(因為將沖突可能性轉移到了js中嘛),**并且這樣直接搜索時所見即所得,顯示詞即為搜索詞。**
```php
// 獲取搜索詞
function getQ($mode = 0)
{
$request = Request::instance();
$q = $request->param('q/s', '');
// 默認不做任何處理直接返回
if ($mode == 1) {
// html實體轉義
// html中使用
$q = htmlspecialchars($q, ENT_QUOTES);
} elseif ($mode == 2) {
// 單引號(')雙引號(")反斜杠(\)NULL 之前添加反斜杠
// JS中使用
$q = addslashes($q);
}
return $q;
}
```
### 用表單插值插件處理的搜索詞
```php
// 關鍵字搜索(后端只是去除左右多余空格符,并不會對其進行任何處理)
$keyword = trim($request->param('keyword/s'));
$searchDate['keyword'] = isset($_GET['keyword']) ? $_GET['keyword'] : '';
```
```javascript
var formObj = new Form('user-form');
// json_encode() 會自動轉換雙引號和反斜線,以確保生成正確的json格式字符串,而這種轉移和js是相通的,所以能夠直接使用沒有問題
var searchDate = {:json_encode($searchDate)};
formObj.init(searchDate);
```
* * * * *
#### 總結
1. 我們遇到的問題是,特殊搜索詞的顯示(頁面中顯示與input搜索框中顯示),和再次搜索時的搜索詞不能一致的問題(我很難表述這個問題,需要你自己去感受)。
2. 我們用兩種方式解決了這個問題,**html實體轉義** 和 **js字符串轉義**
* * * * *
### 補充
你有沒有遇到過,將一個頁面鏈接通過微信發給別人,微信中卻不能完整的顯示地址
```
http://www.a.com/index/index/search.html?q=地方'地方
```

比如這個地址,有很多軟件并不能夠很好的識別這樣的鏈接,對這樣的鏈接支持得不是很友好(QQ和微信),導致你發過去后,別人不能夠直接點擊完整的鏈接,非要復制到瀏覽器才能打開完整的鏈接,這個體驗就非常不好。
雖說這是個小問題,但是對于追求極致用戶體驗的我們來說,怎么能夠忍受呢。
有什么辦法解決這個問題呢。
其實瀏覽器向服務器發送的url都是經過url編碼了:
只是瀏覽器地址欄經過優化,為了照顧普通用戶,而顯示的是這樣的地址而已,我們復制就是從地址欄復制的,于是就出現這個問題了。
知道了原因后,解決方式也很簡單。那就是自己降低至url編碼,而不是讓瀏覽器自己編碼,你看大型網站,淘寶,京東都是這樣自己編碼處理的,所以地址欄你看不到搜索的中文關鍵字,都是被編碼過了,這樣就可以愉快的到處分享了。
不過貌似現代新的瀏覽器不會有這個問題了,看到的是中文,復制下來就是已經編碼過的。但是如果你遇到了這樣的問題,那解決方案就在這里。
* * * * *
### 擴展
[PHP addslashes() 函數](http://www.w3school.com.cn/php/func_string_addslashes.asp)
[PHP htmlspecialchars() 函數](http://www.w3school.com.cn/php/func_string_htmlspecialchars.asp)
[前端安全系列(一):如何防止XSS攻擊? - 美團技術團隊的個人空間 - 開源中國](https://my.oschina.net/meituantech/blog/2218539)
[前端安全系列之二:如何防止CSRF攻擊? - 美團技術團隊的個人空間 - 開源中國](https://my.oschina.net/meituantech/blog/2243958)
* * * * *
last update:2018-7-1 10:34:40
- 開始
- 公益
- 更好的使用看云
- 推薦書單
- 優秀資源整理
- 技術文章寫作規范
- SublimeText - 編碼利器
- PSR-0/PSR-4命名標準
- php的多進程實驗分析
- 高級PHP
- 進程
- 信號
- 事件
- IO模型
- 同步、異步
- socket
- Swoole
- PHP擴展
- Composer
- easyswoole
- php多線程
- 守護程序
- 文件鎖
- s-socket
- aphp
- 隊列&并發
- 隊列
- 講個故事
- 如何最大效率的問題
- 訪問式的web服務(一)
- 訪問式的web服務(二)
- 請求
- 瀏覽器訪問阻塞問題
- Swoole
- 你必須理解的計算機核心概念 - 碼農翻身
- CPU阿甘 - 碼農翻身
- 異步通知,那我要怎么通知你啊?
- 實時操作系統
- 深入實時 Linux
- Redis 實現隊列
- redis與隊列
- 定時-時鐘-阻塞
- 計算機的生命
- 多進程/多線程
- 進程通信
- 拜占庭將軍問題深入探討
- JAVA CAS原理深度分析
- 隊列的思考
- 走進并發的世界
- 鎖
- 事務筆記
- 并發問題帶來的后果
- 為什么說樂觀鎖是安全的
- 內存鎖與內存事務 - 劉小兵2014
- 加鎖還是不加鎖,這是一個問題 - 碼農翻身
- 編程世界的那把鎖 - 碼農翻身
- 如何保證萬無一失
- 傳統事務與柔性事務
- 大白話搞懂什么是同步/異步/阻塞/非阻塞
- redis實現鎖
- 淺談mysql事務
- PHP異常
- php錯誤
- 文件加載
- 路由與偽靜態
- URL模式之分析
- 字符串處理
- 正則表達式
- 數組合并與+
- 文件上傳
- 常用驗證與過濾
- 記錄
- 趣圖
- foreach需要注意的問題
- Discuz!筆記
- 程序設計思維
- 抽象與具體
- 配置
- 關于如何學習的思考
- 編程思維
- 談編程
- 如何安全的修改對象
- 臨時
- 臨時筆記
- 透過問題看本質
- 程序后門
- 邊界檢查
- session
- 安全
- 王垠
- 第三方數據接口
- 驗證碼問題
- 還是少不了虛擬機
- 程序員如何談戀愛
- 程序員為什么要一直改BUG,為什么不能一次性把代碼寫好?
- 碎碎念
- 算法
- 實用代碼
- 相對私密與絕對私密
- 學習目標
- 隨記
- 編程小知識
- foo
- 落盤
- URL編碼的思考
- 字符編碼
- Elasticsearch
- TCP-IP協議
- 碎碎念2
- Grafana
- EFK、ELK
- RPC
- 依賴注入
- 科目一
- 開發筆記
- 經緯度格式轉換
- php時區問題
- 解決本地開發時調用遠程AIP跨域問題
- 后期靜態綁定
- 談tp的跳轉提示頁面
- 無限分類問題
- 生成微縮圖
- MVC名詞
- MVC架構
- 也許模塊不是唯一的答案
- 哈希算法
- 開發后臺
- 軟件設計架構
- mysql表字段設計
- 上傳表如何設計
- 二開心得
- awesomes-tables
- 安全的代碼部署
- 微信開發筆記
- 賬戶授權相關
- 小程序獲取是否關注其公眾號
- 支付相關
- 提交訂單
- 微信支付筆記
- 支付接口筆記
- 支付中心開發
- 下單與支付
- 支付流程設計
- 訂單與支付設計
- 敏感操作驗證
- 排序設計
- 代碼的運行環境
- 搜索關鍵字的顯示處理
- 接口異步更新ip信息
- 圖片處理
- 項目搭建
- 閱讀文檔的新方式
- mysql_insert_id并發問題思考
- 行鎖注意事項
- 細節注意
- 如何處理用戶的輸入
- 不可見的字符
- 抽獎
- 時間處理
- 應用開發實戰
- python 學習記錄
- Scrapy 教程
- Playwright 教程
- stealth.min.js
- Selenium 教程
- requests 教程
- pyautogui 教程
- Flask 教程
- PyInstaller 教程
- 蜘蛛
- python 文檔相似度驗證
- thinkphp5.0數據庫與模型的研究
- workerman進程管理
- workerman網絡分析
- java學習記錄
- docker
- 筆記
- kubernetes
- Kubernetes
- PaddlePaddle
- composer
- oneinstack
- 人工智能 AI
- 京東
- pc_detailpage_wareBusiness
- doc
- 電商網站設計
- iwebshop
- 商品規格分析
- 商品屬性分析
- tpshop
- 商品規格分析
- 商品屬性分析
- 電商表設計
- 設計記錄
- 優惠券
- 生成唯一訂單號
- 購物車技術
- 分類與類型
- 微信登錄與綁定
- 京東到家庫存系統架構設計
- crmeb
- 命名規范
- Nginx https配置
- 關于人工智能
- 從人的思考方式到二叉樹
- 架構
- 今日有感
- 文章保存
- 安全背后: 瀏覽器是如何校驗證書的
- 避不開的分布式事務
- devops自動化運維、部署、測試的最后一公里 —— ApiFox 云時代的接口管理工具
- 找到自己今生要做的事
- 自動化生活
- 開源與漿果
- Apifox: API 接口自動化測試指南