Python图像处理库PIL中图像格式转换(一)

来源:互联网 发布:微博登录不上网络异常 编辑:程序博客网 时间:2024/04/26 02:09

在数字图像处理中,针对不同的图像格式有其特定的处理算法。所以,在做图像处理之前,我们需要考虑清楚自己要基于哪种格式的图像进行算法设计及其实现。本文基于这个需求,使用Python中的图像处理库PIL来实现不同图像格式的转换。

对于彩色图像,不管其图像格式是PNG,还是BMP,或者JPG,在PIL中,使用Image模块的open()函数打开后,返回的图像对象的模式都是“RGB”。而对于灰度图像,不管其图像格式是PNG,还是BMP,或者JPG,打开后,其模式为“L”。

通过之前的博客对Image模块的介绍,对于PNGBMPJPG彩色图像格式之间的互相转换都可以通过Image模块的open()save()函数来完成。具体说就是,在打开这些图像时,PIL会将它们解码为三通道的“RGB”图像。用户可以基于这个“RGB”图像,对其进行处理。处理完毕,使用函数save(),可以将处理结果保存成PNGBMPJPG中任何格式。这样也就完成了几种格式之间的转换。同理,其他格式的彩色图像也可以通过这种方式完成转换。当然,对于不同格式的灰度图像,也可通过类似途径完成,只是PIL解码后是模式为“L”的图像。

这里,我想详细介绍一下Image模块的convert()函数,用于不同模式图像之间的转换。

Convert()函数有三种形式的定义,它们定义形式如下:

im.convert(mode) image

im.convert(“P”, **options) image

im.convert(mode, matrix) image

使用不同的参数,将当前的图像转换为新的模式,并产生新的图像作为返回值。

通过博客“Python图像处理库PIL的基本概念介绍”,我们知道PIL中有九种不同模式。分别为1LPRGBRGBACMYKYCbCrIF

本文我采用的示例图像是图像处理中经典的lena照片。分辨率为512x512lena图片如下:

一、模式“RGB”转换为其他不同模式

1、 模式“1

模式“1”为二值图像,非黑即白。但是它每个像素用8bit表示,0表示黑,255表示白。下面我们将lena图像转换为“1”图像。

例子:

[python] view plain copy 在CODE上查看代码片派生到我的代码片
  1. >>>from PIL import Image  
  2.   
  3. >>> lena =Image.open("D:\\Code\\Python\\test\\img\\lena.jpg")  
  4.   
  5. >>> lena.mode  
  6.   
  7. 'RGB'  
  8.   
  9. >>> lena.getpixel((0,0))  
  10.   
  11. (19711178)  
  12.   
  13. >>> lena_1 = lena.convert("1")  
  14.   
  15. >>> lena_1.mode  
  16.   
  17. '1'  
  18.   
  19. >>> lena_1.size  
  20.   
  21. (512512)  
  22.   
  23. >>>lena_1.getpixel((0,0))  
  24.   
  25. 255  
  26.   
  27. >>> lena_1.getpixel((10,10))  
  28.   
  29. 255  
  30.   
  31. >>>lena_1.getpixel((10,120))  
  32.   
  33. 0  
  34.   
  35. >>>lena_1.getpixel((130,120))  
  36.   
  37. 255  

 

图像lena_1的模式为“1”,分辨率为512x512,如下:

2、 模式“L

模式“L”为灰色图像,它的每个像素用8bit表示,0表示黑,255表示白,其他数字表示不同的灰度。在PIL中,从模式“RGB”转换为“L”模式是按照下面的公式转换的:

L = R * 299/1000 + G * 587/1000+ B * 114/1000

下面我们将lena图像转换为“L”图像。

例子:

[python] view plain copy 在CODE上查看代码片派生到我的代码片
  1. >>> from PIL importImage  
  2.   
  3. >>> lena = Image.open("D:\\Code\\Python\\test\\img\\lena.jpg")  
  4.   
  5. >>> lena.mode  
  6.   
  7. 'RGB'  
  8.   
  9. >>> lena.getpixel((0,0))  
  10.   
  11. (19711178)  
  12.   
  13. >>> lena_L =lena.convert("L")  
  14.   
  15. >>> lena_L.mode  
  16.   
  17. 'L'  
  18.   
  19. >>> lena_L.size  
  20.   
  21. (512512)  
  22.   
  23. >>>lena.getpixel((0,0))  
  24.   
  25. (19711178)  
  26.   
  27. >>>lena_L.getpixel((0,0))  
  28.   
  29. 132  

对于第一个像素点,原始图像lena(197, 111, 78),其转换为灰色值为:

197 *299/1000 + 111 * 587/1000 + 78 * 114/1000 = 132.952PIL中只取了整数部分,即为132

转换后的图像lena_L如下:

3、 模式“P

模式“P”为8位彩色图像,它的每个像素用8bit表示,其对应的彩色值是按照调色板查询出来的。

下面我们使用默认的调色板将lena图像转换为“P”图像。

例子:

[python] view plain copy 在CODE上查看代码片派生到我的代码片
  1. >>> from PIL importImage  
  2.   
  3. >>> lena = Image.open("D:\\Code\\Python\\test\\img\\lena.jpg")  
  4.   
  5. >>> lena.mode  
  6.   
  7. 'RGB'  
  8.   
  9. >>> lena.getpixel((0,0))  
  10.   
  11. (19711178)  
  12.   
  13. >>> lena_P =lena.convert("P")  
  14.   
  15. >>> lena_P.mode  
  16.   
  17. 'P'  
  18.   
  19. >>>lena_P.getpixel((0,0))  
  20.   
  21. 62  

 

转换后的图像lena_P如下:

4、 模式“RGBA

模式“RGBA”为32位彩色图像,它的每个像素用32bit表示,其中24bit表示红色、绿色和蓝色三个通道,另外8bit表示alpha通道,即透明通道。

下面我们将模式为“RGB”的lena图像转换为“RGBA”图像。

例子: 

[python] view plain copy 在CODE上查看代码片派生到我的代码片
  1.                                                                                                                                 
  2.   
  3. >>> from PIL import Image  
  4.   
  5. >>>lena = Image.open("D:\\Code\\Python\\test\\img\\lena.jpg")  
  6.   
  7. >>>lena.mode  
  8.   
  9. 'RGB'  
  10.   
  11. >>>lena.getpixel((0,0))  
  12.   
  13. (197,11178)  
  14.   
  15. >>>lena_rgba = lena.convert("RGBA")  
  16.   
  17. >>>lena_rgba.mode  
  18.   
  19. 'RGBA'  
  20.   
  21. >>>lena_rgba.getpixel((0,0))  
  22.   
  23. (197,11178255)  
  24.   
  25. >>>lena_rgba.getpixel((0,1))  
  26.   
  27. (196,11077255)  
  28.   
  29. >>>lena.getpixel((0,0))  
  30.   
  31. (197,11178)  
  32.   
  33. >>>lena.getpixel((0,1))  
  34.   
  35. (196,11077)  

 

从实例中可以看到,使用当前这个方式将“RGB”图像转为“RGBA”图像时,alpha通道全部设置为255,即完全不透明。

转换后的图像lena_rgba如下:

5、 模式“CMYK

模式“CMYK”为32位彩色图像,它的每个像素用32bit表示。模式CMYK”就是印刷四分色模式,它是彩色印刷时采用的一种套色模式,利用色料的三原色混色原理,加上黑色油墨,共计四种颜色混合叠加,形成所谓“全彩印刷”。

四种标准颜色是:CCyan =青色,又称为‘天蓝色’或是‘湛蓝’MMagenta =品红色,又称为‘洋红色’;YYellow =黄色;KKey Plate(blacK) =定位套版色(黑色)。

下面我们将模式为“RGB”的lena图像转换为“CMYK”图像。

例子:

[python] view plain copy 在CODE上查看代码片派生到我的代码片
  1. >>>from PIL import Image  
  2.   
  3. >>> lena =Image.open("D:\\Code\\Python\\test\\img\\lena.jpg")  
  4.   
  5. >>> lena_cmyk =lena.convert("CMYK")  
  6.   
  7. >>> lena_cmyk.mode  
  8.   
  9. 'CMYK'  
  10.   
  11. >>>lena_cmyk.getpixel((0,0))  
  12.   
  13. (581441770)  
  14.   
  15. >>> lena_cmyk.getpixel((0,1))  
  16.   
  17. (591451780)  
  18.   
  19. >>>lena.getpixel((0,0))  
  20.   
  21. (19711178)  
  22.   
  23. >>>lena.getpixel((0,1))  
  24.   
  25. (19611077)  

 

从实例中可以得知PIL中“RGB”转换为“CMYK”的公式如下:

C = 255 - R
M = 255 - G
Y = 255 - B
K = 0

由于该转换公式比较简单,转换后的图像颜色有些失真。

转换后的图像lena_cmyk如下:

6、 模式“YCbCr

模式“YCbCr”为24位彩色图像,它的每个像素用24bit表示。YCbCr其中Y是指亮度分量,Cb指蓝色色度分量,而Cr指红色色度分量。人的肉眼对视频的Y分量更敏感,因此在通过对色度分量进行子采样来减少色度分量后,肉眼将察觉不到的图像质量的变化。

模式“RGB”转换为“YCbCr”的公式如下:

Y= 0.257*R+0.504*G+0.098*B+16
Cb = -0.148*R-0.291*G+0.439*B+128
Cr = 0.439*R-0.368*G-0.071*B+128

下面我们将模式为“RGB”的lena图像转换为“YCbCr”图像。

例子:

[python] view plain copy 在CODE上查看代码片派生到我的代码片
  1. >>>from PIL import Image  
  2.   
  3. >>> lena =Image.open("D:\\Code\\Python\\test\\img\\lena.jpg")  
  4.   
  5. >>> lena_ycbcr =lena.convert("YCbCr")  
  6.   
  7. >>>lena_ycbcr.mode  
  8.   
  9. 'YCbCr'  
  10.   
  11. >>>lena_ycbcr.getpixel((0,0))  
  12.   
  13. (13297173)  
  14.   
  15. >>>lena.getpixel((0,0))  
  16.   
  17. (19711178)  

 

按照公式,Y =0.257*197+0.564*111+0.098*78+16= 136.877

Cb= -0.148*197-0.291*111+0.439*78+128= 100.785
Cr = 0.439*197-0.368*111-0.071*78+128 = 168.097

由此可见,PIL中并非按照这个公式进行“RGB”到“YCbCr”的转换。

转换后的图像lena_ycbcr如下:

7、 模式“I

模式“I”为32位整型灰色图像,它的每个像素用32bit表示,0表示黑,255表示白,(0,255)之间的数字表示不同的灰度。在PIL中,从模式“RGB”转换为“I”模式是按照下面的公式转换的:

I = R * 299/1000 + G * 587/1000 + B * 114/1000

下面我们将模式为“RGB”的lena图像转换为“I”图像。

例子:

[python] view plain copy 在CODE上查看代码片派生到我的代码片
  1. >>> from PIL import Image  
  2.   
  3. >>>lena = Image.open("D:\\Code\\Python\\test\\img\\lena.jpg")  
  4.   
  5. >>>lena.getpixel((0,0))  
  6.   
  7. (197,11178)  
  8.   
  9. >>>lena.getpixel((0,1))  
  10.   
  11. (196,11077)  
  12.   
  13. >>> lena_I =lena.convert("I")  
  14.   
  15. >>> lena_I.mode  
  16.   
  17. 'I'  
  18.   
  19. >>>lena_I.getpixel((0,0))  
  20.   
  21. 132  
  22.   
  23. >>>lena_I.getpixel((0,1))  
  24.   
  25. 131  
  26.   
  27. >>> lena_L =lena.convert("L")  
  28.   
  29. >>>lena_L.getpixel((0,0))  
  30.   
  31. 132  
  32.   
  33. >>>lena_L.getpixel((0,1))  
  34.   
  35. 131  

 

从实验的结果看,模式“I”与模式“L”的结果是完全一样,只是模式“L”的像素是8bit,而模式“I”的像素是32bit

8、 模式“F

模式“F”为32位浮点灰色图像,它的每个像素用32bit表示,0表示黑,255表示白,(0,255)之间的数字表示不同的灰度。在PIL中,从模式“RGB”转换为“F”模式是按照下面的公式转换的:

F = R * 299/1000+ G * 587/1000 + B * 114/1000

下面我们将模式为“RGB”的lena图像转换为“F”图像。

例子:

[python] view plain copy 在CODE上查看代码片派生到我的代码片
  1. >>>from PIL import Image  
  2.   
  3. >>> lena =Image.open("D:\\Code\\Python\\test\\img\\lena.jpg")  
  4.   
  5. >>>lena.getpixel((0,0))  
  6.   
  7. (19711178)  
  8.   
  9. >>>lena.getpixel((0,1))  
  10.   
  11. (19611077)  
  12.   
  13. >>> lena_F =lena.convert("F")  
  14.   
  15. >>> lena_F.mode  
  16.   
  17. 'F'  
  18.   
  19. >>>lena_F.getpixel((0,0))  
  20.   
  21. 132.95199584960938  
  22.   
  23. >>>lena_F.getpixel((0,1))  
  24.   
  25. 131.95199584960938  

 

模式“F”与模式“L”的转换公式是一样的,都是RGB转换为灰色值的公式,但模式“F”会保留小数部分,如实验中的数据。

(未完待续)

0 0