開(kāi)篇先來(lái)聊一聊縮略圖吧,其經(jīng)典應(yīng)用場(chǎng)景就是商品列表、詳情以及查看大圖時(shí)返回不同尺寸的圖片。好處,額,閉上眼睛自己體會(huì)。我看到過(guò)一些做法是在圖片保存時(shí)就生成三套縮略圖提供給前端訪問(wèn),這樣其實(shí)是可以滿(mǎn)足基本需求的,只是局限比較大,譬如下面這幾種情況:
1、系統(tǒng)初期是基于pc做的開(kāi)發(fā),生成的縮略圖也是提供給pc端使用。某個(gè)月黑風(fēng)高的夜晚,領(lǐng)導(dǎo)突然拉著你的小手說(shuō)我們明天開(kāi)始做移動(dòng)端,并且移動(dòng)端的需要的圖片尺寸和pc端不一樣。
2、換領(lǐng)導(dǎo)了,新來(lái)的領(lǐng)導(dǎo)說(shuō)列表頁(yè)為什么加載這么慢,把寬320px的圖片換成319px的,以前的數(shù)據(jù)全部要換~換~換~。
關(guān)于生成縮略圖的思考
為了技(yi)術(shù)(lao)創(chuàng)(yong)新(yi),我們重新設(shè)計(jì)了縮略圖的生成方式,與其生成固定不可更改的縮略圖,不如根據(jù)前端的需(bian)求(hua)來(lái)生成縮略圖,這樣無(wú)論前端是需要什么尺寸我們都可以輕松應(yīng)對(duì)。至于安全性,我們可以在后端配置允許訪問(wèn)的尺寸集合,遇到非法的請(qǐng)求,直接給他一張小黃圖就好。于是我們的圖片訪問(wèn)流程大概變成了這個(gè)樣子:
1、用戶(hù)發(fā)起圖片請(qǐng)求,服務(wù)端拿到請(qǐng)求,檢查圖片是否存在,不存在則返回小黃圖。
2、驗(yàn)證圖片請(qǐng)求是否帶有尺寸參數(shù),沒(méi)有則返回原圖,尺寸超過(guò)原圖也直接返回原圖。
3、如果帶有尺寸參數(shù),驗(yàn)證是否在服務(wù)器允許的參數(shù)范圍內(nèi),不在則返回小黃圖。
4、判斷該尺寸的縮略圖是否存在,不存在則生成縮略圖,存在即直接返回縮略圖。
縮略圖只會(huì)在第一次訪問(wèn)時(shí)生成,總體來(lái)說(shuō)不會(huì)太影響圖片訪問(wèn)速度。思路大概就是這個(gè)樣子,接下來(lái)就只需要從服務(wù)端拿到用戶(hù)的圖片請(qǐng)求并處理。
關(guān)于圖片請(qǐng)求url的思考
平時(shí)我們?cè)L問(wèn)圖片的方式大概是這個(gè)樣子,http://xx.com/xx.png,那么帶參數(shù)的圖片請(qǐng)求會(huì)是個(gè)什么樣子呢,我想最好像這樣吧:http://xx.com/xx.png?width=320。
后端如何拿到這個(gè)請(qǐng)求的參數(shù)并控制其返回的內(nèi)容。我能想到的有兩個(gè)辦法:
1、將圖片訪問(wèn)集中到一個(gè)webapi接口,將圖片路徑和尺寸當(dāng)做參數(shù)傳過(guò)來(lái)統(tǒng)一處理,webapi接口以流的形式返回圖片。
2、是否可以將特定的圖片請(qǐng)求使用一個(gè)自定義的IHttpHandler來(lái)處理。
其實(shí)仔細(xì)一想,第一個(gè)辦法實(shí)現(xiàn)起來(lái)并不是那么優(yōu)雅,并且也不能達(dá)到我們http://xx.com/xx.png?width=320的需求,甚至圖片返回的速度也會(huì)變慢不少,那么我們來(lái)分析第二個(gè)辦法的可行性。我的圖片都是存儲(chǔ)在/upload/image/這個(gè)目錄之下,那么前端的圖片訪問(wèn)也必然是http://file.com/upload/image/xx.png,我們只需要將帶/upload/image/的請(qǐng)求映射到我們自定義的IHttpHandler來(lái)處理即可。
解決方案
1、自定義HttpHandler:
using System.Drawing;using System.Web;using cczcrv.Web.File.Uploader.Image;using System.IO;namespace cczcrv.Web.File.Filters { public class GetImageHandler : IHttpHandler { public void ProcessRequest(HttpContext context) { //圖片不存在,返回默認(rèn)圖片 &n