99999久久久久久亚洲,欧美人与禽猛交狂配,高清日韩av在线影院,一个人在线高清免费观看,啦啦啦在线视频免费观看www

熱線電話:13121318867

登錄
首頁精彩閱讀技能 | 如何使用Python將文本轉(zhuǎn)為圖片
技能 | 如何使用Python將文本轉(zhuǎn)為圖片
2016-06-07
收藏

Python的發(fā)展在數(shù)據(jù)分析和數(shù)據(jù)挖掘領(lǐng)域可謂帶來了一場風(fēng)暴,其強(qiáng)大的嵌入性和豐富的庫使其越來越受歡迎。

比如,有時候,我們需要將文本轉(zhuǎn)換為圖片,比如發(fā)長微博,或者不想讓人輕易復(fù)制我們的文本內(nèi)容等時候。目前類似的工具已經(jīng)有了不少,不過我覺得用得都不是很趁手,于是便自己嘗試實現(xiàn)了一個。


Python 中,PIL (Python Imaging Library) 是最常用的繪圖庫,自然地,嘗試從 PIL 開始。

1、使用 PIL 將文字轉(zhuǎn)換為圖片

說轉(zhuǎn)換其實并不恰當(dāng),真實的過程是:先在內(nèi)存中生成一張圖片,將需要的文字繪制到這個圖片上,再將圖片保存到指定位置。代碼如下:

生成的圖片如下:

杯具發(fā)生了,漢字沒有正常顯示!

網(wǎng)上搜索了一圈,發(fā)現(xiàn)這好像是 PIL 的一個 bug,PIL 目前的版本中,不能正確處理非 ASCII 字符的點(diǎn)陣字體的渲染。對于像宋體這樣的字體來說,只有 >= 18px 時,才會被當(dāng)作矢量字體處理,也就是說只有當(dāng)字體 >= 18px 時,文字才能正常顯示:

效果如下:

增大字體雖然解決了漢字不能正常顯示的問題,但還是沒有解決我們一開始的初衷:使用點(diǎn)陣字體進(jìn)行渲染。但是,這個目標(biāo)使用現(xiàn)階段的 PIL 似乎有點(diǎn)難以實現(xiàn)了。

2、使用 pyGame 渲染點(diǎn)陣字體

Python 的第三方模塊或組件非常多,可用來繪圖的除了 PIL 之外,就還有 Pycairo、matplotlib、pyGame 等。在這兒,我使用 pyGame 來完成點(diǎn)陣字體的渲染工作。

代碼如下:

效果如下:

可以看到,使用 pyGame ,點(diǎn)陣字體的問題終于搞定了。

3、結(jié)合 PIL 和 pyGame

pyGame 雖然可以解決點(diǎn)陣字體的渲染問題,但講到對圖片的處理,還是 PIL 更為強(qiáng)大。那么,我們?yōu)槭裁床话褍烧呓Y(jié)合起來呢?用 pyGame 渲染點(diǎn)陣字體,然后用 PIL 生成整張圖片。

代碼如下:

原理很簡單,先將文字用 pyGame 渲染為圖片,將渲染結(jié)果保存在一個 StringIO 對象中,然后再用 PIL 加載它。使用 StringIO 的好處是,一切操作都是在內(nèi)存中進(jìn)行的,不需要先將它保存到硬盤再用 PIL 讀取,因為硬盤 IO 的效率相對來說是比較低的。

最終效果如下:

到這兒,使用 Python 將文本轉(zhuǎn)為圖片的功能就基本實現(xiàn)了,用到了 PIL 和 pyGame。

當(dāng)然,上面的代碼還只解決了最基本的問題,一個真正可用的文本轉(zhuǎn)圖片工具,還應(yīng)該解決以下問題:長文本換行問題、英文單詞斷字問題、標(biāo)點(diǎn)符號換行問題等。關(guān)于這些問題的分析篇幅也不短,這一次就先略過了。下面是一個綜合考慮了諸多因素之后生成的《荷塘月色》的效果圖:

文 | oldj
原文鏈接:http://oldj.net/article/text-to-image/

數(shù)據(jù)分析咨詢請掃描二維碼

若不方便掃碼,搜微信號:CDAshujufenxi

數(shù)據(jù)分析師資訊
更多

OK
客服在線
立即咨詢
客服在線
立即咨詢
') } function initGt() { var handler = function (captchaObj) { captchaObj.appendTo('#captcha'); captchaObj.onReady(function () { $("#wait").hide(); }).onSuccess(function(){ $('.getcheckcode').removeClass('dis'); $('.getcheckcode').trigger('click'); }); window.captchaObj = captchaObj; }; $('#captcha').show(); $.ajax({ url: "/login/gtstart?t=" + (new Date()).getTime(), // 加隨機(jī)數(shù)防止緩存 type: "get", dataType: "json", success: function (data) { $('#text').hide(); $('#wait').show(); // 調(diào)用 initGeetest 進(jìn)行初始化 // 參數(shù)1:配置參數(shù) // 參數(shù)2:回調(diào),回調(diào)的第一個參數(shù)驗證碼對象,之后可以使用它調(diào)用相應(yīng)的接口 initGeetest({ // 以下 4 個配置參數(shù)為必須,不能缺少 gt: data.gt, challenge: data.challenge, offline: !data.success, // 表示用戶后臺檢測極驗服務(wù)器是否宕機(jī) new_captcha: data.new_captcha, // 用于宕機(jī)時表示是新驗證碼的宕機(jī) product: "float", // 產(chǎn)品形式,包括:float,popup width: "280px", https: true // 更多配置參數(shù)說明請參見:http://docs.geetest.com/install/client/web-front/ }, handler); } }); } function codeCutdown() { if(_wait == 0){ //倒計時完成 $(".getcheckcode").removeClass('dis').html("重新獲取"); }else{ $(".getcheckcode").addClass('dis').html("重新獲取("+_wait+"s)"); _wait--; setTimeout(function () { codeCutdown(); },1000); } } function inputValidate(ele,telInput) { var oInput = ele; var inputVal = oInput.val(); var oType = ele.attr('data-type'); var oEtag = $('#etag').val(); var oErr = oInput.closest('.form_box').next('.err_txt'); var empTxt = '請輸入'+oInput.attr('placeholder')+'!'; var errTxt = '請輸入正確的'+oInput.attr('placeholder')+'!'; var pattern; if(inputVal==""){ if(!telInput){ errFun(oErr,empTxt); } return false; }else { switch (oType){ case 'login_mobile': pattern = /^1[3456789]\d{9}$/; if(inputVal.length==11) { $.ajax({ url: '/login/checkmobile', type: "post", dataType: "json", data: { mobile: inputVal, etag: oEtag, page_ur: window.location.href, page_referer: document.referrer }, success: function (data) { } }); } break; case 'login_yzm': pattern = /^\d{6}$/; break; } if(oType=='login_mobile'){ } if(!!validateFun(pattern,inputVal)){ errFun(oErr,'') if(telInput){ $('.getcheckcode').removeClass('dis'); } }else { if(!telInput) { errFun(oErr, errTxt); }else { $('.getcheckcode').addClass('dis'); } return false; } } return true; } function errFun(obj,msg) { obj.html(msg); if(msg==''){ $('.login_submit').removeClass('dis'); }else { $('.login_submit').addClass('dis'); } } function validateFun(pat,val) { return pat.test(val); }