关于libjpeg修改图片质量的讨论

大多的开源或不开源的软件处理jpg图像的时候均使用libjpeg开源库,目前最新版本为libjpeg-8b,下载链接为

http://freshmeat.net/projects/libjpeg


里面很多makefile文件,linux下不多说,在win32下,构建vc6工程只要将.vc6后缀名搜索出,将make*dsp.vc6修改为make*dsp.dsp,所有make*dsw.vc6修改为make*dsw.dsw,这里得到makeadsw.dsw和makejdsw.dsw,前者为所有编解码及测试程序工程,后者为简单的libjpeg工程。对于libjpeg的makejdsw生成了一个win32的的.lib库,这里可将库名称修改为libjpeg.lib。这样我们可参考编码程序将图片编码为jpeg图片了。

比如,现在我们利用GDI+打开任意格式图片,而后得到图片的解码数据,便可编译为jpeg图片了。

ps:为何不用GDI+直接保存jpg图片,因GDI+中生成的jpeg质量不满意,嘿嘿

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
// jpeg
void write_my_jpeg()
{
    // 初始化数据
    Bitmap bitmap(_T("a.bmp"));
    int iWidth = bitmap.GetWidth();
    int iHeight = bitmap.GetHeight();
 
    unsigned char *pDataConv = new unsigned char[iWidth * iHeight * 3];
 
    init_data(bitmap, pDataConv);
 
    struct jpeg_compress_struct jcs;
 
    // 声明错误处理器,并赋值给jcs.err域
    struct jpeg_error_mgr jem;
    jcs.err = jpeg_std_error(&jem);
 
    jpeg_create_compress(&jcs);
 
    FILE * f = _wfopen(_T("my.jpg"), _T("wb"));
    if (f==NULL)
    {
        delete [] pDataConv;
        return;
    }
    jpeg_stdio_dest(&jcs, f);
 
 
    jcs.image_width = iWidth;    // 为图的宽和高,单位为像素
    jcs.image_height = iHeight;
    jcs.input_components = 3;   // 在此为1,表示灰度图, 如果是彩色位图,则为3
    jcs.in_color_space = JCS_RGB; //JCS_GRAYSCALE表示灰度图,JCS_RGB表示彩色图像
     
 
    jpeg_set_defaults(&jcs);
 
    // 指定亮度及色度质量
    jcs.q_scale_factor[0] = jpeg_quality_scaling(100);
    jcs.q_scale_factor[1] = jpeg_quality_scaling(100);
 
        // 图像采样率,默认为2 * 2
    jcs.comp_info[0].v_samp_factor = 1;
    jcs.comp_info[0].h_samp_factor = 1;
 
        // 设定编码jpeg质量
    jpeg_set_quality (&jcs, 100, true);
 
    jpeg_start_compress(&jcs, TRUE);
 
    JSAMPROW row_pointer[1];   // 一行位图
    int row_stride;      // 每一行的字节数
 
    row_stride = jcs.image_width * 3; // 如果不是索引图,此处需要乘以3
 
    // 对每一行进行压缩
    while (jcs.next_scanline < jcs.image_height)
       {
        row_pointer[0] = (JSAMPROW)(pDataConv + jcs.next_scanline * row_stride);   
        jpeg_write_scanlines(&jcs, row_pointer, 1);
    }
 
    jpeg_finish_compress(&jcs);
 
    jpeg_destroy_compress(&jcs);
 
    //
    delete [] pDataConv;
}

// 数据转换

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
// data
void init_data(Bitmap &bitmap, unsigned char *pDataConv)
{  
    // Btimap
     
    int iWidth = bitmap.GetWidth();
    int iHeight = bitmap.GetHeight();
    Gdiplus::BitmapData bitmapData;
    Gdiplus::Rect rectPiece(0, 0, iWidth, iHeight);
    // lock,获取Bitmap数据
    bitmap.LockBits(&rectPiece, Gdiplus::ImageLockModeWrite,
        PixelFormat32bppARGB, &bitmapData);
    int iBitmapPieceStride = bitmapData.Stride;
    unsigned long *pBitmapPiecePixels = (unsigned long*)bitmapData.Scan0;
    // 获取像素点数据
    unsigned long ulColorValueTemp = 0;
    unsigned char usR, usG, usB;
 
    for (int y = 0; y != iHeight; ++y)
    {
        for (int x = 0; x != iWidth; ++x)
        {
            ulColorValueTemp = pBitmapPiecePixels[y * iBitmapPieceStride / 4 + x];
 
            usR = (ulColorValueTemp >> 16);
            usG = (ulColorValueTemp >> 8);
            usB = ulColorValueTemp;
 
            *pDataConv = usR;
            ++pDataConv;
            *pDataConv = usG;
            ++pDataConv;
            *pDataConv = usB;
            ++pDataConv;
        }
    }
    //
//  memcpy(pDataConv, pBitmapPiecePixels, iWidth * iHeight * 3);
 
    bitmap.UnlockBits(&bitmapData);
}

默认情况下,不管是GDI+,还是CxImage,OpenCV等常见的开源库均只提供了设定质量的函数,即

1
jpeg_set_quality()

 

正常情况下,这样也可以满足我们的基本需求,但对于有些图片,比如黑色背景,写白色字体的图片,在压缩时,基本设定质量最高为100,但压缩质量仍不满意,原因便在压缩时的图像采样率上,ASDSee保存jpeg图片格式时,有选择采样率的2 * 1和1* 2,而libjpeg默认为2 * 2,因此会发现ACDSee质量要好些的缘故,如果不介意,可以直接设定为1 * 1,质量会更好些,不过对于这样的图片压缩后的图片大小可能大于原bmp图片大小,具体情况请读者自行参考。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值