使用jpeglib库实现bmp转jpg (转)

一、vc编译jpeglib库

1、下载源代码

下载地址:http://www.ijg.org/。注意:一定要下载win32 版本

2、编译源代码.

    A、解压源代码,修改源代码中jconfig.vcjconfig.h

    B、添加环境变量PATHC:/Program Files/Microsoft Visual Studio/VC98/Bin 

    C、修改makefile.vc,将 Pull in standard variable definitions下面的一行换为:!include <C:/Program Files/Microsoft Visual Studio/VC98/Include/win32.mak> 

    D、进入命令提示环境下,输入:vcvars32 回车,这是一个用来设置VC路径环境的批处理;

    E、编译生成库文件 命令:nmake /f makefile.vc nodebug=1

这样就OK

3jpeglibVC下使用

对于库的使用只需要有相应的.lib文件和头文件就可以了。Vc中要添加libjpeg.lib库的连接

         将这几个文件拷贝到你的项目中:libjpeg.libjconfig.hjmorecfg.hjpeglib.h,在你需要进行压缩的文件中加入

extern "C" {

         #include "jpeglib.h"

         #include "jmorecfg.h"

         #include "jconfig.h"

}

参考:

http://blog.csdn.net/xingyu19871124/archive/2009/06/30/4310800.aspx

小知识:bmp文件的前54个字节是头,后面才是像素值。

二、使用jpeg库压缩

在源代码的文件夹下面有一个example.c的文件和libjpeg.txt,很有参考价值。

1、基本思路

首先调用截屏程序,将屏幕的位图信息存放在一个buffer里面,然后调用jpg压缩函数,在当前的目录下生成一个ok.jpg的文件。

2、出现的问题:

A、运行是总是报错:

我参考源代码的例子,也用JSAMPLE * image_buffer;来指向位图的像素的首地址,编译可以通过但是运行时就会报错,后来我用BYTE *image_buffer;来定义就可以正常运行了。

B、生成的jpg图像失真:

由于window的位图的像素格式是:BGRA4个字节,jpeglib库使用的是RGB3个字节的格式,所以需要将源像素去掉其透明字节,同时改变RGB的顺序。代码如下:

//RGB顺序调整

for (int i=0, int j=0; j < 1440*900*4; i+=3, j+=4)

{

    *(image_buffer+i)=*(image_buffer+j+2);

    *(image_buffer+i+1)=*(image_buffer+j+1);

    *(image_buffer+i+2)=*(image_buffer+j);

}

C、生成的jpg文件图像是倒的:

原因没有找到,后来修改了压缩函数的代码,生成了正确的jpg文件

while (cinfo.next_scanline < cinfo.image_height) {

         //这里我做过修改,由于jpg文件的图像是倒的,所以改了一下读的顺序

         //这是原代码:row_pointer[0] = & bits[cinfo.next_scanline * row_stride];

         row_pointer[0] = & bits[(cinfo.image_height - cinfo.next_scanline - 1) * row_stride];

    (void) jpeg_write_scanlines(&cinfo, row_pointer, 1);

}

3、测试结果:

我编写了测试代码,连续截屏并生成jpg文件100次,大约花费7秒左右,也就是说1秒可以截屏13次左右。同时生成的jpg文件有100K的样子。

三、代码:

 

#include "stdafx.h"
#include <atlbase.h>
#include <afxwin.h>
#include <WINDOWSX.H>
 
#define JPEG_QUALITY 50      // 它的大小决定jpg的质量好坏
 
extern "C" {
    #include "jpeglib.h"
    #include "jmorecfg.h"
    #include "jconfig.h"
}
 
int savejpeg( char *filename, unsigned  char *bits,  int width,  int height,  int depth);
void CapScreen( char filename[]);
 
BYTE *image_buffer;  // 指向位图buffer的全局指针,window下像素格式: BGRA(4个字节)
 
int main( int argc,  char* argv[])
{
 
    image_buffer = (BYTE *)malloc(1440 * 900 * 4);
 
     for( int i = 0; i < 100; i++){
        CapScreen("ok.bmp");   
         // RGB顺序调整
         for ( int i=0,  int j=0; j < 1440*900*4; i+=3, j+=4)
        {
            *(image_buffer+i)=*(image_buffer+j+2);
            *(image_buffer+i+1)=*(image_buffer+j+1);
            *(image_buffer+i+2)=*(image_buffer+j);
        }
        savejpeg("ok.jpg", image_buffer, 1440, 900, 3);
    }
 
    delete [] image_buffer;
     return 0;
}
 
 
/* ===================================================================================
function:       jpeg压缩
input:          1:生成的文件名,2:bmp的指针,3:位图宽度,4:位图高度,5:颜色深度
return:         int
description:    bmp的像素格式为(RGB)
===================================================================================
*/
int savejpeg( char *filename, unsigned  char *bits,  int width,  int height,  int depth)
{
         struct jpeg_compress_struct cinfo;
         struct jpeg_error_mgr jerr;
        FILE * outfile;                  /*  target file  */
        JSAMPROW row_pointer[1];         /*  pointer to JSAMPLE row[s]  */
         int     row_stride;              /*  physical row width in image buffer  */
 
        cinfo.err = jpeg_std_error(&jerr);
        jpeg_create_compress(&cinfo);
 
         if ((outfile = fopen(filename, "wb")) == NULL) {
                fprintf(stderr, "can't open %s/n", filename);
                 return -1;
        }
        jpeg_stdio_dest(&cinfo, outfile);
 
        cinfo.image_width = width;       /*  image width and height, in pixels  */
        cinfo.image_height = height;
        cinfo.input_components = 3;          /*  # of color components per pixel  */
        cinfo.in_color_space = JCS_RGB;          /*  colorspace of input image  */
 
        jpeg_set_defaults(&cinfo);
        jpeg_set_quality(&cinfo, JPEG_QUALITY, TRUE  /*  limit to baseline-JPEG values  */);
 
        jpeg_start_compress(&cinfo, TRUE);
 
        row_stride = width * depth;  /*  JSAMPLEs per row in image_buffer  */
 
         while (cinfo.next_scanline < cinfo.image_height) {
             // 这里我做过修改,由于jpg文件的图像是倒的,所以改了一下读的顺序
            
// 这是原代码:row_pointer[0] = & bits[cinfo.next_scanline * row_stride];
            row_pointer[0] = & bits[(cinfo.image_height - cinfo.next_scanline - 1) * row_stride];
            ( void) jpeg_write_scanlines(&cinfo, row_pointer, 1);
        }
 
        jpeg_finish_compress(&cinfo);
        fclose(outfile);
 
        jpeg_destroy_compress(&cinfo);
         return 0;
}
 
 
void CapScreen( char filename[])
{
    CDC *pDC;
    pDC = CDC::FromHandle(GetDC(GetDesktopWindow()));
     if(pDC == NULL)  return;
     int BitPerPixel = pDC->GetDeviceCaps(BITSPIXEL);
     int Width = pDC->GetDeviceCaps(HORZRES);
     int Height = pDC->GetDeviceCaps(VERTRES);
 
    CDC memDC;
     if(memDC.CreateCompatibleDC(pDC) == 0)  return;
   
    CBitmap memBitmap, *oldmemBitmap;
     if(memBitmap.CreateCompatibleBitmap(pDC, Width, Height) == NULL)  return;
 
    oldmemBitmap = memDC.SelectObject(&memBitmap);
     if(oldmemBitmap == NULL)  return;
     if(memDC.BitBlt(0, 0, Width, Height, pDC, 0, 0, SRCCOPY) == 0)  return;
 
    BITMAP bmp;
    memBitmap.GetBitmap(&bmp);
   
     // fp = fopen(filename, "w+b");
 
    BITMAPINFOHEADER bih = {0};
    bih.biBitCount = bmp.bmBitsPixel;
    bih.biCompression = BI_RGB;
    bih.biHeight = bmp.bmHeight;
    bih.biPlanes = 1;
    bih.biSize =  sizeof(BITMAPINFOHEADER);
    bih.biSizeImage = bmp.bmWidthBytes * bmp.bmHeight;
    bih.biWidth = bmp.bmWidth;
   
    BITMAPFILEHEADER bfh = {0};
    bfh.bfOffBits =  sizeof(BITMAPFILEHEADER) +  sizeof(BITMAPINFOHEADER);
    bfh.bfSize = bfh.bfOffBits + bmp.bmWidthBytes * bmp.bmHeight;
    bfh.bfType = (WORD)0x4d42;
   
     // fwrite(&bfh, 1, sizeof(BITMAPFILEHEADER), fp);
    
// fwrite(&bih, 1, sizeof(BITMAPINFOHEADER), fp);
   
    image_buffer =  new BYTE[bmp.bmWidthBytes * bmp.bmHeight];
 
    GetDIBits(memDC.m_hDC,
        (HBITMAP) memBitmap.m_hObject,
     0,
     Height,
     image_buffer,
     (LPBITMAPINFO) &bih,
     DIB_RGB_COLORS);
    memDC.SelectObject(oldmemBitmap);
     // fwrite(p, 1, 1280 * 800 * 4, fp);
    
// fclose(fp);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个简单的C++程序,可以将JPEG图像换为BMP图像: ```cpp #include <cstdio> #include <cstdlib> #include <cstring> #include <jpeglib.h> #include <setjmp.h> #define BMP_HEADER_SIZE 54 const int BMP_FILE_HEADER_SIZE = 14; const int BMP_INFO_HEADER_SIZE = 40; typedef unsigned char BYTE; typedef unsigned short WORD; typedef unsigned int DWORD; typedef int LONG; #pragma pack(push, 1) typedef struct { WORD bfType; DWORD bfSize; WORD bfReserved1; WORD bfReserved2; DWORD bfOffBits; } BMP_FILE_HEADER; typedef struct { DWORD biSize; LONG biWidth; LONG biHeight; WORD biPlanes; WORD biBitCount; DWORD biCompression; DWORD biSizeImage; LONG biXPelsPerMeter; LONG biYPelsPerMeter; DWORD biClrUsed; DWORD biClrImportant; } BMP_INFO_HEADER; #pragma pack(pop) struct my_error_mgr { struct jpeg_error_mgr pub; jmp_buf setjmp_buffer; }; typedef struct my_error_mgr *my_error_ptr; METHODDEF(void) my_error_exit(j_common_ptr cinfo) { my_error_ptr myerr = (my_error_ptr) cinfo->err; (*cinfo->err->output_message) (cinfo); longjmp(myerr->setjmp_buffer, 1); } bool read_JPEG_file(const char *filename, BYTE **image_buffer, int *image_width, int *image_height) { struct jpeg_decompress_struct cinfo; struct my_error_mgr jerr; FILE *infile; JSAMPARRAY buffer; int row_stride; if ((infile = fopen(filename, "rb")) == NULL) { printf("Can't open %s\n", filename); return false; } cinfo.err = jpeg_std_error(&jerr.pub); jerr.pub.error_exit = my_error_exit; if (setjmp(jerr.setjmp_buffer)) { jpeg_destroy_decompress(&cinfo); fclose(infile); return false; } jpeg_create_decompress(&cinfo); jpeg_stdio_src(&cinfo, infile); jpeg_read_header(&cinfo, TRUE); jpeg_start_decompress(&cinfo); *image_width = cinfo.output_width; *image_height = cinfo.output_height; int num_channels = cinfo.output_components; *image_buffer = (BYTE *) malloc((*image_width) * (*image_height) * num_channels); row_stride = (*image_width) * num_channels; buffer = (*cinfo.mem->alloc_sarray)((j_common_ptr) &cinfo, JPOOL_IMAGE, row_stride, 1); BYTE *pBuf = *image_buffer; while (cinfo.output_scanline < cinfo.output_height) { jpeg_read_scanlines(&cinfo, buffer, 1); memcpy(pBuf, buffer[0], row_stride); pBuf += row_stride; } jpeg_finish_decompress(&cinfo); jpeg_destroy_decompress(&cinfo); fclose(infile); return true; } bool write_BMP_file(const char *filename, BYTE *image_buffer, int image_width, int image_height) { int image_size = image_width * image_height * 3; int file_size = BMP_HEADER_SIZE + image_size; FILE *f = fopen(filename, "wb"); if (!f) { printf("Can't create %s\n", filename); return false; } BMP_FILE_HEADER bmp_file_header = {0}; bmp_file_header.bfType = 0x4D42; // 'BM' bmp_file_header.bfSize = file_size; bmp_file_header.bfOffBits = BMP_HEADER_SIZE; BMP_INFO_HEADER bmp_info_header = {0}; bmp_info_header.biSize = BMP_INFO_HEADER_SIZE; bmp_info_header.biWidth = image_width; bmp_info_header.biHeight = image_height; bmp_info_header.biPlanes = 1; bmp_info_header.biBitCount = 24; bmp_info_header.biSizeImage = image_size; fwrite(&bmp_file_header, BMP_FILE_HEADER_SIZE, 1, f); fwrite(&bmp_info_header, BMP_INFO_HEADER_SIZE, 1, f); // BMP图像的像素是从下往上排列的,因此需要从最后一行开始写入 BYTE *pBuf = image_buffer + (image_height - 1) * image_width * 3; for (int i = 0; i < image_height; i++) { for (int j = 0; j < image_width; j++) { BYTE b = *pBuf++; BYTE g = *pBuf++; BYTE r = *pBuf++; BYTE color[] = {b, g, r}; fwrite(color, 3, 1, f); } pBuf -= image_width * 6; } fclose(f); return true; } int main(int argc, char *argv[]) { if (argc < 3) { printf("Usage: jpg2bmp input.jpg output.bmp\n"); return 0; } BYTE *image_buffer = NULL; int image_width = 0, image_height = 0; if (!read_JPEG_file(argv[1], &image_buffer, &image_width, &image_height)) { printf("Error: failed to read JPEG file\n"); return -1; } if (!write_BMP_file(argv[2], image_buffer, image_width, image_height)) { printf("Error: failed to write BMP file\n"); return -1; } if (image_buffer) { free(image_buffer); image_buffer = NULL; } return 0; } ``` 这个程序使用jpeglib来读取JPEG图像,然后将RGB数据写入BMP图像。它只支持24位真彩色图像,但可以轻松地修改以支持其他格式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值