查看原文
其他

30 行代码带你用 Python 在命令行查看图片

派森酱 Python技术 2022-09-11

文 | 豆豆

来源:Python 技术「ID: pythonall」

每次看黑客类的电影时都惊叹于黑客的技术之高超,黑客的手在键盘上飞快的敲击,屏幕上各种字符狂闪不止,接着系统就被黑掉了。

我们都知道黑客在命令行发送的各种命令都是英文字母或者数字,那么我们是否可以直接在命令行查看图片呢?

答案是可以的,今天派森酱就带你在终端查看图片。

先来看看我们最终实现的效果。

安装

我们今天要用到 pillow 库,直接使用 pip 安装即可。

$ pip install pillow

使用之前需要先将相应模块引入我们的程序。

from PIL import Imagefrom PIL import ImageFilter

Hello World

PIL(Python Imaging Library)是 Python 平台上一个非常好用的图像处理库,功能强大且 API 简单易用,但仅支持到 Python 2.7,所以有人就在 PIL 的基础上创建了新的版本 pillow,可以支持到 Python 3.x。

来看看模糊效果,只需三四行代码。

im = Image.open('/tmp/qq.jpg')im2 = im.filter(ImageFilter.BLUR) # 应用模糊滤镜:im2.save('/tmp/qq_2.jpg', 'jpeg')

图像缩放,几行代码轻松搞定。

im = Image.open('/tmp/qq.jpg')w, h = im.sizeim.thumbnail((w // 2, h // 2)) # 缩放到 50%im.save('/tmp/qq_thumbnail.jpg', 'jpeg')

字符画

众所周知,终端只可以显示字符,那如果可以把图片转化为字符画,岂不是可以在终端正常显示了,事实上还真可以这么做。其原理就是将图片不同位置的像素点用不同的字符来代替,从而将由像素组成的图片转变成由字符组成的字符画。

但是像素是有颜色的,我们如何将不同颜色的像素编码为对应的字符呢?

这就要涉及到灰度图的概念了。

灰度图,又称灰阶图,把白色与黑色之间按对数关系分为若干等级,称为灰度,共分为 256 阶,0 到 255,0 为黑色,255 为白。用灰度表示的图像称作灰度图。

事实上任何颜色都有红、绿、蓝三原色组成,假如某个像素点的颜色为 RGB(R,G,B),那么,我们可以通过固定公式将其转换为指定灰度即可。

GRAY = 0.2126 * R + 0.7152 * G + 0.0722 * B

拿到像素点的灰度值之后,我们将其对应到指定字符即可,为了最大限度的还原原图像,灰度值较高的像素点我们使用视觉上较深的字符表示(如$),灰度值较低的像素点我们使用视觉上较浅的字符(如:0)表示。

所以,字符的种类与数量越多,能表现的颜色也就越多,字符画的层次感也就会更好。最理想情况是将灰度值和字符一一对应,然而事实上字符数量是远远少于灰度值值域的,所以就会有多个灰度值对应到同一个字符上的情况。

比如我们只用 0 到 9 十个阿拉伯数字来作为我们的字符集,那么一个字符对应的灰度值区间大小就是 25.6(区间大小 = 256 / 字符集长度)。

灰度值区间和字符对应关系:

[0, 25.6) -- 0[25.6, 51.2) -- 1[51.2, 76.8) -- 2...[204.8, 230.4) -- 8[230.4, 256) -- 9

RGB 转换字符

有了以上的理论基础,我们将 RGB 转换为字符的函数定义如下:

ascii_char = list("$@B%8&WM#*oahkbdpqwmZO0QLCJUYXzcvunxrjft/\|()1{}[]?-_+~<>i!lI;:,\"^`'. ")def get_char_by_rgb(r, g, b): length = len(ascii_char) gray = int(0.2126 * r + 0.7152 * g + 0.0722 * b) unit = 256.0 / length return ascii_char[int(gray / unit)]

处理图像

完成了上面的转换函数后,我们就要开始对处理图像了。

首先我们需要调整下图片的大小,以免图片太大或者太小,然后我们遍历图片的每个像素点获取到 RGB 值,然后根据函数 get_char_by_rgb(r, g, b) 获取到对应的字符,最后将所有的字符拼拼接起来输出到终端即可。

注意,我们通过函数 getpixel(x,y) 获取坐标 (x,y) 的像素的 RGB 值,该函数的返回值为一个元组,比如 (1,2,3) 或者 (1,2,3,0) 最后一位的 0 是 alpha 值。alpha 的值为 0 是表示该位置空白。

所以,我们需要修改下 get_char_by_rgb(r, g, b) 函数,添加 alpha 参数。

def get_char_by_rgb(r, g, b, alpha = 256): if alpha == 0: return ' ' length = len(ascii_char) gray = int(0.2126 * r + 0.7152 * g + 0.0722 * b) unit = 256.0 / length return ascii_char[int(gray / unit)]

遍历图片获取像素点,然后转换为相应的字符。

def process_image(image_path, file_path='out.txt'): img = Image.open(image_path) img = img.resize((WIDTH, HEIGHT)) width, height = img.size txt = "" for x in range(height): for y in range(width): txt += get_char_by_rgb(*img.getpixel((y, x))) txt += '\n'
with open(file_path, 'w') as f: f.write(txt) print(txt)

最后编写入口函数,就大功告成啦。

if __name__ == '__main__': image_path, file_path = '/tmp/qq.jpg', '/tmp/qq.txt' process_image(image_path, file_path)

至此,我们完成了从原始图像到字符画的转换。

总结

今天我们实现了将图片转换字符画,以便在终端查看图片,可能在不同的终端会显示不同的效果,大家可酌情调整宽高比例。

PS公号内回复「Python」即可进入Python 新手学习交流群,一起100天计划!


老规矩,兄弟们还记得么,右下角的 “在看” 点一下如果感觉文章内容不错的话,记得分享朋友圈让更多的人知道!

代码获取方式

识别文末二维码,回复:200527

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存