2011-05-10 69 views
4

我试图用PIL将照片拆分为多个部分。如何在Python中将图像拆分为多个部分

def crop(Path,input,height,width,i,k,x,y,page): 
    im = Image.open(input) 
    imgwidth = im.size[0] 
    imgheight = im.size[1] 
    for i in range(0,imgheight-height/2,height-2): 
     print i 
     for j in range(0,imgwidth-width/2,width-2): 
      print j 
      box = (j, i, j+width, i+height) 
      a = im.crop(box) 
      a.save(os.path.join(Path,"PNG","%s" % page,"IMG-%s.png" % k)) 
      k +=1 

但它似乎并没有工作。它拆分照片,但不是以一种确切的方式(你可以尝试)。如果从 图像保存 代码中分离出来的 裁剪代码

+0

什么你的意思是“确切”的宽度和高度? – kindall 2011-05-10 16:47:49

+0

您可以使用列表解析,请参阅我的回答下面的 – Nir 2018-02-10 20:02:31

回答

13
from PIL import Image 
import Image 

def crop(Path, input, height, width, k, page, area): 
    im = Image.open(input) 
    imgwidth, imgheight = im.size 
    for i in range(0,imgheight,height): 
     for j in range(0,imgwidth,width): 
      box = (j, i, j+width, i+height) 
      a = im.crop(box) 
      try: 
       o = a.crop(area) 
       o.save(os.path.join(Path,"PNG","%s" % page,"IMG-%s.png" % k)) 
      except: 
       pass 
      k +=1 
+0

给出一个完整的工作示例.. – 2018-02-07 07:09:51

5
  1. crop将是一个更可重复使用的 功能。这也会使呼叫 签名更简单。
  2. im.crop返回一个 Image._ImageCrop实例。这样的 实例没有保存方法。 相反,你必须在 Image._ImageCrop实例粘贴到一个新的 Image.Image
  3. 你的范围内没有合适的 步长。 (为什么在height-2而不是 height

所以,你可以尝试,而不是像这样:

import Image 
import os 

def crop(infile,height,width): 
    im = Image.open(infile) 
    imgwidth, imgheight = im.size 
    for i in range(imgheight//height): 
     for j in range(imgwidth//width): 
      box = (j*width, i*height, (j+1)*width, (i+1)*height) 
      yield im.crop(box) 

if __name__=='__main__': 
    infile=... 
    height=... 
    width=... 
    start_num=... 
    for k,piece in enumerate(crop(infile,height,width),start_num): 
     img=Image.new('RGB', (height,width), 255) 
     img.paste(piece) 
     path=os.path.join('/tmp',"IMG-%s.png" % k) 
     img.save(path) 
+0

感谢您的解决方案,但它不适用于我,图片没有裁剪好,我看到红色,我想这个问题也许在这里:'img.paste piece)' – ElTero 2011-05-10 21:07:41

+0

如果你有内存限制,这是一个特别好的解决方案。使用image_slicer时,大图像可能会在内存不足的机器上失败。 – 2017-01-12 13:54:01

12

简单得多所有这些是使用车轮别人发明了:)这可能是更多地参与建立,但那么这是一个简单的使用。

这些说明适用于Windows 7;他们可能需要适应其他操作系统。

here获取并安装点。

下载安装档案,并将其解压到您的根目录Python安装目录。通过PIP

python get-pip.py install 

然后获取并安装image_slicer模块,通过在控制台输入以下命令:打开一个控制台和类型(如果我没有记错)

python -m pip install image_slicer 

复制想要的图像切片到Python根目录下,打开一个蟒壳(不是“命令行”),并输入以下命令:

import image_slicer 
image_slicer.slice('huge_test_image.png', 14) 

该模块的优点在于它

  1. 安装在蟒
  2. 可以调用用的两行代码的图像分割
  3. 接受任何偶数作为图像切片参数(例如在这个例子中)
  4. 注意到该参数并自动给定的图像分成很多片段,并在同一目录下自动保存生成的编号的块,最后
  5. 具有这样的功能拼接图像块重新走到一起14 (我还没有测试过);文件显然必须按照惯例命名,您将在测试image_slicer.slice函数后在分割文件中看到该约定。您希望将图像分解到四个

    tiles = [im[x:x+M,y:y+N] for x in range(0,im.shape[0],M) for y in range(0,im.shape[1],N)] 
    

    在这种情况下:

+1

它看起来不错,但其文档很差。一旦它们被创建,它也可以很好地控制瓷砖,但不容易看出图像将如何切割。我期待有一种元组来设置行数和列数 – 2017-06-08 14:01:35

2

分割图像M×N个像素的砖(假设IM是numpy.ndarray)

M = im.shape[0]//2 
N = im.shape[1]//2 

砖[0]保留左上方的图块

0

这是我的脚本工具,它很容易将css-sprit图片分割为图标:

Usage: split_icons.py img dst_path width height 
Example: python split_icons.py icon-48.png gtliu 48 48 

保存代码到split_icons.py:

#!/usr/bin/env python 
# -*- coding:utf-8 -*- 
import os 
import sys 
import glob 
from PIL import Image 

def Usage(): 
    print '%s img dst_path width height' % (sys.argv[0]) 
    sys.exit(1) 

if len(sys.argv) != 5: 
    Usage() 

src_img = sys.argv[1] 
dst_path = sys.argv[2] 

if not os.path.exists(sys.argv[2]) or not os.path.isfile(sys.argv[1]): 
    print 'Not exists', sys.argv[2], sys.argv[1] 
    sys.exit(1) 

w, h = int(sys.argv[3]), int(sys.argv[4]) 
im = Image.open(src_img) 
im_w, im_h = im.size 
print 'Image width:%d height:%d will split into (%d %d) ' % (im_w, im_h, w, h) 
w_num, h_num = int(im_w/w), int(im_h/h) 

for wi in range(0, w_num): 
    for hi in range(0, h_num): 
     box = (wi*w, hi*h, (wi+1)*w, (hi+1)*h) 
     piece = im.crop(box) 
     tmp_img = Image.new('L', (w, h), 255) 
     tmp_img.paste(piece) 
     img_path = os.path.join(dst_path, "%d_%d.png" % (wi, hi)) 
     tmp_img.save(img_path) 
相关问题