你看过字符画吗?用 Python 自己实现一个吧

举报
梦想橡皮擦 发表于 2022/08/31 15:04:55 2022/08/31
【摘要】 ⛳️ 实战场景如果你是一名程序员,你肯定在网上的代码中看到过下述注释,那有没有想过自己实现一个呢?先说原理:图片由像素点构成,然后将每个像素点替换成一个字符,就可以将图片转换成字符画。这是最简单的说明,但是每个像素点都有一种颜色,而且该颜色是由红绿蓝三色组成,红绿蓝即 RBG 值,如果将所有颜色与字符替换,那存在 256 x 256 x 256 = 1638400 中颜色,完全与字符替换是...

⛳️ 实战场景

如果你是一名程序员,你肯定在网上的代码中看到过下述注释,那有没有想过自己实现一个呢?

先说原理:
图片由像素点构成,然后将每个像素点替换成一个字符,就可以将图片转换成字符画。

这是最简单的说明,但是每个像素点都有一种颜色,而且该颜色是由红绿蓝三色组成,红绿蓝即 RBG 值,如果将所有颜色与字符替换,那存在 256 x 256 x 256 = 1638400 中颜色,完全与字符替换是不可能的,此时就需要降低替换数量,这里就引入了灰度值,每个颜色值都可以转换成灰度值,而且灰度值的范围是 0-255,这样大幅降低了待替换数量,并且在 Python 中有现成的将图片转换为灰度图的库,实现起来就更加便捷了。

这里存在一个 RGB 转灰度值的公式:

gray = 0.2126 * r + 0.7152 * g + 0.0722 * b
# 也可以使用下述公式,含义一致。
gray = (2126 * r + 7152 * g + 722 * b) / 10000

提前安装 pillow 库,用于将图片转换为灰度图。

⛳️ pillow 转灰度图

使用 pillow 库操作下述图片,将其转换为字符图:

由于需要进行字符串替换,所以需要了解一些基本规则:

  • 颜色越浅,字符留白越多,当颜色为白时,使用空格字符;
  • 颜色越深,使用的字符笔画越多,例如黑色,可以使用 @ 这样的符号;

接下来我们就把图片转换成灰度值:

def img_to_char(path):
    img = Image.open(path)
    # 原图的宽和高
    ow, oh = img.size
    print(ow, oh)
    img = img.convert('L')
    img.show()


s = img_to_char('1.png')
print(s)

此时运行代码,就会得到一张灰色图片,效果如下:

接下来将图片中每个像素点的灰度值都提取出来,做后一步计算。

from PIL import Image

def img_to_char(path):
    img = Image.open(path)
    # 原图的宽和高
    ow, oh = img.size
    print(ow, oh)
    img = img.convert('L')
    # img.show()
    char_img = ""
    for y in range(oh):
        s = ''
        for x in range(ow):
            # 取灰度值
            gray = img.getpixel((x, y))
            # char = gray_to_char(gray)
            s += str(gray)
        char_img = char_img + s + '\n'
    return char_img
s = img_to_char('1.png')
print(s)

此时会得到下图效果,一堆数字,在数字中还是可以发现部分差异的。

数字有了,接下来就是数字转换为字符了,你可以依据前文提及的【黑白】程度进行排列,例如下述内容:

@#$%&MNBEFRWYLIkbtj?*984532menocvzst{}[]1|()<>=+~-;:i^"'.

数量和内容都可以自行控制,最后放置一个空格,用来替换白色。

# 将灰度值转换成对应的字符
def gray_to_char(gray):
    all_char = """@#$%&MNBEFRWYLIkbtj?*984532menocvzst{}[]1|()<>=+~-;:i^"'. """
    char = all_char[int(len(all_char) * gray / 256)]
    return char

然后把前文数字拼接的位置,替换成该函数。

 char_img = ""
    for y in range(oh):
        s = ''
        for x in range(ow):
            # 取灰度值
            gray = img.getpixel((x, y))
            char = gray_to_char(gray)
            s += char
        char_img = char_img + s + '\n'
    return char_img

此时在运行代码,就可以得到一张超大的字符图,下图是表情包眼睛区域。

接下来对图片进行一下优化,降低大小。

def img_to_char(path):
    img = Image.open(path)
    # 原图的宽和高
    ow, oh = img.size
    print(ow, oh)
    # 缩小高度和宽度
    w = int(ow * 0.3)
    h = int(oh * 0.3)
    # 调整图片大小
    img = img.resize((w, h))
    img = img.convert('L')
    # img.show()
    char_img = ""
    for y in range(h):
        s = ''
        for x in range(w):
            # 取灰度值
            gray = img.getpixel((x, y))
            char = gray_to_char(gray)
            s += char
        char_img = char_img + s + '\n'
    return char_img

等比例压缩之后,发现图片被拉伸了,原因是像素转换成文本,行与行之间有间隔,要缩小高度才能得到较好效果。

高度缩小到原来的二分之一,代码如下所示:

# 缩小高度和宽度
w = int(ow * 0.3)
h = int(oh * 0.3)

h = h // 2
# 调整图片大小
img = img.resize((w, h))
img = img.convert('L')

此时就可以得到一个比较合理大小的图片了,是不是一模一样!

📢📢📢📢📢📢
💗 你正在阅读 【梦想橡皮擦】 的博客
👍 阅读完毕,可以点点小手赞一下
🌻 发现错误,直接评论区中指正吧
📆 橡皮擦的第 711 篇原创博客

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。