有关读tif的问题!!

ricky20045 2006-01-16 08:52:35
最近想直接读tif格式的图像,搞了好久也没什么头绪
那位大虾有读Tif图像的代码,仅参考一下!
不胜感激!高分送上。
...全文
175 4 打赏 收藏 转发到动态 举报
写回复
用AI写文章
4 条回复
切换为时间正序
请发表友善的回复…
发表回复
TianChong 2006-01-17
  • 打赏
  • 举报
回复
1、VC创建一个MFC AppWizard(exe)工程取名TiffTest,选择单文档程序。
2、添加TiffStruct.h文件,定义IFH和DE结构(参考前面的结构描述),用来接收读TIF文件的信息。

#ifndef _TIFFSTRUCT_
#define _TIFFSTRUCT_

typedef struct tagIMAGEFILEHEADER
{
WORD byteOrder;
WORD version;
DWORD offsetToIFD;
}IFH;

typedef struct tagDIRECTORYENTRY
{
WORD tag;
WORD type;
DWORD length;
DWORD valueOffset;
}DE;

#endif

3、在文档类中添加4个公有变量,并将其初始化为0。在TiffTestDoc.cpp中#include "TiffStruct.h" DWORD m_dwBmSize; //图象的数据部分的大小
CPalette m_palDIB; //BMP图象调色板
HANDLE m_hDIB; //BMP图象内存块句柄
CSize m_sizeDoc; //图象的长和宽

4、在文档类的OnOpenDocument函数中定义局部工具变量并读文件 DWORD dwFileLength = 0;
CString strTemp = _T("");
WORD wDECount = 0;
BYTE* pDIB = NULL;
int i = 0;

IFH ifh;
ZeroMemory(&ifh, sizeof(IFH));

CFile file;
CFileException fe;

if(0 == file.Open(lpszPathName, CFile::modeRead | CFile::shareDenyWrite, &fe))
{
AfxMessageBox("打开文件失败");
return FALSE;
}
dwFileLength = file.GetLength();

读IFH文件头 if(sizeof(IFH) != file.Read(&ifh, sizeof(IFH)))
{
AfxMessageBox("读TIF文件头失败");
return FALSE;
}

if(0x2a != ifh.version)
{
AfxMessageBox("该文件不是TIF格式,读文件失败");
return FALSE;
}

if(0x4949 != ifh.byteOrder)
{
AfxMessageBox("该TIF文件不是IBMPC字节序,读文件失败");
return FALSE;
}

file.Seek(ifh.offsetToIFD, CFile::begin);//将文件指针定位到IFD

读文件有多少个目录入口 if(2 != file.Read(&wDECount, 2))
{
AfxMessageBox("无法获得TIF文件目录入口数量");
return FALSE;
}
strTemp.Format("该TIF文件有%d个目录入口", wDECount);
AfxMessageBox(strTemp);

创建DE数组,接收信息,数组中有wDECount个元素 DE* pde = new DE[wDECount];
DE* pTemp = pde;
memset(pde, 0, sizeof(DE)*wDECount);
if(sizeof(DE)*wDECount != file.ReadHuge(pde, sizeof(DE)*wDECount))
{
AfxMessageBox("读图象文件目录失败");
delete []pde;
return FALSE;
}

显示图像文件目录信息 for(i=0; itag, i, pTemp->type, i, pTemp->length, i, pTemp->valueOffset);
AfxMessageBox(strTemp);
}

把图像的大小和图像数据的容量保存到成员变量中 for(i=0; i<wDECount; i++)
{
pTemp = pde + i;
if(256 == pTemp->tag) //tag为256的目录入口中的变量标识了图象宽度
{
m_sizeDoc.cx = pTemp->valueOffset;
}
if(257 == pTemp->tag) //图象高度
{
m_sizeDoc.cy = pTemp->valueOffset;
}
if(273 == pTemp->tag) //计算图象数据占用字节数
{
//m_dwBmSize = pTemp->valueOffset - sizeof(IFH);
//或者把tag=256的valueOffset乘以tag=257的valueOffset
m_dwBmSize = m_sizeDoc.cx * m_sizeDoc.cy;
}
}

在文档类中创建一个成员工具函数CreateBmpBuffer,申请全局内存块以存放BMP文件结构数据 BOOL CTiffTestDoc::CreateBmpBuffer()
{
//申请BMP内存块
m_hDIB = GlobalAlloc(GMEM_MOVEABLE | GMEM_ZEROINIT,
sizeof(BITMAPFILEHEADER) +
sizeof(BITMAPINFOHEADER) +
256*sizeof(RGBQUAD) +
m_dwBmSize);
if(NULL == m_hDIB)
{
AfxMessageBox("申请BMP内存块失败");
return FALSE;
}
else
{
return TRUE;
}
}

回到OnOpenDocument成员函数中调用工具函数并获得全局内存块指针 //构造BMP图象内存块
if(!CreateBmpBuffer())
{
AfxMessageBox("构造BMP图象内存块失败");
delete []pde;
return FALSE;
}

//获得BMP内存块指针
pDIB = (BYTE*)GlobalLock(m_hDIB);
if(NULL == pDIB)
{
AfxMessageBox("获得BMP内存块指针失败");
GlobalUnlock(m_hDIB);
delete []pde;
return FALSE;
}

  以下是将图像信息填充到BMP内存块中,网上介绍BMP格式的文章很多,这里就不详述了。因测试图像数据表达的是0x00-0xff灰度,正好和BMP文件调色板索引值巧合。 故在代码中直接把图像数据信息当成索引即可,减少了编码复杂度。注:BMP文件中图像数据的第一行代表的是最终显示光栅的最后一行,所以在数据排列中要颠倒过来。 //构造BITMAPFILEHEADER并复制到BMP内存块
BITMAPFILEHEADER bmfHdr;
memset(&bmfHdr, 0, sizeof(BITMAPFILEHEADER));
bmfHdr.bfOffBits = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) +
256*sizeof(RGBQUAD);
bmfHdr.bfReserved1 = 0;
bmfHdr.bfReserved2 = 0;
bmfHdr.bfSize = bmfHdr.bfOffBits + m_dwBmSize;
bmfHdr.bfType = 0x4d42;

memmove(pDIB, &bmfHdr, sizeof(BITMAPFILEHEADER));

//构造BITMAPINFOHEADER并复制到BMP内存块
BITMAPINFOHEADER bmiHdr;
memset(&bmiHdr, 0, sizeof(BITMAPINFOHEADER));
bmiHdr.biBitCount = 8;
bmiHdr.biClrImportant = 0;
bmiHdr.biClrUsed = 0;
bmiHdr.biCompression = 0;
bmiHdr.biHeight = m_sizeDoc.cy;
bmiHdr.biPlanes = 1;
bmiHdr.biSize = sizeof(BITMAPINFOHEADER);
bmiHdr.biSizeImage = 0;
bmiHdr.biWidth = m_sizeDoc.cx;
bmiHdr.biXPelsPerMeter = 2834;
bmiHdr.biYPelsPerMeter = 2834;

memmove((BITMAPFILEHEADER*)pDIB + 1, &bmiHdr, sizeof(BITMAPINFOHEADER));

//构造256个RGBQUAD并复制到BMP内存块
RGBQUAD* pRgbQuad = (RGBQUAD*)(pDIB + sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER));
RGBQUAD* pOldQuad = pRgbQuad;
RGBQUAD rgbQuad;
memset(&rgbQuad, 0, sizeof(RGBQUAD));
for(i=0; i<256; i++)
{
rgbQuad.rgbBlue = i;
rgbQuad.rgbGreen = i;
rgbQuad.rgbRed = i;
rgbQuad.rgbReserved = 0;
pRgbQuad = pOldQuad + i;
memmove(pRgbQuad, &rgbQuad, sizeof(RGBQUAD));
}

//填充所有像素数据, 颠倒图象数据从最后一行开始读起
int j = 0;
for(i=m_sizeDoc.cy-1; i>=0; i--)
{
file.Seek(sizeof(IFH) + i*m_sizeDoc.cx, CFile::begin);
file.Read((BYTE*)(pRgbQuad + 1) + j*m_sizeDoc.cx, m_sizeDoc.cx);
j++;
}

初始化BMP调色板,为显示BMP文件做准备 //初始化专用调色板
BYTE buf[2+2+4*256];
LOGPALETTE* pPal = (LOGPALETTE*)buf;
pPal->palVersion = 0x300;
pPal->palNumEntries = 256;
for(i=0; i<255; i++)
{
pPal->palPalEntry[i].peBlue = i;
pPal->palPalEntry[i].peFlags = 0;
pPal->palPalEntry[i].peGreen = i;
pPal->palPalEntry[i].peRed = i;
}
m_palDIB.CreatePalette(pPal);

最后是OnOpenDocument成员函数返回前的清理工作 GlobalUnlock(m_hDIB);
delete []pde;
return TRUE;

  至此,TIF文件信息已转换为BMP图像并保存在全局内存块中了,接下来就可以在OnDraw中调用WinAPI函数StretchDIBits来显示它。 if(NULL == pDoc->m_hDIB)
{
return;
}
HDC hdc = pDC->m_hDC;
BYTE* pBuf = (BYTE*)GlobalLock(pDoc->m_hDIB);
pBuf += sizeof(BITMAPFILEHEADER);
BYTE* pData = pBuf + sizeof(BITMAPINFOHEADER) + 256*sizeof(RGBQUAD);

CPalette* pOldPal = pDC->SelectPalette(&pDoc->m_palDIB, TRUE);
pDC->RealizePalette();

::SetStretchBltMode(hdc, COLORONCOLOR);
::StretchDIBits(hdc, 10, 10, pDoc->m_sizeDoc.cx, pDoc->m_sizeDoc.cy,
0, 0, pDoc->m_sizeDoc.cx, pDoc->m_sizeDoc.cy, pData,
(BITMAPINFO*)pBuf, DIB_RGB_COLORS, SRCCOPY);

pDC->SelectPalette(pOldPal, FALSE);
GlobalUnlock(pDoc->m_hDIB);


最后别忘了在文档类析构函数~CTiffTestDoc中销毁全局内存块 if(NULL != m_hDIB)
{
GlobalFree(m_hDIB);
m_hDIB = NULL;
}

zcjcom830 2006-01-16
  • 打赏
  • 举报
回复
libtif
lixiaosan 2006-01-16
  • 打赏
  • 举报
回复
参考一下,我也不太了解

分析一下源码吧。。

http://www.codeproject.com/bitmap/graphicsuite.asp
http://www.codeproject.com/bitmap/BitmapsToTiffs.asp
http://www.codeproject.com/csharp/tiffmanager.asp
http://www.codeguru.com/Cpp/G-M/bitmap/otherformats/article.php/c4933
pomelowu 2006-01-16
  • 打赏
  • 举报
回复
http://www.vckbase.com/document/viewdoc/?id=1569

16,551

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC相关问题讨论
社区管理员
  • 基础类社区
  • Creator Browser
  • encoderlee
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……

试试用AI创作助手写篇文章吧