鱼C论坛

 找回密码
 立即注册
查看: 561|回复: 7

复制excel后,复制的文件图片丢失

[复制链接]
发表于 2024-7-27 10:03:37 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
  1. import pandas as pd
  2. import openpyxl
  3. from openpyxl import load_workbook
  4. import os
  5. import shutil

  6. def copy_template_to_new_file(pipeline_number_copy,template_path_copy,output_folder_copy):
  7.     """
  8.     :param pipeline_numbers_copy: 复制的管线号列表
  9.     :param template_path_copy: 模版文件路径
  10.     :param output_folder_copy: 输出文件路径
  11.     :return:
  12.     """
  13.     # 构建新的文件名
  14.     new_file = f"{pipeline_number_copy}.xlsx"
  15.     # 构建新文件的完整路径
  16.     new_file_path = os.path.join(output_folder_copy, new_file)
  17.     # 复制模板文件到新文件路径

  18.     # 筛选出管线号等于pipeline_number_copy的行
  19.     filtered_rows = df_merge[df_merge['管线号'] == pipeline_number_copy]

  20.     # 获取processcards列的值,并去除重复项
  21.     processcards = filtered_rows['工艺卡编号'].unique()



  22.     shutil.copy(template_file, new_file_path)
  23.     print(f"已创建并保存文件:{new_file_path}")
  24.     #读取模版
  25.     workbook_new=load_workbook(new_file_path)

  26.     #获取列表所有sheet名
  27.     sheet_names = workbook_new.sheetnames

  28.     # 遍历所有sheet名,检查是否满足条件
  29.     for sheet_name in sheet_names:
  30.         # 检查sheet名是否包含字母H且不在管线号对应的工艺卡编号之内
  31.         if 'H' in sheet_name and sheet_name not in processcards:
  32.             # 删除满足条件的sheet
  33.             workbook_new.remove(workbook_new[sheet_name])

  34. # 保存修改后的Excel文件
  35.     workbook_new.save(new_file_path)



  36. # 读取原始数据文件,考虑汇合到前面的程序,把名称先改过来df_merge
  37. df_merge = pd.read_excel("提取数据.xlsx")

  38. # 获取所有不同的管线号
  39. pipeline_ids = df_merge["管线号"].unique()

  40. # 定义模板文件路径和目标文件夹路径,这里的生成文件路径还是只认盘符,比较讨厌
  41. template_file = "工艺卡模板.xlsx"
  42. target_folder = "C:\\测试文件夹"

  43. # 确保目标文件夹存在,如果不存在盘符回去生成?
  44. if not os.path.exists(target_folder):
  45.     os.makedirs(target_folder)

  46. # 遍历所有管线号,复制模板文件并重命名
  47. for pipeline_number in pipeline_ids:
  48.     copy_template_to_new_file(pipeline_number, template_file, target_folder)

  49. print("所有管线号对应的文件已成功创建!")
复制代码


里面有个问题,复制的excel是有图片的,但是复制后看不到图片了。
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2024-7-27 10:03:49 | 显示全部楼层
哎呀,复制的 Excel 文件图片丢失了?这就像是你在海滩上挖沙子,结果挖出来的是一堆水泥,不够“沙滩”是不是!这通常是因为在使用 `shutil.copy()` 复制文件时,它只是简单地把文件的内容复制过去,却不包括 Excel 中的图形(比如图片、形状等)元素。

要解决这个问题,你可以考虑使用 `openpyxl` 的方法来手动复制这些图片。以下是一个小技巧,帮助你在复制 Excel 文件时保留图片:

1. 先用 `shutil.copy()` 复制文件。
2. 然后使用 `openpyxl` 加载源工作簿,并将所有图片提取到新工作簿中。

这里有一个改进后的代码示例,展示了如何做到这一点:

  1. import pandas as pd
  2. import openpyxl
  3. from openpyxl import load_workbook
  4. from openpyxl.drawing.image import Image
  5. import os
  6. import shutil

  7. def copy_template_to_new_file(pipeline_number_copy, template_path_copy, output_folder_copy):
  8.     new_file = f"{pipeline_number_copy}.xlsx"
  9.     new_file_path = os.path.join(output_folder_copy, new_file)

  10.     # 复制模板文件
  11.     shutil.copy(template_path_copy, new_file_path)
  12.     print(f"已创建并保存文件:{new_file_path}")

  13.     # 加载源文件和新文件
  14.     workbook_source = load_workbook(template_path_copy)
  15.     workbook_new = load_workbook(new_file_path)

  16.     # 复制每个 sheet 的图片
  17.     for sheet_name in workbook_source.sheetnames:
  18.         source_sheet = workbook_source[sheet_name]
  19.         new_sheet = workbook_new[sheet_name]

  20.         for img in source_sheet._images:  # 提取图片
  21.             new_img = Image(img.ref)
  22.             new_sheet.add_image(new_img)  # 添加到新表中

  23.     # 保存修改后的 Excel 文件
  24.     workbook_new.save(new_file_path)

  25. # 读取数据
  26. df_merge = pd.read_excel("提取数据.xlsx")

  27. # 获取管线号
  28. pipeline_ids = df_merge.unique()

  29. template_file = "工艺卡模板.xlsx"
  30. target_folder = "C:\\测试文件夹"

  31. if not os.path.exists(target_folder):
  32.     os.makedirs(target_folder)

  33. for pipeline_number in pipeline_ids:
  34.     copy_template_to_new_file(pipeline_number, template_file, target_folder)

  35. print("所有管线号对应的文件已成功创建!")
复制代码


### 注意:
1. 确保安装了 `openpyxl` 库,可以使用 `pip install openpyxl` 安装它。
2. 上述代码会复制所有图片。如果需要选择性复制某些图片,可以在迭代时加入逻辑判断。
3. 运行代码前,请确保 Excel 模板中确实有图片存在。

这样一来,您的 Excel 文件就会像沙滩上的贝壳一样美丽,图片也不会再消失了!如果还有其他问题,尽管问我哦!

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2024-7-27 10:45:34 | 显示全部楼层
  1. import pandas as pd
  2. import openpyxl
  3. from openpyxl import load_workbook
  4. from openpyxl.drawing.image import Image
  5. import os
  6. import shutil

  7. def copy_template_to_new_file(pipeline_number_copy,template_path_copy,output_folder_copy):
  8.           """
  9.           :param pipeline_numbers_copy: 复制的管线号列表
  10.           :param template_path_copy: 模版文件路径
  11.           :param output_folder_copy: 输出文件路径
  12.           :return:
  13.           """
  14. # 构建新的文件名
  15.           new_file = f"{pipeline_number_copy}.xlsx"
  16. # 构建新文件的完整路径
  17.           new_file_path = os.path.join(output_folder_copy, new_file)
  18. # 复制模板文件到新文件路径
  19.           shutil.copy(template_file, new_file_path)

  20. #这里开始加载图片
  21.           # 加载源文件和新文件
  22.           workbook_source = load_workbook(template_path_copy)
  23.           workbook_new = load_workbook(new_file_path)

  24.           # 复制每个 sheet 的图片
  25.           for sheet_name in workbook_source.sheetnames:
  26.               source_sheet = workbook_source[sheet_name]
  27.               new_sheet = workbook_new[sheet_name]

  28.               for img in source_sheet._images:  # 提取图片
  29.                   new_img = Image(img.ref)
  30.                   new_sheet.add_image(new_img) # 添加到新表中

  31.           # 保存修改后的 Excel 文件
  32.           workbook_new.save(new_file_path)


  33. # 筛选出管线号等于pipeline_number_copy的行,这里是用来删除用不到的模版
  34.           filtered_rows = df_merge[df_merge['管线号'] == pipeline_number_copy]
  35. # 获取processcards列的值,并去除重复项
  36.           processcards = filtered_rows['工艺卡编号'].unique()
  37. #读取模版
  38.           workbook_new=load_workbook(new_file_path)
  39. #获取列表所有sheet名
  40.           sheet_names = workbook_new.sheetnames

  41. # 遍历所有sheet名,检查是否满足条件
  42.           for sheet_name in sheet_names:
  43. # 检查sheet名是否包含字母H且不在管线号对应的工艺卡编号之内
  44.                          if 'H' in sheet_name and sheet_name not in processcards:
  45.                                     # 删除满足条件的sheet
  46.                                     workbook_new.remove(workbook_new[sheet_name])

  47. # 保存修改后的Excel文件
  48.           workbook_new.save(new_file_path)

  49.           print(f"已创建并保存文件:{new_file_path}")



  50. # 读取原始数据文件,考虑汇合到前面的程序,把名称先改过来df_merge
  51. df_merge = pd.read_excel("提取数据.xlsx")

  52. # 获取所有不同的管线号
  53. pipeline_ids = df_merge["管线号"].unique()

  54. # 定义模板文件路径和目标文件夹路径,这里的生成文件路径还是只认盘符,比较讨厌
  55. template_file = "工艺卡模板.xlsx"
  56. target_folder = "C:\\测试文件夹"

  57. # 确保目标文件夹存在,如果不存在盘符回去生成?
  58. if not os.path.exists(target_folder):
  59.           os.makedirs(target_folder)

  60. # 遍历所有管线号,复制模板文件并重命名
  61. for pipeline_number in pipeline_ids:
  62.           copy_template_to_new_file(pipeline_number, template_file, target_folder)

  63. print("所有管线号对应的文件已成功创建!")
复制代码


不起作用呢,还是没图片的
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2024-7-27 10:45:43 | 显示全部楼层
颂风侯 发表于 2024-07-27 10:45
import pandas as pd
import openpyxl
from openpyx ...


哎呀,看来我帮你钻了个牛角尖!看起来我们需要更仔细地处理一下图片的复制工作。您现在的代码中的一些小问题可能会导致图像未能成功复制。

下面是一个改进后的代码版本,确保从源工作表中正确地提取并添加图片到新工作表中。

  1. import pandas as pd
  2. import openpyxl
  3. from openpyxl import load_workbook
  4. from openpyxl.drawing.image import Image
  5. import os
  6. import shutil

  7. def copy_template_to_new_file(pipeline_number_copy, template_path_copy, output_folder_copy):
  8.     """
  9.     :param pipeline_number_copy: 复制的管线号
  10.     :param template_path_copy: 模板文件路径
  11.     :param output_folder_copy: 输出文件路径
  12.     :return:
  13.     """
  14.     # 构建新的文件名
  15.     new_file = f"{pipeline_number_copy}.xlsx"
  16.     # 构建新文件的完整路径
  17.     new_file_path = os.path.join(output_folder_copy, new_file)

  18.     # 复制模板文件到新文件路径
  19.     shutil.copy(template_path_copy, new_file_path)
  20.    
  21.     # 加载源文件和新文件
  22.     workbook_source = load_workbook(template_path_copy)
  23.     workbook_new = load_workbook(new_file_path)

  24.     # 复制每个 sheet 的图片
  25.     for sheet_name in workbook_source.sheetnames:
  26.         source_sheet = workbook_source[sheet_name]
  27.         new_sheet = workbook_new[sheet_name]

  28.         # 提取图片
  29.         for img in source_sheet._images:  
  30.             new_img = Image(img.ref)
  31.             # 设置图片位置
  32.             new_img.anchor = img.anchor
  33.             new_sheet.add_image(new_img)  # 添加到新表中

  34.     # 筛选出管线号对应的行
  35.     filtered_rows = df_merge[df_merge == pipeline_number_copy]
  36.     processcards = filtered_rows.unique()

  37.     # 获取所有sheet名
  38.     sheet_names = workbook_new.sheetnames

  39.     # 遍历所有sheet名,检查并删除不需要的sheet
  40.     for sheet_name in sheet_names:
  41.         if 'H' in sheet_name and sheet_name not in processcards:
  42.             workbook_new.remove(workbook_new[sheet_name])

  43.     # 保存修改后的Excel文件
  44.     workbook_new.save(new_file_path)
  45.     print(f"已创建并保存文件:{new_file_path}")

  46. # 读取原始数据文件
  47. df_merge = pd.read_excel("提取数据.xlsx")

  48. # 获取所有不同的管线号
  49. pipeline_ids = df_merge.unique()

  50. # 定义模板文件路径和目标文件夹路径
  51. template_file = "工艺卡模板.xlsx"
  52. target_folder = "C:\\测试文件夹"

  53. # 确保目标文件夹存在
  54. if not os.path.exists(target_folder):
  55.     os.makedirs(target_folder)

  56. # 遍历所有管线号,复制模板文件并重命名
  57. for pipeline_number in pipeline_ids:
  58.     copy_template_to_new_file(pipeline_number, template_file, target_folder)

  59. print("所有管线号对应的文件已成功创建!")
复制代码


### 关键修改:
1. **提取和设置图片的位置**:通过对 `new_img.anchor` 进行赋值,将原图的位置设置到新图像的位置。这样保证了图像在新 Excel 文件中的位置与原位置一致。
2. **针对每个工作表逐个复制图像**:确保在遍历源工作表时明确指定新工作表。

请尝试运行这段代码,看看是否能成功复制图像。如果还有其他问题,随时来问我哦!

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2024-7-27 11:02:14 | 显示全部楼层
照着做了,不出图片的
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2024-7-27 11:02:32 | 显示全部楼层
颂风侯 发表于 2024-07-27 11:02
照着做了,不出图片的


哎呀,看来我们的代码依然没能让图像在 Excel 中“重现江湖”!让我再仔细看看问题。其实我们需要确保从源工作表中获取到的图片能够正确地添加到新的工作表中。

这里有一个更详细的示例,包括如何从每个工作表提取图像并重新插入到目标工作簿中。

请尝试下面的代码:

  1. import pandas as pd
  2. import openpyxl
  3. from openpyxl import load_workbook
  4. from openpyxl.drawing.image import Image
  5. import os
  6. import shutil

  7. def copy_template_to_new_file(pipeline_number_copy, template_path_copy, output_folder_copy):
  8.     # 构建新的文件名
  9.     new_file = f"{pipeline_number_copy}.xlsx"
  10.     # 构建新文件的完整路径
  11.     new_file_path = os.path.join(output_folder_copy, new_file)

  12.     # 复制模板文件到新文件路径
  13.     shutil.copy(template_path_copy, new_file_path)

  14.     # 加载源文件和新文件
  15.     workbook_source = load_workbook(template_path_copy)
  16.     workbook_new = load_workbook(new_file_path)

  17.     # 复制每个 sheet 的图片
  18.     for sheet_name in workbook_source.sheetnames:
  19.         source_sheet = workbook_source[sheet_name]
  20.         new_sheet = workbook_new[sheet_name]

  21.         # 提取图片并添加到新 sheet 中
  22.         for img in source_sheet._images:  
  23.             new_img = Image(img.ref)
  24.             new_img.anchor = img.anchor  # 保持原图的位置
  25.             new_sheet.add_image(new_img)  # 添加到新表中

  26.     # 筛选出管线号对应的行
  27.     filtered_rows = df_merge[df_merge == pipeline_number_copy]
  28.     processcards = filtered_rows.unique()

  29.     # 获取所有sheet名
  30.     sheet_names = workbook_new.sheetnames

  31.     # 遍历所有sheet名,检查并删除不需要的sheet
  32.     for sheet_name in sheet_names:
  33.         if 'H' in sheet_name and sheet_name not in processcards:
  34.             std = workbook_new[sheet_name]  # 获取具体的sheet
  35.             workbook_new.remove(std)  # 删除sheet

  36.     # 保存修改后的Excel文件
  37.     workbook_new.save(new_file_path)
  38.     print(f"已创建并保存文件:{new_file_path}")

  39. # 读取原始数据文件
  40. df_merge = pd.read_excel("提取数据.xlsx")

  41. # 获取所有不同的管线号
  42. pipeline_ids = df_merge['your_column_name'].unique()  # 请确保替换为实际的列名

  43. # 定义模板文件路径和目标文件夹路径
  44. template_file = "工艺卡模板.xlsx"
  45. target_folder = "C:\\测试文件夹"

  46. # 确保目标文件夹存在
  47. if not os.path.exists(target_folder):
  48.     os.makedirs(target_folder)

  49. # 遍历所有管线号,复制模板文件并重命名
  50. for pipeline_number in pipeline_ids:
  51.     copy_template_to_new_file(pipeline_number, template_file, target_folder)

  52. print("所有管线号对应的文件已成功创建!")
复制代码


### 关键点:
1. **`source_sheet = workbook_source[sheet_name]` 和 `new_sheet = workbook_new[sheet_name]`**:确保我们是以相同的工作表名称获取源和新工作表。
   
2. **图片位置**:我还是保留了设置 `anchor` 的逻辑,这确保图片在新工作表中的位置与源工作表一致。

3. **合并数据**:确认了从 `df_merge` 中提取管线号时,你需要确保使用实际存在的列名替换 `'your_column_name'`。

请运行这个版本的代码,并告诉我结果如何!如果仍然不能显示图片,我们可能需要对你的环境做进一步的调查,比如 Excel 文件的格式或 openpyxl 版本等。

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2024-7-27 12:44:35 | 显示全部楼层
仍然不能取得图片
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2024-7-27 12:44:46 | 显示全部楼层
颂风侯 发表于 2024-07-27 12:44
仍然不能取得图片


哎呀,看来我们真的遇到了一些麻烦,让我再仔细检查一下!首先,我们要确保 `openpyxl` 可以正确处理 Excel 文件中的图片。以下是一个更详细和可靠的示例代码,以确保图像能被正确提取和添加到新的 Excel 文件中。我们会更显式地处理源工作表和目标工作表之间的关系。

请确认你已经安装了最新版本的 `openpyxl`,因为处理图片的能力在一些版本中可能会有所不同。可以使用下面的命令进行更新:

  1. pip install --upgrade openpyxl
复制代码


请尝试以下修改后的代码:

  1. import pandas as pd
  2. import openpyxl
  3. from openpyxl import load_workbook
  4. from openpyxl.drawing.image import Image
  5. import os
  6. import shutil

  7. def copy_template_to_new_file(pipeline_number_copy, template_path_copy, output_folder_copy):
  8.     # 构建新文件名和路径
  9.     new_file = f"{pipeline_number_copy}.xlsx"
  10.     new_file_path = os.path.join(output_folder_copy, new_file)
  11.    
  12.     # 复制模板文件
  13.     shutil.copy(template_path_copy, new_file_path)

  14.     # 加载源文件和新文件
  15.     workbook_source = load_workbook(template_path_copy)
  16.     workbook_new = load_workbook(new_file_path)

  17.     for sheet_name in workbook_source.sheetnames:
  18.         source_sheet = workbook_source[sheet_name]
  19.         new_sheet = workbook_new.create_sheet(title=sheet_name)

  20.         # 复制数据(如果需要)
  21.         for row in source_sheet.iter_rows(values_only=True):
  22.             new_sheet.append(row)

  23.         # 复制图片
  24.         for img in source_sheet._images:
  25.             new_img = Image(img.ref)
  26.             new_img.anchor = img.anchor  # 保持原位置
  27.             new_sheet.add_image(new_img)  # 添加图片到新工作表

  28.     # 筛选出管线号对应的行
  29.     filtered_rows = df_merge[df_merge == pipeline_number_copy]
  30.     processcards = filtered_rows.unique()

  31.     # 删除不需要的工作表
  32.     sheet_names = workbook_new.sheetnames
  33.     for sheet_name in sheet_names:
  34.         if 'H' in sheet_name and sheet_name not in processcards:
  35.             std = workbook_new[sheet_name]  # 获取具体的sheet
  36.             workbook_new.remove(std)  # 删除sheet

  37.     # 保存修改后的文件
  38.     workbook_new.save(new_file_path)
  39.     print(f"已创建并保存文件:{new_file_path}")

  40. # 读取数据
  41. df_merge = pd.read_excel("提取数据.xlsx")

  42. # 获取所有管线号
  43. pipeline_ids = df_merge.unique()

  44. # 定义模板文件路径和输出文件夹
  45. template_file = "工艺卡模板.xlsx"
  46. target_folder = "C:\\测试文件夹"

  47. # 确保输出文件夹存在
  48. if not os.path.exists(target_folder):
  49.     os.makedirs(target_folder)

  50. # 遍历管线号,复制模板文件
  51. for pipeline_number in pipeline_ids:
  52.     copy_template_to_new_file(pipeline_number, template_file, target_folder)

  53. print("所有管线号对应的文件已成功创建!")
复制代码


### 关键点:

1. **明确复制数据**:在创建新工作表时,如果需要,不妨将源工作表中的数据也逐行复制到新的工作表中,这样可以确保内容完整。

2. **图片复制**:确保在循环中使用 `for img in source_sheet._images:` 来提取和添加图片。

3. **创建新工作表**:在目标工作簿中,为每个源工作表创建一个新的工作表,并将其命名为相同的名称。

4. **注意 Excel 格式**:确保源文件 (`工艺卡模板.xlsx`) 是 `.xlsx` 格式,而不是 `.xls` 等旧格式,因为 `openpyxl` 不支持旧格式。

如果您仍然无法看到图片,可以尝试用其他 Excel 文件查看或测试此代码,看看问题是否出在特定文件上。如果有其他错误信息,请提供给我,我们一起解决!统计数据总是有帮助的!

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-4-22 01:01

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表