用Python在Excel里画出蒙娜丽莎的方法示例


Posted in Python onApril 28, 2020

之前看到过很多头条,说哪国某人坚持了多少年自学使用excel画画,效果十分惊艳。
对于他们的耐心我十分敬佩。
但是作为一个程序员,自然也得挑战一下自己。
这种需求,我们十分钟就可以完成!

用Python在Excel里画出蒙娜丽莎的方法示例

基本思路

实现这个需求的基本思路是读取这张图片每一个像素的色彩值,然后给excel里的每一个单元格填充上颜色。所以主要用到的是PILopenpyxl这两个库。

PIL使用

PIL是Python里面做图像处理的时候十分常用的一个库,功能也是十分的强大,这里只需要用到PIL里一小部分的功能。

from PIL import Image
img = Image.open(img_path) # 读取图片
width, height = img.size # 获取图片大小
r, g, b = img.getpixel((w - 1, h - 1)) # 获取像素色彩值

Image.open()是PIL里面打开一张图片的函数,支持多种图片类型

img_path是图片路径,可以是相对路径,也可以是绝对路径

img.size是获取图片的size属性,包含图片的宽和高

img.getpixel() 是获取图片色彩值的函数,需传入一个tuplelist,值为像素坐标xy

openpyxl使用

openpyxl几乎是Python里功能最全的操作excel文件的库了,这里也只需要用到它的一小部分功能。

import openpyxl
from openpyxl.styles import fills

workbook = openpyxl.Workbook() 
worksheet = workbook.active
cell.fill = fills.PatternFill(fill_type="solid", fgColor=hex_rgb)
workbook.save(out_file)

openpyxl.Workbook() 新建一个excel文件

workbook.active 激活一个工作表

cell.fill = fills.PatternFill(fill_type="solid", fgColor=hex_rgb)填充一个单元格,fill_type="solid"是填充类型,fgColor=hex_rgb是填充的颜色

workbook.save()保存文件,需传入要保存的文件名

写一段代码

​ 写这一个画图的需求需要用到的核心就是上面介绍的PILopenpyxl的几种用法。但是在实际写的时候,还会有一些其他问题,比如:

getpixel()获取的颜色值是rgb十进制的,但fills.PatternFill 里的fgColor`参数接收到的颜色值是十六进制的值

这个问题其实就是十进制转十六进制,很容易解决

def int_to_16(num):
  num1 = hex(num).replace('0x', '')
  num2 = num1 if len(num1) > 1 else '0' + num1 # 位数只有一位的时候在前面补零
  return num2

excel的单元格默认是长方形,修改为正方形才不会使图片变形

if h == 1:
 _w = cell.column
 _h = cell.col_idx
 # 调整列宽
 worksheet.column_dimensions[_w].width = 1

# 调整行高
worksheet.row_dimensions[h].height = 6

这里用到了双重for循环,外层是width,里层是height,是一列一列的填充颜色,因此判断if h == 1,避免多次调整列宽。

excel支持的样式数量有限

这个问题比较严重。如果直接对高清大图进行操作,最后输出的excel文件在打开的时候,可能会提示我们文件有问题,需要自动修复。

但是等它修复完成之后,会发现填充的所有颜色都消失了!

用Python在Excel里画出蒙娜丽莎的方法示例

一开始以为是使用的行列数过多了原因。

查询资料后发现,13版excel支持的大行数是1048576,最大列数是16384,我们使用的单元格数量还远没达到限制。

在经过更换图片、更换excel版本,修改代码等不充分各种测试,才找到问题的原因所在。

原来是因为,excel的原形是由多个xml文件,填充的颜色都存储在一个style.xml文件里面,当这个文件过大就会导致打开的时候报错。

所以为了解决这个问题,有两个解决方案,第一是缩小图片,第二是减少图片颜色。缩小图片的时候自带减少图片颜色的功能,减少图片颜色的方法可以采用灰度化、二值化等方法。

总体上来讲,就是需要控制颜色数量*单元格数<阈值(3300w左右)

MAX_WIDTH = 300
MAX_HEIGHT = 300
def resize(img):
  w, h = img.size
  if w > MAX_WIDTH:
    h = MAX_WIDTH / w * h
    w = MAX_WIDTH

  if h > MAX_HEIGHT:
    w = MAX_HEIGHT / h * w
    h = MAX_HEIGHT
  return img.resize((int(w), int(h)), Image.ANTIALIAS)

最终效果

苍天不负有心人,打开最后输出的excel已经可以看到效果了!

所以说,一切能用Python解决的问题,最终都会用Python来解决。

用Python在Excel里画出蒙娜丽莎的方法示例

全部代码

# draw_excel.py

from PIL import Image
import openpyxl
from openpyxl.styles import fills
import os

MAX_WIDTH = 300
MAX_HEIGHT = 300

def resize(img):
  w, h = img.size
  if w > MAX_WIDTH:
    h = MAX_WIDTH / w * h
    w = MAX_WIDTH

  if h > MAX_HEIGHT:
    w = MAX_HEIGHT / h * w
    h = MAX_HEIGHT
  return img.resize((int(w), int(h)), Image.ANTIALIAS)


def int_to_16(num):
  num1 = hex(num).replace('0x', '')
  num2 = num1 if len(num1) > 1 else '0' + num1
  return num2


def draw_jpg(img_path):

  img_pic = resize(Image.open(img_path))
  img_name = os.path.basename(img_path)
  out_file = './result/' + img_name.split('.')[0] + '.xlsx'
  if os.path.exists(out_file):
    os.remove(out_file)

  workbook = openpyxl.Workbook()
  worksheet = workbook.active

  width, height = img_pic.size

  for w in range(1, width + 1):

    for h in range(1, height + 1):
      if img_pic.mode == 'RGB':
        r, g, b = img_pic.getpixel((w - 1, h - 1))
      elif img_pic.mode == 'RGBA':
        r, g, b, a = img_pic.getpixel((w - 1, h - 1))

      hex_rgb = int_to_16(r) + int_to_16(g) + int_to_16(b)

      cell = worksheet.cell(column=w, row=h)

      if h == 1:
        _w = cell.column
        _h = cell.col_idx
        # 调整列宽
        worksheet.column_dimensions[_w].width = 1
      # 调整行高
      worksheet.row_dimensions[h].height = 6
      
      cell.fill = fills.PatternFill(fill_type="solid", fgColor=hex_rgb)

    print('write in:', w, ' | all:', width + 1)
  print('saving...')
  workbook.save(out_file)
  print('success!')

if __name__ == '__main__':
  draw_jpg('mona-lisa.jpg')

附:

上面说到 颜色数量*单元格数<阈值(256^4左右)的时候,可能有人会有疑惑,这个256^4是怎么来的。

这个当然是我测试信口开河得来的。

既然是有颜色数量单元格数这两个变量,那么自然要有两个测试方法以上。一个观察颜色数量,一个观察单元格数

但我在这里只做了颜色数量的一个测试。(最大的原因就是生成上万行*上万列的excel实在是太久了...懒...)

count = 0
  MAX_WIDTH = 255
  for w in range(1, MAX_WIDTH + 1):
    for h in range(1, MAX_WIDTH + 1):
      cell = worksheet.cell(column=w, row=h)
      if h == 1:
        _w = cell.column
        _h = cell.col_idx
        # 调整列宽
        worksheet.column_dimensions[_w].width = 1
      # 调整行高
      worksheet.row_dimensions[h].height = 6
      
      if count < 255 ** 3:
        back = int_to_16(num=count)
        back = '0' * (6 - len(back)) + back
      else:
        back = ''.join([int_to_16(random.randint(0, 255)) for _ in range(3)])
      cell.fill = fills.PatternFill(fill_type="solid", fgColor=back)
      count += 1

count是记录颜色的变量,确保每一个颜色都不重复,但目前计算机RGB表示的颜色最多只有256^3种

通过调整MAX_WIDTH的值来测试excel的阈值

最后生成的测试excel如下:

用Python在Excel里画出蒙娜丽莎的方法示例

...还有点好看。!??

最后

由于精力有限,能力也有限 ,所以没有测试单一颜色的,也可能有其他的方法能没有这个阈值的限制。

代码已经全部上传至github,欢迎大家友好交流讨论 [握手]

到此这篇关于用Python在Excel里画出蒙娜丽莎的方法示例的文章就介绍到这了,更多相关Python Excel画蒙娜丽莎内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
Python中计算三角函数之cos()方法的使用简介
May 15 Python
基于pycharm导入模块显示不存在的解决方法
Oct 13 Python
解决python 无法加载downsample模型的问题
Oct 25 Python
Pycharm更换python解释器的方法
Oct 29 Python
python tkinter实现界面切换的示例代码
Jun 14 Python
Django Form 实时从数据库中获取数据的操作方法
Jul 25 Python
python实现翻译word表格小程序
Feb 27 Python
Anconda环境下Vscode安装Python的方法详解
Mar 29 Python
Python利用PyPDF2库获取PDF文件总页码实例
Apr 03 Python
Python如何获取文件路径/目录
Sep 22 Python
python实现计算器简易版
Dec 17 Python
详解解决jupyter不能使用pytorch的问题
Feb 18 Python
全网首秀之Pycharm十大实用技巧(推荐)
Apr 27 #Python
python实时监控logstash日志代码
Apr 27 #Python
python实现秒杀商品的微信自动提醒功能(代码详解)
Apr 27 #Python
浅析python 定时拆分备份 nginx 日志的方法
Apr 27 #Python
python异步Web框架sanic的实现
Apr 27 #Python
python库skimage给灰度图像染色的方法示例
Apr 27 #Python
python实现密度聚类(模板代码+sklearn代码)
Apr 27 #Python
You might like
Flash空降上海 化身大魔王接受挑战
2020/03/02 星际争霸
linux下实现定时执行php脚本
2015/02/13 PHP
php内嵌函数用法实例
2015/03/20 PHP
深入浅析PHP7.0新特征(五大新特征)
2015/10/29 PHP
PHP数据库处理封装类实例
2016/12/24 PHP
php生成毫秒时间戳的实例讲解
2017/09/22 PHP
Chosen 基于jquery的选择框插件使用方法
2012/05/30 Javascript
使用js实现关闭js弹出层的窗口
2014/02/10 Javascript
JavaScript操作DOM元素的childNodes和children区别
2015/04/01 Javascript
javascript实现右侧弹出“分享到”窗口效果
2016/02/01 Javascript
浅谈Javascript中的函数、this以及原型
2016/10/09 Javascript
详解JS-- 浮点数运算处理
2016/11/28 Javascript
在vue中,v-for的索引index在html中的使用方法
2018/03/06 Javascript
微信小程序基于Taro的分享图片功能实践详解
2019/07/12 Javascript
关于JavaScript数组去重的一些理解汇总
2020/09/10 Javascript
[42:27]DOTA2上海特级锦标赛主赛事日 - 3 败者组第三轮#2Fnatic VS OG第三局
2016/03/05 DOTA
Python实现学生成绩管理系统
2020/04/05 Python
深入分析python数据挖掘 Json结构分析
2018/04/21 Python
使用python读取csv文件快速插入数据库的实例
2018/06/21 Python
浅谈python 中类属性共享的问题
2019/07/02 Python
Python通过递归获取目录下指定文件代码实例
2019/11/07 Python
python实现串口通信的示例代码
2020/02/10 Python
Python使用plt.boxplot() 参数绘制箱线图
2020/06/04 Python
python脚本第一行如何写
2020/08/30 Python
使用Python+Appuim 清理微信的方法
2021/01/26 Python
CSS3盒子模型详解
2013/04/24 HTML / CSS
澳大利亚现代波西米亚风格女装网站:Bohemian Traders
2018/04/16 全球购物
英语专业毕业生求职简历的自我评价
2013/10/24 职场文书
人力资源主管岗位职责
2014/01/29 职场文书
迎新晚会主持词
2014/03/24 职场文书
标准版离职证明书
2014/09/12 职场文书
2015年教师新年寄语
2014/12/08 职场文书
学校青年志愿者活动总结
2015/05/06 职场文书
集结号观后感
2015/06/08 职场文书
员工旷工检讨书
2015/08/15 职场文书
Python selenium模拟网页点击爬虫交管12123违章数据
2021/05/26 Python