将1bpp的bmp图像存储为1bpp或者2bpp的tiff格式



// 将1bpp的位图转换为 1bit/2bit tiff
/**
参数:BYTE *src  二值图像的像素数据,不包含头部信息, 1bpp, int src_width  原图的宽度, in pixles,int src_height  原图的高度, in pixlesint bpp   转换tiff指定的bpp
*/
static BYTE *BW2Tif(BYTE *src, int src_width, int src_height, int bpp, DWORD& totalSize)
{int dst_width  = (bpp == 1) ? src_width : src_width/2;int dst_height = src_height;// get src and dst scan widthint src_pitch = (src_width + 7)/8 + 3 & ~3; int dst_pitch = (dst_width * bpp + 31)/32*4;// get src and dst sizeunsigned int tif_size, bmp_size;if(0x80000000 & dst_height){bmp_size= src_pitch*(0 - src_height);tif_size = dst_pitch*(0 - dst_height);}else{bmp_size = src_pitch*src_height;tif_size = dst_pitch*dst_height;}// Image File Headerconst unsigned char tif_ifh[10]={'I','I',					//intel顺序0x2a,0x00,					//tiff版本号0x08,0x00,0x00,0x00,		//IFD(图形文卷目录 Image File Diretory)偏移, 8个字节0x0F,0x00,					// 这两个字节属于IFD了,指定有多少个Directory Entry};// Directory Entry ,每个都是12个字节的 , TIFF 6.0规定,标签要以升序的方式排列unsigned long tif_ifd[0x0f*3+4]={0x000400FE, 0x01,	0x000,		// NewSubfileType,   0x00030100, 0x01,	0x018,		// ImageWidth,The number of columns in the image, i.e., the number of pixels per scanline0x00030101, 0x01,	0x018,		// ImageLength, The number of rows (sometimes described as scanlines) in the image.0x00030102, 0x01,	bpp,		// BitPerSample  2580x00030103, 0x01,	0x001,		// Compression, 1无压缩0x00030106, 0x01,	0x000,		// PhotometricInterpretation,  对于bilevel图像,值0=WhiteIsZero,1=BlackIsZero0x00040111, 0x01,	0x0CE,		// StripOffsets,For each strip, the byte offset of that strip, // N = StripsPerImage for PlanarConfiguration equal to 1; // N = SamplesPerPixel * StripsPerImage for PlanarConfiguration equal to 2// 从字面上看来, StripOffsets是个数组,小于64kb,  这里是0xce, 就是头部的206个字节了0x00030112, 0x01,	0x001,		// Orientation0x00030115, 0x01,	0x001,		// SamplesPerPixel0x00030116, 0x01,	0xffff,		// RowsPerStrip, The number of rows in each strip (except possibly the last strip.)// RowsPerStrip and ImageLength together tell us the number of strips in the entire image. default is 2^32-1, 推荐每个strip是8k数据// image. The equation is:     StripsPerImage= floor ((ImageLength + RowsPerStrip - 1) / RowsPerStrip).0x00040117, 0x01,	0x07F,		// StripByteCounts0x0005011A, 0x01,	0x0BE,		// XResolution0x0005011B, 0x01,	0x0C6,		// YResolution0x0003011C, 0x01,	0x001,		// PlanarConfiguration , 1是单平面格式, 2多平面格式0x00030128, 0x01,	0x002,		// ResolutionUnit, 0x002 = inch0x60031280, 0x01000000,			// X分辨率ofset=1640x60031280, 0x01000000,			// Y分辨率ofset=16C};//tif_ifd[ 5] = dst_width;tif_ifd[ 8] = dst_height;tif_ifd[29] = (tif_size + 0x0f) & 0xfffffff0;  // 简单的做成一个striptif_ifd[32] = tif_size;// allocate dst image  hpglunsigned int total_size = sizeof(tif_ifh) + sizeof(tif_ifd) + tif_size;TRACE1("tiff size=0x%x\n", sizeof(tif_ifh) + sizeof(tif_ifd));BYTE *dst =  new BYTE[total_size];if(NULL == dst) return FALSE;BYTE *curr = dst;totalSize = total_size;// memcpy(curr, tif_ifh, sizeof(tif_ifh));curr += sizeof(tif_ifh);memcpy(curr, tif_ifd, sizeof(tif_ifd));curr += sizeof(tif_ifd);// 当src图像的宽度是整数个字节的时候,可以直接复制数据,如果多出几个bit,就不好搞了//memcpy(curr, src, tif_size);// gray2bmp_to_tif.c 中程序,是把每个像素用一个字节来存储了,而这里不是的// 所以不能照着写程序了int width_align = src_width & 0xffffFFF8;int nBytes = width_align/8;if (src_width - width_align)   // 有边角bit位nBytes += 1;LPBYTE src2 = src;  LPBYTE dst2 = curr;int x,y;for (y = 0; y < src_height; y++){src2 = src + y * src_pitch;dst2 = curr + y * nBytes;    //写入tiff的数据流,不要自己补充空白数据memcpy(dst2, src2, nBytes);src2 += nBytes;dst2 += nBytes;}return dst;
}// 这个是测试代码
void CPagePt::OnBnClickedButtonCv()
{CFileDialog fd(TRUE, _T("bmp"), NULL,  NULL,  _T("All files(*.*)|*.*|Wibdows Bitmap(*.bmp)|*.bmp||"));if(fd.DoModal() != IDOK)return;CString filename = fd.GetPathName();HBITMAP hImage = (HBITMAP)LoadImage(NULL, filename, IMAGE_BITMAP,0, 0, LR_LOADFROMFILE|LR_CREATEDIBSECTION|LR_DEFAULTSIZE);//CBitmap* m_Bitmap = CBitmap::FromHandle(hImage);BITMAP bm;				//获取位图信息 GetObject(hImage, sizeof(BITMAP), &bm); // Sumit: memory allocation is still 1800x1800 in your code..BYTE* bmpBuffer=(BYTE*)bm.bmBits;    // GlobalAlloc(GPTR, bm.bmWidthBytes * bm.bmHeight);      //allocate memoryDWORD tifSize;BYTE *tif = BW2Tif(bmpBuffer, bm.bmWidth, bm.bmHeight, 2, tifSize);CFile cf("C:\\bw2tif_out.tif", CFile::modeCreate | CFile::modeWrite);cf.Write(tif, tifSize);cf.Close();delete tif;}


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/243292.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

关于联合体的内容

关于联合体的内容 联合体只储存一个值 联合体只存储一个元素&#xff0c;但这个元素可以有多种表示方法&#xff0c;每种表示可以是不同的数据类型。 联合体的声明语法类似于结构体&#xff0c;联合体的成员的引用也跟结构体一样。 union{int i;int unsigned u; }data; ... d…

Point-BERT:一种基于Transformer架构的点云深度网络

目录 1. 前言 2. Point Tokenization 3. Transformer Backbone 4. Masked Point Modeling 5. Experiments Reference 1. 前言 从PointNet [1] 开始&#xff0c;点云深度网络逐渐成为解决点云特征提取与语义分析的主要研究方向。尤其在OpenAI的GPT模型获得了突破性成果后&#…

GNS3 VoIP Lab (Cisco 3725 and CME 4.3)

Here is a simple VoIP Lab in GNS3 environment. It is only used for my lab test and recorded here for future reference. 1. Topology: GNS3 Topology:Logic Topology:xp(192.168.2.60)——–C3725 Router(192.168.2.10) 2. Enviroment: ESXi 5.5 (or Vmware Workstation…

谁知道这个代码片段干嘛的

int value 0xAAAA;for (int i0; i<8; i){int tmp value & 0x3; // 取出第两个比特位置if (tmp 0x0){//}else if (tmp 0x1){//}else if (tmp 0x2){//}else if(tmp 0x3){TRACE0("???");}TRACE1("tmp0x%x\n\n", tmp);value >> 2;}

关于数组的内容

关于数组的内容 Verilog数组声明的基本语法 <data_type><vector_size><array_name><array_dimension> 例如&#xff1a; reg[15:0] RAM [0:4095];//储存器数组SystemVerilog允许任何数据类型的非压缩数组 SystemVerilog将非压缩数组的声明进行了扩展…

Touch Driver介绍

Touch Driver介绍 一&#xff0e;相关知识介绍 1&#xff0e;Touch Driver的加载过程 GWES到[HKEY_LOCAL_MACHINE/HARDWARE/DEVICEMAP/TOUCH]的“Driver name”获取Driver DLL的名字&#xff0c;如果没有找到该键值&#xff0c;则使用默认名字Touch.dll。 Touch Driver的加…

BreadCrumb控件

BreadCrumb控件&#xff0c;如上图所示&#xff0c;即面包屑导航控件&#xff0c;类似于TreeCtrl&#xff0c;但不是一次显示所有的Item&#xff0c;VC 2010可以编译通过&#xff0c;稍微修改一下其他的也可以编译&#xff0c;源代码下载&#xff1a; http://download.csdn.net…

foreach数组循环结构体

foreach数组循环结构体 foreach循环遍历任何维数的数组 Systemverilog增加了foreach循环&#xff0c;它可用来对一维或多维数组中的元素进行迭代&#xff0c;而不必指定数组每个维度的宽度。foreach循环的自变量是数组名&#xff0c;它后面是方括号内用逗号隔开的循环变量列表…

Android简介

最近Android很火&#xff0c;小弟也想了解一下它的结构。跟CE或者Mobile比起来&#xff0c;它的结构是有点凌乱&#xff0c;也难怪&#xff0c;毕竟是基于别人的内核在上层开发了一些应用 ---------------------------------------------------------------------------------…

说不尽的刘恒

认识刘恒快三十年了&#xff0c;作为曾经的同事和他小说的责任编辑&#xff0c;我只写过他一篇文章&#xff0c;还是在二十多年前。有时候特别熟悉的人反而不知道从何写起&#xff0c;因为一想起往事&#xff0c;各种记忆像开闸的水一样涌满眼前&#xff0c;让人很难落笔。1985…

印前处理的“发动机”——RIP

对于许多印刷厂来说&#xff0c;数字印前技术仍是个谜&#xff0c;尤其是光栅处理器RIP。除了RIP之处&#xff0c;我们也总听说打印机的内置控制单元。其实&#xff0c;RIP与内置控制单元在本质上是一样的&#xff0c;但也有所不同。听说起来好像有些玄乎&#xff0c;下面我们来…

组合逻辑过程块

组合逻辑过程块 always_comb代表组合逻辑 always_comb过程块表示建立组合逻辑模型 always_comb if(!mode)y a b; elsey a - b;always_comb能推断出其敏感表 与通用always过程块不同&#xff0c;always_comb块的后面不需要指明敏感表。软件工具已经知道设计的意图是建立一个…

外行看Flash的存储原理

突然在网上看到别人两年前写的一篇关于nor和nand的好文章&#xff0c;做为csdn的合法公民&#xff0c;有必要转 一、存储数据的原理 两种闪存都是用三端器件作为存储单元&#xff0c;分别为源极、漏极和栅极&#xff0c;与场效应管的工作原理相同&#xff0c;主要是利用电场的…

数码印刷

数码印刷 目前&#xff0c;RIP已经变成了印前生产的核心问题。它影响到从色彩和文件管理到印刷的整个生产过程的方方面面。而且&#xff0c;像陷印和拼大版这些以前需要单独的应用程序处理的功能&#xff0c;现在也被加到了RIP中。    新的RIP产品和销售商有很多。象Agfa,Ha…

锁存逻辑过程块

锁存逻辑过程块 always_latch描述锁存逻辑 always_latch过程块表示过程块描述的是基于锁存器的逻辑。和always_comb一样&#xff0c;always_latch的敏感表示推断出来的。 always_latchif(enable) q < d;always_latch与always_comb语义相同 always_latch过程的语义规则与al…

一种User Mode下访问物理内存及Kernel Space的简单实现

一种User Mode下访问物理内存及Kernel Space的实现 一&#xff0e;背景 WinCE发展到6.0之后&#xff0c;内存结构和管理方法进行了完善。对应用程序影响比较大的有Virtual Memory Layout的变化&#xff0c;如每个进程的虚拟内存空间扩展为2GB。对驱动程序影响比较大的有Pointe…

什么是ICC曲线

什么是ICC曲线 很多人都问及ICC曲线&#xff0c;但又都很不了解&#xff0c;甚至有些人认为有了ICC曲线就能打印出很漂亮的图片&#xff0c;其实不然&#xff0c;下面就我所掌握的知识给大家做个通俗的理解.国际色彩协会The international Color Consortium简称ICC&#xff0c;…

时序逻辑过程块

时序逻辑过程块 always_ff描述时序逻辑 always_ff专用过程块表示设计的意图是描述可综合的时序逻辑。 always_ff&#xff08;posedge clock,negedge resetN&#xff09;if(!resetN) q <0;else q < d;always_ff过程块的敏感表必须明确列出。这样就可以根据敏感表的内容&…

快速计算整数的二进制表示法中1的个数

快速计算整数的二进制表示法中1的个数 题目&#xff1a;给定一个无符号32位整数x&#xff0c;求x的二进制表示法中含1的个数&#xff1f; 第一种算法&#xff1a; int OneCount(unsigned int x){ for(int count0; x>0; count) x&x-1;//把最后面的1变0 return …

对任务和函数的改进

对任务和函数的改进 1、任务和函数的隐式语句组 SystemVerilog会推断出begin…end SystemVerilog简化了任务和函数的定义&#xff0c;有多条语句时不在需要begin …end对多条语句进行打包。打包省略之后&#xff0c;任务或函数中的语句将会顺序执行&#xff0c;就像仍然在begi…