學習python3第一句大概率是 print(“hello world”) 吧?既然可以逐行逐個地輸出字符,那么把字符組成漢字應該也不難吧?經過一番搜索與嘗試,實現了通過python print來打印點陣字的小代碼,請看——
image
點陣字
實現此效果,首先是點陣字的概念:
點陣字體是把每一個字符都分成16×16或24×24個點,然后用每個點的虛實來表示字符的輪廓。點陣字體也叫位圖字體,其中每個字形都以一組二維像素信息表示。
我們拿Python的print來打印的話,最好是有現成可用的點陣字庫,這樣根據其像素信息直接轉化為print的字符就可以了。
HZK16字庫
HZK即漢字庫的首字母縮寫,HZK16字庫是符合GB2312標準的16×16點陣字庫,支持的漢字有6763個,每個漢字模型需要16×16一共需要256個點來顯示,每個點是二進制位也就是2的256次方數據,即32個字節。
那么思路清晰了,根據字符串中漢字字符編碼,去HZK16字庫中獲取點陣信息,拿到信息后根據16*16點陣每個點的數據,print出不同字符。
代碼思路
代碼細節可以點擊左下角“閱讀原文”查看,或者后臺回復print代碼獲取,在這里只列部分實現單字的代碼片段。
import binascii
KEYS = [0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01]
#初始化16*16的點陣位置,每個漢字需要16*16=256個點來表示
rect_list = [] * 16
for i in range(16):
rect_list.append([] * 16)
#拿“贊”字來演示
text = "贊"
#獲取中文的編碼
gb2312 = text.encode('gb2312')
hex_str = binascii.b2a_hex(gb2312)
result = str(hex_str, encoding='utf-8')
#根據編碼計算“贊”在漢字庫中的位置
area = eval('0x' + result[:2]) - 0xA0
index = eval('0x' + result[2:]) - 0xA0
offset = (94 * (area-1) + (index-1)) * 32
font_rect = None
#讀取HZK16漢字庫文件中“贊”字數據
with open("HZK16", "rb") as f:
f.seek(offset)
font_rect = f.read(32)
#根據讀取到HZK中數據給我們的16*16點陣賦值
for k in range(len(font_rect) // 2):
row_list = rect_list[k]
for j in range(2):
for i in range(8):
asc = font_rect[k * 2 + j]
flag = asc & KEYS[i]
row_list.append(flag)
#根據獲取到的16*16點陣信息,打印到控制臺
for row in rect_list:
for i in row:
if i:
#前景字符(即用來表示漢字筆畫的輸出字符)
print('0', end=' ')
else:
# 背景字符(即用來表示背景的輸出字符)
print('.', end=' ')
print()
單字效果
image
多字效果
image
以及Pycharm編輯器中輸出效果如下
image
最終成果
用戶自定義輸入漢字或短語,運行代碼用自定義的符號來展示點陣字:
源代碼:
import binascii
KEYS = [0x80, 0x40, 0x20, 0x10, 0x08, 0x04, 0x02, 0x01]
def printPlay(textStr,line,background):
# 初始化16*16的點陣位置,每個漢字需要16*16=256個點來表示,需要32個字節才能顯示一個漢字
# 之所以32字節:256個點每個點是0或1,那么總共就是2的256次方,一個字節是2的8次方
rect_list = [] * 16
for i in range(16):
rect_list.append([] * 16)
for text in textStr:
#獲取中文的gb2312編碼,一個漢字是由2個字節編碼組成
gb2312 = text.encode('gb2312')
#將二進制編碼數據轉化為十六進制數據
hex_str = binascii.b2a_hex(gb2312)
#將數據按unicode轉化為字符串
result = str(hex_str, encoding='utf-8')
#前兩位對應漢字的第一個字節:區碼,每一區記錄94個字符
area = eval('0x' + result[:2]) - 0xA0
#后兩位對應漢字的第二個字節:位碼,是漢字在其區的位置
index = eval('0x' + result[2:]) - 0xA0
#漢字在HZK16中的絕對偏移位置,最后乘32是因為字庫中的每個漢字字模都需要32字節
offset = (94 * (area-1) + (index-1)) * 32
font_rect = None
#讀取HZK16漢字庫文件
with open("HZK16", "rb") as f:
#找到目標漢字的偏移位置
f.seek(offset)
#從該字模數據中讀取32字節數據
font_rect = f.read(32)
#font_rect的長度是32,此處相當于for k in range(16)
for k in range(len(font_rect) // 2):
#每行數據
row_list = rect_list[k]
for j in range(2):
for i in range(8):
asc = font_rect[k * 2 + j]
#此處&為Python中的按位與運算符
flag = asc & KEYS[i]
#數據規則獲取字模中數據添加到16行每行中16個位置處每個位置
row_list.append(flag)
#根據獲取到的16*16點陣信息,打印到控制臺
for row in rect_list:
for i in row:
if i:
#前景字符(即用來表示漢字筆畫的輸出字符)
print(line, end=' ')
else:
# 背景字符(即用來表示背景的輸出字符)
print(background, end=' ')
print()
inpt = input("寫你所想:")
lineSign = '■'
#lineSign = "0"
backgroundSign = '○'
#backgroundSign = "."
printPlay(inpt,lineSign,backgroundSign)