一、vc编译jpeglib库

1、下载源代码

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

2、编译源代码.

A、解压源代码,修改源代码中jconfig.vc为jconfig.h;

B、添加环境变量PATH,C:/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了

3、jpeglib库VC下使用

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

将这几个文件拷贝到你的项目中:libjpeg.lib,jconfig.h,jmorecfg.h,jpeglib.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的位图的像素格式是:BGRA,4个字节,jpeglib库使用的是RGB,3个字节的格式,所以需要将源像素去掉其透明字节,同时改变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文件有100多K的样子。

三、代码:

#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);

}

from:http://blog.csdn.net/huxiangyang4/article/details/5728888

使用jpeglib库实现bmp转jpg的更多相关文章

  1. Linux 下V4l2摄像头采集图片,实现yuyv转RGB,RGB转BMP,RGB伸缩,jpeglib 库实现压缩RGB到内存中,JPEG经UDP发送功(转)

    ./configure CC=arm-linux-gnueabihf-gcc LD=arm-linux-gnueabihf-ld --host=arm-linux --prefix=/usr/loca ...

  2. 【Linux开发】Linux下jpeglib库的安装详解

    Linux下jpeglib库的安装详解 首先要下载所需的库压缩包:jpegsrc.v6b.tar.gz或 jpegsrc.v8b.tar.gz 然后将下载的压缩包随便放在和解压到你喜欢的地方. # t ...

  3. 几个关于JPEGLIB库的博客

    1.http://blog.csdn.net/huxiangyang4/archive/2010/07/12/5728888.aspx 我认为是最好的 2.http://blog.csdn.net/a ...

  4. Linux jpeglib库的安装

    tar -zxvf jpegsrc.v9.tar.gz cd jpeg9 ./configure --enable-shared  --enable-static 分别对动态链接库和静态链接库的支持 ...

  5. jpeglib使用指南

    您可以到http://www.ijg.org/网站下载libjpeg的源码, IJG JPEG Library就是jpeg压缩库,是以源码的形式提供给软件开发人员的,当然在软件包里也有编译好的库文件, ...

  6. 【Linux开发】jpeglib使用指南

    您可以到www.ijg.org网站下载libjpeg的源码, IJG JPEG Library就是jpeg压缩库,是以源码的形式提供给软件开发人员的,当然在软件包里也有编译好的库文件,我们这里就只用到 ...

  7. 立体匹配:关于Middlebury提供的源码的简化使用

    Middlebury提供的源码,虽然花了不到一个小时就运行起来啦.但说实话,它那循环读取脚本命令来执行算法真是让我费了不少头脑,花了近三天时间,我才弄明白了它的运行机制.你说,我就想提取一下算法,你给 ...

  8. CentOS6.8配置GO语言开发环境

    Go语言是谷歌2009发布的第二款开源编程语言,Go语言专门针对多处理器系统应用程序的编程进行了优化,使用Go编译的程序可以媲美C或C++代码的速度,而且更加安全.支持并行进程. 鉴于原来越多的开源项 ...

  9. 如何编写go代码

    go是一种静态编译型的语言,它的编译速度非常快. go的官方编译器称为gc,包括编译工具5g,6g和8g,连接工具5l,6l和8l.其中的数字表示处理器的架构.我们不必关心如何挑选这些工具,因为go提 ...

随机推荐

  1. python之函数的使用

    备注:本篇文章主要讲一讲函数及集合的一些常用用法: 一.首先先看下,集合(set): 集合的特点:无序.不重复(这点跟字典有点像) <1>,在需要访问集合的时候,由于集合本身是无序的,所以 ...

  2. linux shell if语句

    #!/bin/bash read -p "please input Y/N" keyWord if [ "$keyWord" == "Y" ...

  3. nginx上传模块nginx_upload_module和nginx_uploadprogress_module模块进度显示,如何传递GET参数等。

    ownload:http://www.grid.net.ru/nginx/download/nginx_upload_module-2.2.0.tar.gzconfigure and make : . ...

  4. javascript 跨域汇总

    什么是跨域?当两个域具有相同的协议.相同的端口.相同的host时,那么我们就可以认为它们是相同的域.比如:http://www.example.com/a.html 和 http://www.exam ...

  5. Qt widgets deeps--烧鸡

    1,Qt类读取目录 QDir读取目录内容--将读取结果输出到一个QMultiLineEdit对象 QMultiLineEdit *medit; medit = new QMultiLineEdit(t ...

  6. 全国计算机等级考试二级教程-C语言程序设计_第5章_循环结构

    for循环结构的嵌套 外层循环每循环一次,内层循环会完整循环一次. 外层循环是竖. 内层循环是横. for, do...while, while的选择: 如果有固定次数,如阶乘! ,判断素数,用 fo ...

  7. C++求二叉树的最大高度差

    #include <iostream> #include <string.h> using namespace std; template<typename Type&g ...

  8. iOS使用ffmpeg播放rstp实时监控视频数据流

    一.编译针对iOS平台的ffmpeg库(kxmovie) 最近有一个项目.须要播放各种格式的音频.视频以及网络摄像头实时监控的视频流数据,经过多种折腾之后,最后选择了kxmovie,kxmovie项目 ...

  9. UVA11387 - The 3-Regular Graph(推理)

    题目链接 题意:给n个点,问能否画出一个无向图.且每一个顶点连接3条边.假设能够的话输出连接的边. 思路:当添加一条边时,总的无向图的度数会添加2,所以度数之和n*2为偶数.当n为奇数时,度数之和为奇 ...

  10. Spark里面的任务调度:离SparkContext开始

    SparkContext这是发达国家Spark入学申请,它负责的相互作用和整个集群,它涉及到创建RDD.accumulators and broadcast variables.理解力Spark架构, ...