图片转字符画基本思想
图片转字符画的关键思想是将图片的灰度值与你自己设定的字符集之间建立映射关系,不同区间的灰度值对应不同的字符,之后将图片每一个像素对应的字符打印出来。
前置知识
一、RGB颜色转灰度的算法公式
- 心理学公式
Gray = R0.299 + G0.587 + B0.114 - 整数运算算法(缩放1000倍):避免低俗的浮点运算
Gray = (R299 + G587 + B114 + 500) / 1000
(RGB一般是8位精度,现在缩放1000倍,所以上面的运算是32位整型的运算。注意后面那个除法是整数除法,所以需要加上500来实现四舍五入。) - 整数运算算法的变种
Gray = (R30 + G59 + B11 + 50) / 100 - 移位
将系数缩放成 2的整数幂。习惯上使用16位精度,2的16次幂是65536,所以这样计算系数:
0.299 * 65536 = 19595.264 ≈ 19595
0.587 * 65536 + (0.264) = 38469.632 + 0.264 = 38469.896 ≈ 38469
0.114 * 65536 + (0.896) = 7471.104 + 0.896 = 7472
(舍入方式是去尾法)
Gray = (R19595 + G38469 + B*7472) >> 16
二、字符列表
灰度值小(暗)的用列表开头的符号,灰度值大(亮)的用列表末尾的符号。字符的种类越多,色彩越丰富。(可调整字符顺序,达到自己想要的效果)
三、Image类的方法
- Convert
将当前图像转换为其他模式,并且返回新的图像:im.convert(mode)⇒ image
当从一个颜色图像转换为黑白图像: im.convert(“L”) - Getpixel
返回给定位置的像素值。如果图像为多通道,则返回一个元组:im.getpixel(xy) ⇒ value or tuple
具体实现
有两种方法:
1.先将彩色图片转换为黑白图片,然后直接将每个像素点的灰度值与字符集建立映射。
2.获取图片的RGB值,利用公式计算每个像素点的灰度值,之后再建立映射即可。
代码示例:
from PIL import Image
codeLib = '''@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~<>i!lI;:,"^`'. '''#生成字符画所需的字符集
count = len(codeLib)
## 1.先将彩色图片转换为黑白图片,然后直接将每个像素点的灰度值与字符集建立映射。
def transform1(image_file):
image_file = image_file.convert("L")#转换为黑白图片,参数"L"表示黑白模式
codePic = ''
for h in range(0,image_file.size[1]): #size属性表示图片的分辨率,'0'为横向大小,'1'为纵向 height
for w in range(0,image_file.size[0]): #width
gray = image_file.getpixel((w,h)) #返回指定位置的像素,如果所打开的图像是多层次的图片,那这个方法就返回一个元组
codePic = codePic + codeLib[int(((count-1)*gray)/256)]#建立灰度与字符集的映射
codePic = codePic+'\r\n'
return codePic
## 2.获取图片的RGB值,利用公式计算每个像素点的灰度值,之后再建立映射即可。
def transform2(image_file):
codePic = ''
for h in range(0,image_file.size[1]):
for w in range(0,image_file.size[0]):
g,r,b = image_file.getpixel((w,h))
gray = int(r* 0.299+g* 0.587+b* 0.114)
codePic = codePic + codeLib[int(((count-1)*gray)/256)]
codePic = codePic+'\r\n'
return codePic
WIDTH = 60 # 字符画的宽
HEIGHT = 30 # 字符画的高
if __name__ == '__main__':
img = '2.jpg' # 图片所在位置
im = Image.open(img)
im = im.resize((WIDTH, HEIGHT), Image.NEAREST)
txt = ""
# 将字符画 写入文件中
with open("output.txt", 'w') as f:
f.write(transform2(im))