stretchdibits(stretchdibits画灰度)

http://www.itjxue.com  2023-02-11 16:17  来源:未知  点击次数: 

stretchblt为什么会造成图片失真

VC 下 StretchBlt 的模式及失真问题

用一个CCD的相机,采集图像,然后在对话框的图片控件中显示。

现象:原本应该是黑白相机的灰度图像,显示出来的却是花花绿绿的彩色图像。

开始的时候

怀疑是采集的数据有问题,可是查看保存下来的bmp文件,却正常的很。可以确定是显示程序的

问题。问题是,基本上是按照sdk的sample逐行抄的,唯一的一点区别就是,sdk是在一个单

文档的view中显示,我的程序是在对话框的图片控件中显示,最有可能出错的就是

StretchDIBits(pDC-GetSafeHdc(),

rect.left,

rect.top,

rect.Width(), //显示窗口宽度

rect.Height(), //显示窗口高度

0,

0,

Width, //图像宽度

Height, //图像高度

m_pImageBuffer, //图像缓冲区

m_pBmpInfo, //BMP图像描述信息

DIB_RGB_COLORS,

SRCCOPY

);

认真地检查了所有的参数,没有发现异常。

跑到baidu和google上疯狂地搜,又重新建立了一个对话框工程,load一个图片来显示,问题

重现,偶然看到 里面

提到StretchDIBits引起的失真问题,

遂增加pDC- SetStretchBltMode(HALFTONE);

问题解决!

附上网文:

 在 VC 中使用 StretchBlt 会碰到一些与点阵图大小缩放相关的一些问题。在扩展一个点阵图时,StretchBlt必须复制图素行或列。如果放大倍数不是原图的整数倍,那么此操作会造成产生的图像有些失真。

如果目的矩形比来源矩形小,那么StretchBlt在缩小图像时就必须把两行(或列)或者多 行(或列)的图素合并到一行(或列)。完成此操作有四种方法,它根据装置内容伸展模式属性来选择其中一种方法。您可使用 SetStretchBltMode 函数来修改这个属性。

SetStretchBltMode (hdc, iMode) ;

iMode 可取下列值:

BLACKONWHITE 或者 STRETCH_ANDSCANS(内定):如果两个或多个图素得合并成一个图素,那么StretchBlt会对图素执行一个逻辑AND运算。这样的结果是只有全部的原始图素是白色时该图素才为白 色,其实际意义是黑色图素控制了白色图素。这适用于白背景中主要是黑色的单色点阵图。

WHITEONBLACK 或 STRETCH_ORSCANS:如果两个或多个图素得合并成一个图素,那么StretchBlt 执行逻辑OR运算。这样的结果是只有全部的原始图素都是黑色时才是黑色,也就是说由白色图 素决定颜色。这适用於黑色背景中主要是白色的单色点阵图。

COLORONCOLOR 或 STRETCH_DELETESCANS:StretchBlt 简单地消除图素行或列,而没有任何逻辑组合。这是通常是处理彩色点阵图的最佳方法。

HALFTONE 或 STRETCH_HALFTONE:Windows根据组合起来的来源颜色来计算目的的平均颜色。这将与半调调色盘联合使用,第十六章将展示这一程序。

利用StretchBlt缩小图片时有时会出现颜色失真。解决步骤如下:

1、先把目标DC (也就是 HDC hDestDC) ::SetStretchBltMode (hDestDC, HALFTONE);

2、调用一下 ::SetBrushOrgEx(hDestDC, 0, 0, NULL);

3、最后调用 CImage 的 StretchBlt

或者这样解决:

1。hbit = (HBITMAP)LoadImage( NULL,cBmpPath,IMAGE_BITMAP,0,0,LR_LOADFROMFILE);

if(hbit != NULL) {

Bitmap.Attach(hbit);

DCCompatible.CreateCompatibleDC(GetDC());

DCCompatible.SelectObject(Bitmap);

Bitmap.GetObject(sizeof(bm),bm);}

2。OnPaint():

pDC-SetStretchBltMode(HALFTONE);

pDC-StretchBlt(MAP_LEFT,MAP_TOP,MAP_WIDTH,MAP_HEIGHT,

DCCompatible,0,0,bm.bmWidth,bm.bmHeight,SRCCOPY);

CDIB什么意思

VC里的类

Visual C++ MFC中没有提供一个专门的类来处理DIB位图,因此,为了方便地使用位图文件,我们有必要派生一个CDib类。类的源代码如下:

(1) CDib类的声明

// DIB.h:类CDib声明头文件

#ifndef __DIB_H__

#define __DIB_H__

#include wingdi.h

class CDib

{

public:

CDib();

~CDib();

BOOL Load( const char * );

BOOL Save( const char * );

BOOL Draw( CDC *, int nX = 0, int nY = 0, int nWidth = -1, int nHeight = -1, int mode = SRCCOPY);

BOOL SetPalette( CDC * );

private:

CPalette m_Palette;

unsigned char *m_pDib, *m_pDibBits;

DWORD m_dwDibSize;

BITMAPINFOHEADER *m_pBIH;

RGBQUAD *m_pPalette;

int m_nPaletteEntries;

};

#endif

(2) CDib类的实现

// DIB.cpp:类CDib实现文件

#include "stdafx.h"

#include "DIB.h"

CDib::CDib()

{

m_pDib = NULL;

}

CDib::~CDib()

{

// 如果位图已经被加载,释放内存

if (m_pDib != NULL)

delete []m_pDib;

}

下面这个函数非常重要,其功能为加载位图,类似于CBitmap类的LoadBitmap函数:

BOOL CDib::Load(const char *pszFilename)

{

CFile cf;

// 打开位图文件

if (!cf.Open(pszFilename, CFile::modeRead))

return (FALSE);

// 获得位图文件大小,并减去BITMAPFILEHEADER的长度

DWORD dwDibSize;

dwDibSize = cf.GetLength() - sizeof(BITMAPFILEHEADER);

// 为DIB位图分配内存

unsigned char *pDib;

pDib = new unsigned char[dwDibSize];

if (pDib == NULL)

return (FALSE);

BITMAPFILEHEADER BFH;

// 读取位图文件数据

try

{

// 文件格式是否正确有效

if ( cf.Read(BFH, sizeof(BITMAPFILEHEADER)) != sizeof(BITMAPFILEHEADER) ||

BFH.bfType != ’MB’ || cf.Read(pDib, dwDibSize) != dwDibSize)

{

delete []pDib;

return (FALSE);

}

}

catch (CFileException *e)

{

e-Delete();

delete []pDib;

return (FALSE);

}

// delete先前加载的位图

if (m_pDib != NULL)

delete m_pDib;

// 将临时Dib数据指针和Dib大小变量赋给类成员变量

m_pDib = pDib;

m_dwDibSize = dwDibSize;

// 为相应类成员变量赋BITMAPINFOHEADER和调色板指针

m_pBIH = (BITMAPINFOHEADER*)m_pDib;

m_pPalette = (RGBQUAD*) m_pDib[sizeof(BITMAPINFOHEADER)];

// 计算调色板中实际颜色数量

m_nPaletteEntries = 1 m_pBIH-biBitCount;

if (m_pBIH-biBitCount 8)

m_nPaletteEntries = 0;

else if (m_pBIH-biClrUsed != 0)

m_nPaletteEntries = m_pBIH-biClrUsed;

// 为相应类成员变量赋image data指针

m_pDibBits = m_pDib[sizeof(BITMAPINFOHEADER) + m_nPaletteEntries * sizeof (RGBQUAD)];

// delete先前的调色板

if (m_Palette.GetSafeHandle() != NULL)

m_Palette.DeleteObject();

// 如果位图中存在调色板,创建LOGPALETTE 及CPalette

if (m_nPaletteEntries != 0)

{

LOGPALETTE *pLogPal = (LOGPALETTE*)new char[sizeof(LOGPALETTE) + m_nPaletteEntries *sizeof(PALETTEENTRY)];

if (pLogPal != NULL)

{

pLogPal-palVersion = 0x300;

pLogPal-palNumEntries = m_nPaletteEntries;

for (int i = 0; i m_nPaletteEntries; i++)

{

pLogPal-palPalEntry[i].peRed = m_pPalette[i].rgbRed;

pLogPal-palPalEntry[i].peGreen = m_pPalette[i].rgbGreen;

pLogPal-palPalEntry[i].peBlue = m_pPalette[i].rgbBlue;

}

//创建CPalette并释放LOGPALETTE的内存

m_Palette.CreatePalette(pLogPal);

delete []pLogPal;

}

}

return (TRUE);

}

//函数功能:保存位图入BMP文件

BOOL CDib::Save(const char *pszFilename)

{

if (m_pDib == NULL)

return (FALSE);

CFile cf;

if (!cf.Open(pszFilename, CFile::modeCreate | CFile::modeWrite))

return (FALSE);

try

{

BITMAPFILEHEADER BFH;

memset(BFH, 0, sizeof(BITMAPFILEHEADER));

BFH.bfType = ’MB’;

BFH.bfSize = sizeof(BITMAPFILEHEADER) + m_dwDibSize;

BFH.bfOffBits = sizeof(BITMAPFILEHEADER) +

sizeof(BITMAPINFOHEADER) + m_nPaletteEntries *sizeof(RGBQUAD);

cf.Write(BFH, sizeof(BITMAPFILEHEADER));

cf.Write(m_pDib, m_dwDibSize);

}

catch (CFileException *e)

{

e-Delete();

return (FALSE);

}

return (TRUE);

}

下面这个函数也非常重要,其功能为在pDC指向的CDC中绘制位图,起点坐标为(nX,nY),绘制宽度和高度为nWidth、nHeight,最后一个参数是光栅模式:

BOOL CDib::Draw(CDC *pDC, int nX, int nY, int nWidth, int nHeight, int mode)

{

if (m_pDib == NULL)

return (FALSE);

// 获取位图宽度和高度赋值

if (nWidth == - 1)

nWidth = m_pBIH-biWidth;

if (nHeight == - 1)

nHeight = m_pBIH-biHeight;

// 绘制位图

StretchDIBits(pDC-m_hDC, nX, nY, nWidth, nHeight, 0, 0, m_pBIH-biWidth, m_pBIH-biHeight, m_pDibBits, (BITMAPINFO*)m_pBIH, BI_RGB, mode);

return (TRUE);

}

//函数功能:设置调色板

BOOL CDib::SetPalette(CDC *pDC)

{

if (m_pDib == NULL)

return (FALSE);

// 检查当前是否有一个调色板句柄,对于大于256色的位图,为NULL

if (m_Palette.GetSafeHandle() == NULL)

return (TRUE);

// 选择调色板,接着实施之,最后恢复老的调色板

CPalette *pOldPalette;

pOldPalette = pDC-SelectPalette(m_Palette, FALSE);

pDC-RealizePalette();

pDC-SelectPalette(pOldPalette, FALSE);

return (TRUE);

}

从整个CDib类的代码中我们可以看出,DIB位图的显示需遵循如下步骤:

(1)读取位图,本类中使用pDib = new unsigned char[dwDibSize]为位图中的信息分配内存,另一种方法是调用API函数CreateDIBSection,譬如:

m_hBitmap = ::CreateDIBSection(pDC-GetSafeHdc(),

(LPBITMAPINFO) m_lpBMPHdr, DIB_RGB_COLORS,

(LPVOID*) m_lpDIBits, NULL, 0);

m_hBitmap定义为:

HBITMAP m_hBitmap;

(2)根据读取的位图信息,计算出调色板大小,然后创建调色板;

(3)调用CDib::SetPalette( CDC *pDC )设置调色板,需要用到CDC::SelectPalette及CDC::RealizePalette两个函数;

(4)调用CDib::Draw(CDC *pDC, int nX, int nY, int nWidth, int nHeight, int mode)函数绘制位图。在此函数中,真正发挥显示位图作用的是对StretchDIBits API函数的调用。StretchDIBits函数具有缩放功能,其最后一个参数也是光栅操作的模式。

下面给出DIB位图的打开及显示并在其中加入天极网logo的函数源代码。"DIB位图"父菜单下"打开"子菜单的单击事件消息处理函数为(其功能为打开位图并显示之):

void CBitMapExampleDlg::OnOpendibpic()

{

// 弹出文件对话框,让用户选择位图文件

CFileDialog fileDialog(TRUE, "*.BMP", NULL, NULL,"位图文件(*.BMP)|*.bmp;*.BMP|");

if (IDOK == fileDialog.DoModal())

{

// 加载位图并显示之

CDib dib;

if (dib.Load(fileDialog.GetPathName()))

{

CClientDC dc(this);

dib.SetPalette(dc);

dib.Draw(dc);

}

}

}

"DIB位图"父菜单下"标记"子菜单的单击事件消息处理函数为(其功能为给位图加上天极网logo):

void CBitMapExampleDlg::OnMarkDibpic()

{

// 弹出文件对话框,让用户选择标记logo

CFileDialog fileDialog(TRUE, "*.BMP", NULL, NULL, "标记位图文件(*.BMP)|*.bmp;*.BMP|");

if (IDOK == fileDialog.DoModal())

{

// 加载标记logo位图并与目标位图相与

CDib dib;

if (dib.Load(fileDialog.GetPathName()))

{

CClientDC dc(this);

dib.SetPalette(dc);

dib.Draw(dc, 0, 0, - 1, - 1, SRCAND);

}

}

MFC中关于StretchDIBits的问题

??if?(x1/y1x2/y2)

??{

???xDest=0;

???yDest=(y2-y1*(x2/x1))/2;

???DestWidth=x2;

???DestHeight=y2-yDest;

??}

??else

??{

???xDest=(x2-x1*(y2/y1))/2;

???yDest=0;

???DestWidth=x2-xDest;

???DestHeight=y2;

??}

?}

你这里的计算有问题。

第一个情况,yDest+DestHeight = y2 = wRect的高度。也就是在最下方显示了,而不是高度居中。

??if?(x1*1.0/y1x2*1.0/y2)

??{

???xDest=0;

???DestWidth=x2;

???DestHeight=y1*x2/x1;

???yDest=(y2-DestHeight)/2;

??}

这样应该就对了,else自己写一下。

StretchDIBits显示内存中的数据问题

用StretchDIBits函数vac假设你的数据存储在下面两个结构中:BITMAPINFO bmiPBYTEpbits在视图类的OnDraw函数中添加一句:::StretchDIBits(pDC->GetSafeHdc()iko \0 06 bmi.bmiHeader.biWidth, bmi.bmiHeader.biHeight, \0, 0, bmi.bmiHeader.biWidth, bmi.bmiHeader.biHeight,\pbits, bmi, DIB_RGB_COLORS, SRCCOPY );即可

c#中调用函数StretchDIBits的参数问题,函数功能涉及图像处理显示

Graphics g

StretchDIBits(g.GetHdc(), 0, 0, 512, 512, 0, 0, 512, 512, pt, bm, DIB_RGB_COLORS, SRCCOPY);

别人写的

public void DrawImage(byte[] pixels, Graphics g)

{

// Create a BITMAPINFO object representing the image to be drawn

// Image is 512 x 512, 8bpp

BITMAPINFO bmi = new BITMAPINFO();

bmi.biWidth = 512;

bmi.biHeight = 512;

bmi.biPlanes = 1;

bmi.biBitCount = 8;

bmi.biSizeImage = pixels.Length;

IntPtr pt = Marshal.AllocHGlobal(Marshal.SizeOf(typeof(BITMAPINFO)));

[color=#FF0000] Marshal.Copy(pixels, 0, pt, Marshal.SizeOf(typeof(BITMAPINFO)));

IntPtr bm = Marshal.AllocHGlobal(Marshal.SizeOf(typeof(BITMAPINFO)));[/color] // Draw the image using the graphics object

StretchDIBits(g.GetHdc(), 0, 0, 512, 512, 0, 0, 512, 512, pt, bm, DIB_RGB_COLORS, SRCCOPY);

}

(责任编辑:IT教学网)

更多

相关FTP服务器文章

推荐FTP服务器文章