Python 实现将大图切片成小图,将小图组合成大图的例子

作者:未名亚柳 时间:2023-05-23 18:08:36 

训练keras时遇到了一个问题,就是内存不足,将 .fit 改成 .fit_generator以后还是放不下一张图(我的图片是8192×8192的大图==64M)。于是解决方法是将大图切成小图,把小图扔去训练,跑出来的图再拼成一个大图

实验发现我的keras(win10 - 16G内存)只放得下最多4副小图(2048×2048×4==16M),

再多就会报错exit :

Allocation of 4831838208 exceeds 10% of system memory.

原因大概是除了numpy本身要存这些图,keras训练中也会对应有额外的消耗

一、大图切片成小图


'''
读入一个图片0.bmp,切成指定数目个小图片(16个)
文件夹名out
'''
from PIL import Image
import sys,os
cut_num = 4 # 4*4=16个图片
#将图片填充为正方形
def fill_image(image):
 width, height = image.size  
 #选取长和宽中较大值作为新图片的
 new_image_length = width if width > height else height  
 #生成新图片[白底]
 #new_image = Image.new(image.mode, (new_image_length, new_image_length), color='white')  
 new_image = Image.new(image.mode, (new_image_length, new_image_length))
 #将之前的图粘贴在新图上,居中
 if width > height:#原图宽大于高,则填充图片的竖直维度
 #(x,y)二元组表示粘贴上图相对下图的起始位置
   new_image.paste(image, (0, int((new_image_length - height) / 2)))
 else:
   new_image.paste(image, (int((new_image_length - width) / 2),0))  
 return new_image
#切图
def cut_image(image):
 width, height = image.size
 item_width = int(width / cut_num)
 box_list = []  
 # (left, upper, right, lower)
 for i in range(0,cut_num):#两重循环,生成图片基于原图的位置
   for j in range(0,cut_num):      
     #print((i*item_width,j*item_width,(i+1)*item_width,(j+1)*item_width))
     box = (j*item_width,i*item_width,(j+1)*item_width,(i+1)*item_width)
     box_list.append(box)

image_list = [image.crop(box) for box in box_list]  
 return image_list
#保存
def save_images(image_list):
 index = 1
 for image in image_list:
   image.save('out/'+str(index) + '.bmp', 'BMP')
   index += 1

if __name__ == '__main__':
 file_path = "0.bmp"
 os.mkdir("out")
 image = Image.open(file_path)  
 #image.show()
 image = fill_image(image)
 image_list = cut_image(image)
 save_images(image_list)

二、随机截取指定大小的图


'''
随即截取指定大小的图片
'''
import os
import cv2
import random

#读取图片
img1=cv2.imread('0.bmp')
img2=cv2.imread('1.bmp')

#h、w为想要截取的图片大小
h=2048
w=2048

save_dir1 = "pic_train/"
save_dir2 = "pic_noise/"
if os.path.exists(save_dir1) is False:
 os.makedirs(save_dir1)
if os.path.exists(save_dir2) is False:
 os.makedirs(save_dir2)
count=0
while 1:
 #随机产生x,y 此为像素内范围产生
 y = random.randint(0, 6144)
 x = random.randint(0, 6144)
 #随机截图
 cropImg1 = img1[(y):(y + h), (x):(x + w)]
 cropImg2 = img2[(y):(y + h), (x):(x + w)]
 cv2.imwrite(save_dir1 + str(count) + '.bmp', cropImg1)
 cv2.imwrite(save_dir2 + str(count) + '.bmp', cropImg2)
 count+=1

if count==100:
   break

三、小图组合成大图


'''
将指定文件夹里面的图片拼接成一个大图片
'''
import PIL.Image as Image
import os

IMAGES_PATH = 'out\\' # 图片集地址
IMAGES_FORMAT = ['.bmp', '.BMP'] # 图片格式
IMAGE_SIZE = 2048 # 每张小图片的大小
IMAGE_ROW = 4 # 图片间隔,也就是合并成一张图后,一共有几行
IMAGE_COLUMN = 4 # 图片间隔,也就是合并成一张图后,一共有几列
IMAGE_SAVE_PATH = 'final.bmp' # 图片转换后的地址

# 获取图片集地址下的所有图片名称
image_names = [name for name in os.listdir(IMAGES_PATH) for item in IMAGES_FORMAT if
       os.path.splitext(name)[1] == item]

# 简单的对于参数的设定和实际图片集的大小进行数量判断
if len(image_names) != IMAGE_ROW * IMAGE_COLUMN:
 raise ValueError("合成图片的参数和要求的数量不能匹配!")

# 定义图像拼接函数
def image_compose():
 to_image = Image.new('RGB', (IMAGE_COLUMN * IMAGE_SIZE, IMAGE_ROW * IMAGE_SIZE)) #创建一个新图
 # 循环遍历,把每张图片按顺序粘贴到对应位置上
 for y in range(1, IMAGE_ROW + 1):
   for x in range(1, IMAGE_COLUMN + 1):
     from_image = Image.open(IMAGES_PATH + image_names[IMAGE_COLUMN * (y - 1) + x - 1]).resize(
       (IMAGE_SIZE, IMAGE_SIZE),Image.ANTIALIAS)
     to_image.paste(from_image, ((x - 1) * IMAGE_SIZE, (y - 1) * IMAGE_SIZE))
 to_image = to_image.convert('L')
 return to_image.save(IMAGE_SAVE_PATH) # 保存新图
image_compose() #调用函数

注意文件名的数字顺序,00 01 02 ...11 12 13 ....这样

来源:https://www.cnblogs.com/dzzy/p/11401800.html

标签:Python,大图,切片,小图
0
投稿

猜你喜欢

  • 在SQL Server中使用索引的技巧

    2009-02-24 17:50:00
  • Mysql 建库建表技巧分享

    2024-01-24 02:18:30
  • 在ASP中用FormatDateTime格式化日期

    2010-08-08 19:16:00
  • Python psutil模块简单使用实例

    2023-08-10 21:15:37
  • Vue中的 DOM与Diff详情

    2023-07-02 16:32:37
  • JavaScript拖动层Div代码

    2024-04-16 09:46:48
  • 关于python的对象序列化介绍

    2023-07-27 05:02:31
  • python使用pil生成图片验证码的方法

    2022-03-08 18:39:24
  • SQL Server如何保证可空字段中非空值唯一

    2011-02-24 16:44:00
  • python实现整数序列求和

    2023-12-14 06:53:10
  • Python使用JSON库解析JSON数据的方法

    2021-09-24 13:35:10
  • Python写一个简单的api接口的实现

    2023-07-23 20:20:53
  • asp函数遍历文件夹代码

    2010-06-21 10:38:00
  • 简单的文本内容处理工具

    2010-01-28 12:31:00
  • 20行python代码的入门级小游戏的详解

    2023-07-15 01:25:31
  • 解决MySQl查询不区分大小写的方法讲解

    2024-01-12 13:49:51
  • python模仿网页版微信发送消息功能

    2022-01-15 00:12:54
  • Python学习之文件的创建与写入详解

    2021-03-07 09:27:06
  • JavaScript省市级联下拉菜单实例

    2024-04-10 16:18:18
  • 区别Javascript中的Null与Undefined

    2007-12-13 20:24:00
  • asp之家 网络编程 m.aspxhome.com