如何讓用Python做一個縮放自如的圣誕老人,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。
為平陸等地區(qū)用戶提供了全套網(wǎng)頁設(shè)計制作服務(wù),及平陸網(wǎng)站建設(shè)行業(yè)解決方案。主營業(yè)務(wù)為成都做網(wǎng)站、網(wǎng)站制作、平陸網(wǎng)站設(shè)計,以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務(wù)。我們深信只要達(dá)到每一位用戶的要求,就會得到認(rèn)可,從而選擇與我們長期合作。這樣,我們也可以走得更遠(yuǎn)!
圣誕節(jié)又要到了,雖說我們中國人不提倡過西方的節(jié)日,但是商家們還是很喜歡的,估計有對象的男孩紙女孩紙們也很喜歡吧。
今天的主題是為大家展示如何用python做一個不斷變大的圣誕老人,就像西游記中能夠隨意變幻大小的神仙妖怪那樣,算是送給大家的小禮物,先上個圖吧!
不要心急,盯著圖片看5秒
思路要點:
通過縮放獲取等比大小的一組圖片
將上述圖片疊加到固定大小的底圖中
按順序組合圖片生成動圖
本篇文章的大部分工作都是基于opencv實現(xiàn),而opencv進行圖片縮放是極其容易的,不過這次我們要生成的是一組等比縮放的圖片,所以在cv2.resize方法的使用上可能跟以往略有出入,先來看函數(shù)原型:
cv2.resize(src, dsize[, dst[, fx[, fy[, interpolation]]]])
其中src是原圖片,dsize是目標(biāo)圖片大小,當(dāng)dsize為0的時候,我們就可以通過fx和fy兩個參數(shù)來分別設(shè)置水平軸和垂直軸方向的縮放比例了。這樣說可能有些抽象,我們舉個例子來說明:
for i in range(1, 40, 1): img = cv2.resize(image, (0, 0), fx=i/30, fy=i/30) cv2.imwrite(str(i)+'.png', img)
運行上面這段代碼會生成39張不同比例的圖片,目標(biāo)圖片的大小由縮放比例fx和fy來控制,最小的一幅圖邊長是原圖的1/30,最大的圖片邊長是原圖的1.3倍(下圖):
既然等比縮放的圖片有了,是不是可以選定一個坐標(biāo)原點,直接合成動圖呢?答案是不行,因為常規(guī)的動圖生成方法要求素材圖片必須是相同的尺寸(像素),下面我們就來著重解決這一問題。
python中實現(xiàn)兩幅圖片疊加的辦法有很多,但是他們都存在缺陷——要么疊加的圖片必須是相同大小,要么難以控制圖片疊加的具體位置。對此,小編采取的辦法是在兩幅圖之間進行“像素級”的替換。
1).生成底圖
待疊加的圖片中,上層圖片就使用剛才獲取到的一系列等比縮放圖,下層圖片我們就生成一張固定大小的空白圖片。需要注意,這里生成的空白圖片必須大于最大的一幅縮放圖。
生成空白底圖分兩步完成,第一步生成固定大小(垂直軸和水平軸的長度)的二維數(shù)組;第二步使用cv2.cvtColor進行顏色空間變換。代碼如下:
blank = np.ones((blankh, blankw), dtype=np.uint8) * 255 ret = cv2.cvtColor(blank, cv2.COLOR_GRAY2BGR)
其實上面代碼中的ret本質(zhì)上是一個三維數(shù)組,我們可以把它打印出來查看(下圖),但是通過cv2.imshow方法展示出來就是一張空白圖片了。這其中涉及一些較為底層的內(nèi)容,大家了解就好,文中不再贅述。
2).像素替換
正如剛才所說,opencv中的一幅圖其實是一個三維數(shù)組,其實也可以把它看作是二維數(shù)組,數(shù)組中的
每個元素是形如 [255, 255, 255] 的列表,其中存放的是圖片每個像素的顏色參數(shù)。也就是說,如果我們想實現(xiàn)一幅圖片疊加到另一幅圖片這樣的視覺效果,可以對被疊加圖片對應(yīng)位置的
像素進行替換賦值。代碼形式如下圖所示,其中i和j分別為圖片在垂直方向和水平方向的坐標(biāo)。
ret[i, j, 0] = image[i, j, 0] ret[i, j, 1] = image[i, j, 1] ret[i, j, 2] = image[i, j, 2]
對一幅圖片而言,坐標(biāo)原點是在左上角(下圖所示)。此外,為了保證最終得到動圖的效果,不能簡單的將圖片以坐標(biāo)原點為基準(zhǔn)進行疊加,比較好的辦法是把疊加原點設(shè)在底圖下邊緣的中心位置。
原理搞清楚后就可以開始圖片疊加操作了,在此期間需要進行一些像素對應(yīng)位置的計算,雖然稍微有點繞但是并不復(fù)雜,詳細(xì)的轉(zhuǎn)化公式就不寫了,我們直接看代碼:
上面代碼中的image是已經(jīng)縮放完畢的圣誕老人圖片,blankh和blankw分別是空白圖片的高度和寬度,這個尺寸可以根據(jù)需求自行設(shè)置。
下圖展示的是一幅縮放比例1/2左右的圖片和底圖疊加后的效果,為了觀察方便,我給圖片加了一個邊框。
之前我們已經(jīng)解決了單幅圖片與底圖的疊加,為了準(zhǔn)備合成動圖所需素材,還要對多個等比縮放的圖片進行底圖疊加操作??s放比例間隔越小、準(zhǔn)備的圖片素材越多,生成的動圖也就越平滑。
當(dāng)然,動圖的效果如何還要綜合考慮多個因素,這里小編還是采用39幅圖片組合動圖。其中最小的圖形高度是原圖的1/30,最大的圖形高度是原圖的1.3倍。與底圖疊加后的圖片就是下面這個樣子。
下面來說說動圖的合成,將多個相同尺寸的圖片合成動圖可以使用imageio這個庫來實現(xiàn),核心代碼只有一條:
imageio.mimwrite('目標(biāo)文件名稱.gif', gifList, duration=0.15)
其中第一個參數(shù)是git目標(biāo)文件名稱;gifList是一組列待合成的圖片,也就是上面圖片中展示的那些;最后一個參數(shù)duration表示畫面切換間隔,單位為秒。
現(xiàn)在通過下面這段代碼進行動圖合成。
file_path = 'pic' imgList = os.listdir(file_path) imgList = ['pic/'+img for img in imgList] gifList = [imageio.imread(img) for img in imgList] imageio.mimwrite('gif.gif', gifList, duration=0.15)
來看合成后的動圖效果(下圖),仔細(xì)看看好像有點問題,怎么圖中的圣誕老人忽大忽小?這跟我們預(yù)想的不一樣啊。
其實這個問題是出在合成圖片的順序上,我們嘗試打印上面代碼中的imgList變量,結(jié)果如下:
可以看到,素材圖片并不是按照我們預(yù)想的順序排序。這在python的文件處理中也算是個比較常見的問題,解決方案之一是可以按照圖片的創(chuàng)建時間排序,具體操作是在上面的第二行代碼之后插入一條語句:
imgList = sorted(imgList,key=lambda x: os.path.getmtime(os.path.join(file_path, x)))
現(xiàn)在再次進行動圖合成,就可以實現(xiàn)文章開頭的效果了。
當(dāng)然了,這種動圖制作方法不僅限于圣誕老人,任何圖片理論上都是可以的。比如說,我們還可以做一棵不斷長大的圣誕樹!
看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進一步的了解或閱讀更多相關(guān)文章,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對創(chuàng)新互聯(lián)的支持。
當(dāng)前標(biāo)題:如何讓用Python做一個縮放自如的圣誕老人
文章出自:http://aaarwkj.com/article42/ipooec.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供手機網(wǎng)站建設(shè)、品牌網(wǎng)站建設(shè)、網(wǎng)站策劃、用戶體驗、網(wǎng)站建設(shè)、Google
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)