openpyxl 读写excel2007使用笔记

有一万年没来这儿写blog了,今天等着发版,抽空来写点东西吧。

python有很多模块都是用来操作excel的,比如xlrd,xlwt,pyExcelerator。用着很方便,但是问题是,只能支持到excel2003。虽然一般的应用其实足够了,但是如果遇到了导出大量数据(超过65535条)的需求时,excel2003就不够用了。所以我就只好去找一个能支持excel2007的模块。

 

google了一下,发现了这个openpyxl,不过网上也没什么中文的文档可以看,于是就自己琢磨琢磨。

 

 

I.安装

 

openpyxl的最新版本是1.5.0 下载地址 http://pypi.python.org/pypi/openpyxl/1.5.0

 

需求python的版本是2.6+  ,也就是说,如果centOS系统的机器想用,那得升级系统的python。。。

 

安装就是解压缩,然后cd到目录,然后

 

python setup.py install

 

你懂的~

 

 

 II.读取excel2007文件

 

 

   

 

 

 

 注意的是ws.cell()方法,支持的参数有两种,cell(coordinate=None, row=None, column=None)

coordinate坐标,eg  ws.cell("B1")

row 和 column 是行和列 ,都是从0开始

 

还有,如果想取得格里的值,得用ws.cell("A1").value 取到,如果用过xlrd,因为写法差不多,可能就会忘记加value了。

 

III.写入excel2007

 

 

 

 

 注意的地方:

 

# col是用列号x为参数,调用了这个模块的get_column_letter方法算出来的字母,这个比较蛋疼。

   col = get_column_letter(x)

 

#在为数据格赋值的时候,注意写的格式:要不会有各种不靠谱的问题出现(这个是用坐标的方式写的,其实用row ,col的方式可能没那么麻烦)

  ws.cell('%s%s'%(col, i)).value = '%s' % (record[x-1])

 

关于该模块的API  可以查询官方文档  http://packages.python.org/openpyxl/api.html

 

总体来说,这个模块还是挺方便的,但是问题就是在对于python的版本有一定要求,如果在centOs上用,可能会有些问题。

 

本文只是简单的写了下使用的方法,之后格式超链接神马的之后再填上~

 

 

关于pyExcelerator的使用教学,给朋友做个广告  http://blog.csdn.net/suofiya2008/archive/2010/05/14/5589627.aspx

 

 

 

<think> 我们正在处理一个关于在Django中使用openpyxl导出Excel并插入图片的问题。根据引用,我们知道: - django-import-export库用于导入导出数据,但它可能不直接支持在Excel中插入图片(引用[1])。 - openpyxl是一个处理Excel文件的Python库,支持读写xlsx文件,并且具备插入图片的功能(引用[2])。 - 另外,引用[4]中提到了Django导出Excel并返回给前端下载,但示例中并没有插入图片的部分,且前端使用了XMLHttpRequest。 因此,我们需要结合openpyxl的图片插入功能,在Django视图中生成包含图片的Excel文件,然后将其作为HTTP响应返回给用户。 步骤: 1. 安装openpyxl:首先确保已经安装了openpyxl库。 2. 在Django视图中使用openpyxl创建一个工作簿。 3. 向工作表中添加数据和图片。 4. 将工作簿保存到虚拟文件(例如BytesIO)中。 5. 创建HttpResponse返回Excel文件,设置正确的Content-Type和Content-Disposition。 具体实现: 假设我们要导出一个包含图片的Excel文件,图片来自Django模型的ImageField字段(例如,一个产品模型的产品图片)。 示例模型(models.py): ```python from django.db import models class Product(models.Model): name = models.CharField(max_length=100) image = models.ImageField(upload_to='products/') ``` 视图(views.py): ```python from django.http import HttpResponse from openpyxl import Workbook from openpyxl.drawing.image import Image as OpenpyxlImage from .models import Product from io import BytesIO def export_products_excel(request): # 创建一个工作簿 wb = Workbook() ws = wb.active ws.title = "Products" # 添加表头 ws['A1'] = "Product Name" ws['B1'] = "Image" # 获取所有产品 products = Product.objects.all() # 遍历产品,将产品名称和图片添加到Excel中 for index, product in enumerate(products, start=2): # 从第2行开始 ws.cell(row=index, column=1, value=product.name) # 如果产品有图片 if product.image: # 获取图片的完整路径 img_path = product.image.path # 使用openpyxl加载图片 try: img = OpenpyxlImage(img_path) # 调整图片大小(可选) img.width = 100 img.height = 100 # 将图片添加到工作表的指定位置,例如B列对应的行 ws.add_image(img, f'B{index}') except Exception as e: # 处理图片加载错误 print(f"Error loading image: {e}") # 设置列宽以适应图片(可选) ws.column_dimensions['B'].width = 30 # 设置B列列宽 # 创建虚拟文件对象 virtual_workbook = BytesIO() wb.save(virtual_workbook) virtual_workbook.seek(0) # 将文件指针移动到文件开头 # 创建HTTP响应 response = HttpResponse( virtual_workbook.read(), content_type='application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' ) response['Content-Disposition'] = 'attachment; filename=products.xlsx' return response ``` 注意: - 上述代码中,我们直接使用了图片的物理路径(`product.image.path`)。确保Django服务器有权限访问该路径,并且在生产环境中,需要注意文件存储的位置(例如,如果使用远程存储,可能需要先将图片下载到本地)。 - 如果图片存储在云端(如AWS S3),则需要先将图片下载到临时文件,然后再使用openpyxl加载。例如: ```python from tempfile import NamedTemporaryFile import requests # 假设product.image.url是一个可公开访问的URL(或者使用签名URL) response = requests.get(product.image.url) if response.status_code == 200: img_temp = NamedTemporaryFile(delete=True) img_temp.write(response.content) img_temp.flush() img = OpenpyxlImage(img_temp.name) # ... 然后添加到工作表 # 注意:NamedTemporaryFile在关闭后会自动删除,所以不需要手动删除 ``` 此外,我们还需要在Django项目的urls.py中配置一个URL来访问这个视图。 安全性考虑:确保只有授权用户才能访问该导出视图。 相关问题: 1. 如何调整Excel中图片的大小和位置? 2. 如果Excel中需要插入大量图片,如何优化性能以避免内存溢出? 3. 除了openpyxl,Django中还有哪些库可以用于导出Excel并插入图片? 4. 如何处理存储在云存储(如AWS S3)中的图片插入到Excel? 希望这个回答对你有帮助! </think> ### 在Django中使用openpyxl导出Excel并插入图片的方法 以下是完整实现步骤(基于openpyxl特性[^2]和Django文件响应机制): #### 1. 安装依赖 ```bash pip install openpyxl pillow # pillow用于图片处理 ``` #### 2. 视图函数实现 ```python from django.http import HttpResponse from openpyxl import Workbook from openpyxl.drawing.image import Image from io import BytesIO import os def export_excel_with_images(request): # 创建Excel工作簿 wb = Workbook() ws = wb.active ws.title = "带图片报表" # 添加表头 ws['A1'] = "产品ID" ws['B1'] = "产品名称" ws['C1'] = "产品图片" # 示例数据(实际应从数据库获取) products = [ {"id": 1, "name": "手机", "image_path": "products/phone.jpg"}, {"id": 2, "name": "笔记本", "image_path": "products/laptop.png"}, ] # 插入数据和图片 for row, product in enumerate(products, start=2): ws.cell(row=row, column=1, value=product["id"]) ws.cell(row=row, column=2, value=product["name"]) # 插入图片关键步骤 try: # 获取图片绝对路径(需配置MEDIA_ROOT) img_full_path = os.path.join(settings.MEDIA_ROOT, product["image_path"]) img = Image(img_full_path) # 调整图片尺寸(可选) img.width = 100 img.height = 100 # 定位图片位置(C列对应行) ws.add_image(img, f'C{row}') # 调整行高适应图片 ws.row_dimensions[row].height = 80 except Exception as e: print(f"图片插入失败: {e}") # 设置列宽 ws.column_dimensions['A'].width = 15 ws.column_dimensions['B'].width = 20 ws.column_dimensions['C'].width = 30 # 创建内存文件对象 virtual_file = BytesIO() wb.save(virtual_file) virtual_file.seek(0) # 重置指针位置 # 创建HTTP响应 response = HttpResponse( virtual_file.getvalue(), content_type='application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' ) response['Content-Disposition'] = 'attachment; filename="products_with_images.xlsx"' return response ``` #### 3. URL配置(urls.py) ```python from .views import export_excel_with_images urlpatterns = [ path('export/', export_excel_with_images, name='export-excel'), ] ``` #### 4. 前端调用示例 ```html <a href="{% url 'export-excel' %}" class="btn btn-primary"> 下载带图片的Excel报表 </a> ``` #### 关键技术点说明: 1. **图片处理**: - 使用`openpyxl.drawing.image.Image`加载图片[^2] - 需确保图片路径正确(建议使用Django的`MEDIA_ROOT`设置) - 支持JPG/PNG等常见格式 2. **文件响应**: - 使用`BytesIO`创建内存文件避免物理存储 - 设置MIME类型为`application/vnd.openxmlformats-officedocument.spreadsheetml.sheet` - 通过Content-Disposition触发文件下载 3. **布局调整**: - 使用`column_dimensions`/`row_dimensions`调整行列尺寸 - 图片锚点定位(如`f'C{row}'`表示C列当前行) > **注意**:处理云存储图片时需先下载到本地临时文件,参考以下代码片段: > ```python > from tempfile import NamedTemporaryFile > import requests > > # 从URL下载图片 > response = requests.get(image_url) > with NamedTemporaryFile(delete=False) as tmp: > tmp.write(response.content) > img = Image(tmp.name) > ```
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值