如何为普通pgm格式编写PIL图像过滤器? [英] How to write PIL image filter for plain pgm format?
问题描述
如何为pgm plain ascii format(P2)编写python映像库的过滤器。这里的问题是基本的PIL过滤器假设每个像素的字节数是恒定的。
How can I write a filter for python imaging library for pgm plain ascii format (P2). Problem here is that basic PIL filter assumes constant number of bytes per pixel.
我的目标是使用Image.open()打开feep.pgm。请参阅 http://netpbm.sourceforge.net/doc/pgm.html 或以下。
My goal is to open feep.pgm with Image.open(). See http://netpbm.sourceforge.net/doc/pgm.html or below.
替代解决方案是我找到了PIL和所有主要图形程序支持的其他记录良好的ascii灰度格式。有什么建议吗?
Alternative solution is that I find other well documented ascii grayscale format that is supported by PIL and all major graphics programs. Any suggestions?
feep.pgm:
P2
# feep.pgm
24 7
15
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
0 3 3 3 3 0 0 7 7 7 7 0 0 11 11 11 11 0 0 15 15 15 15 0
0 3 0 0 0 0 0 7 0 0 0 0 0 11 0 0 0 0 0 15 0 0 15 0
0 3 3 3 0 0 0 7 7 7 0 0 0 11 11 11 0 0 0 15 15 15 15 0
0 3 0 0 0 0 0 7 0 0 0 0 0 11 0 0 0 0 0 15 0 0 0 0
0 3 0 0 0 0 0 7 7 7 7 0 0 11 11 11 11 0 0 15 0 0 0 0
0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
编辑谢谢你的答案,它有效... 但我需要一个使用Image.open()的解决方案。大多数python程序都使用PIL进行图形处理(谷歌:python图像打开)。因此,我需要能够将过滤器注册到PIL。然后,我可以使用任何使用PIL的软件。我现在认为主要是scipy,pylab等依赖程序。
edit: Thanks for the answer, It works... but I need a solution that uses Image.open(). Most of python programs out there use PIL for graphics manipulation (google: python image open). Thus, I need to be able to register a filter to PIL. Then, I can use any software that uses PIL. I now think mostly scipy, pylab, etc. dependent programs.
编辑好的,我想我现在知道了。下面是包装器pgm2pil.py:
edit Ok, I think I got it now. Below is the wrapper pgm2pil.py:
import Image
import numpy
def pgm2pil(filename):
try:
inFile = open(filename)
header = None
size = None
maxGray = None
data = []
for line in inFile:
stripped = line.strip()
if stripped[0] == '#':
continue
elif header == None:
if stripped != 'P2': return None
header = stripped
elif size == None:
size = map(int, stripped.split())
elif maxGray == None:
maxGray = int(stripped)
else:
for item in stripped.split():
data.append(int(item.strip()))
data = numpy.reshape(data, (size[1],size[0]))/float(maxGray)*255
return numpy.flipud(data)
except:
pass
return None
def imageOpenWrapper(fname):
pgm = pgm2pil(fname)
if pgm is not None:
return Image.fromarray(pgm)
return origImageOpen(fname)
origImageOpen = Image.open
Image.open = imageOpenWrapper
misha的答案略有升级。必须保存Image.open以防止永不结束循环。如果pgm2pil返回无包装调用pgm2pil,它返回None,调用pgm2pil ...
There is a slight upgrade to misha's answer. Image.open has to be saved in order to prevent never ending loops. If pgm2pil returns None wrapper calls pgm2pil which returns None which calls pgm2pil...
下面是测试函数(feep_false.pgm是格式错误的pgm,例如P2 - > FOO和lena.pgm只是 图像文件):
Below is the test function (feep_false.pgm is a malformed pgm e.g. "P2" -> "FOO" and lena.pgm is just the image file):
import pgm2pil
import pylab
try:
pylab.imread('feep_false.pgm')
except IOError:
pass
else:
raise ValueError("feep_false should fail")
pylab.subplot(2,1,1)
a = pylab.imread('feep.pgm')
pylab.imshow(a)
pylab.subplot(2,1,2)
b = pylab.imread('lena.png')
pylab.imshow(b)
pylab.show()
推荐答案
我目前的方式处理这个问题是通过 numpy :
The way I currently deal with this is through numpy:
- 将图像读入2D
numpy
数组。你不需要使用numpy
,但我发现它比普通的Python 2D数组更容易使用 - 使用
PIL.Image.fromarray
将2D numpy数组转换为
PIL.Image
对象- Read image into a 2D
numpy
array. You don't need to usenumpy
, but I've found it easier to use than the regular Python 2D arrays - Convert 2D numpy array into
PIL.Image
object usingPIL.Image.fromarray
如果你坚持使用 PIL.Image.open
,你可以写一个尝试的包装器首先加载PGM文件(通过查看标题)。如果它是PGM,请使用上述步骤加载图像,否则只需将责任移交给 PIL.Image.open
。
If you insist on using PIL.Image.open
, you could write a wrapper that attempts to load a PGM file first (by looking at the header). If it's a PGM, load the image using the steps above, otherwise just hands off responsibility to PIL.Image.open
.
以下是我用来将 PBM 图片转换为 numpy 的一些代码数组。
Here's some code that I use to get a PBM image into a numpy array.
import re
import numpy
def pbm2numpy(filename):
"""
Read a PBM into a numpy array. Only supports ASCII PBM for now.
"""
fin = None
debug = True
try:
fin = open(filename, 'r')
while True:
header = fin.readline().strip()
if header.startswith('#'):
continue
elif header == 'P1':
break
elif header == 'P4':
assert False, 'Raw PBM reading not implemented yet'
else:
#
# Unexpected header.
#
if debug:
print 'Bad mode:', header
return None
rows, cols = 0, 0
while True:
header = fin.readline().strip()
if header.startswith('#'):
continue
match = re.match('^(\d+) (\d+)$', header)
if match == None:
if debug:
print 'Bad size:', repr(header)
return None
cols, rows = match.groups()
break
rows = int(rows)
cols = int(cols)
assert (rows, cols) != (0, 0)
if debug:
print 'Rows: %d, cols: %d' % (rows, cols)
#
# Initialise a 2D numpy array
#
result = numpy.zeros((rows, cols), numpy.int8)
pxs = []
#
# Read to EOF.
#
while True:
line = fin.readline().strip()
if line == '':
break
for c in line:
if c == ' ':
continue
pxs.append(int(c))
if len(pxs) != rows*cols:
if debug:
print 'Insufficient image data:', len(pxs)
return None
for r in range(rows):
for c in range(cols):
#
# Index into the numpy array and set the pixel value.
#
result[r, c] = pxs[r*cols + c]
return result
finally:
if fin != None:
fin.close()
fin = None
return None
您必须稍微修改它以符合您的目的,即:
You will have to modify it slightly to fit your purposes, namely:
- 处理P2(ASCII ,greyscale)而不是P1(ASCII,bilevel)。
- 如果你没有使用numpy,请使用不同的容器。普通的Python 2D数组工作正常。
编辑
以下是我处理包装器的方法:
Here is how I would handle a wrapper:
def pgm2pil(fname):
#
# This method returns a PIL.Image. Use pbm2numpy function above as a
# guide. If it can't load the image, it returns None.
#
pass
def wrapper(fname):
pgm = pgm2pil(fname)
if pgm is not None:
return pgm
return PIL.Image.open(fname)
#
# This is the line that "adds" the wrapper
#
PIL.Image.open = wrapper
我没有写 pgm2pil
因为它与 pgm2numpy
非常相似。唯一的区别是它将结果存储在 PIL.Image
中,而不是 numpy
数组。我也没有测试包装器代码(抱歉,目前时间有点短),但这是一种相当常见的方法,所以我希望它可以工作。
I didn't write pgm2pil
because it's going to be very similar to pgm2numpy
. The only difference will be that it's storing the result in a PIL.Image
as opposed to a numpy
array. I also didn't test the wrapper code (sorry, a bit short on time at the moment) but it's a fairly common approach so I expect it to work.
现在,听起来你想要使用PIL进行图像加载的其他应用程序能够处理PGM。可以使用上面的方法,但你需要确保上面的包装代码在第一次调用 PIL.Image.open 之前添加 。您可以通过将包装器源代码添加到PIL源代码(如果您有访问权限)来确保这一点。
Now, it sounds like you want other applications that use PIL for image loading to be able to handle PGMs. It's possible using the above approach, but you need to be sure that the above wrapper code gets added before the first call to PIL.Image.open
. You can make sure that happens by adding the wrapper source code to the PIL source code (if you have access).
这篇关于如何为普通pgm格式编写PIL图像过滤器?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!