TIF图像文件的读取(c++代码)

一 TIF图像介绍

    TIFF是最复杂的一种位图文件格式。TIFF是基于标记的文件格式,它广泛地应用于对图像质量要求较高的图像的存储与转换。由于它的结构灵活和包容性大,它已成为图像文件格式的一种标准,绝大多数图像系统都支持这种格式。

   TIFF 是一个灵活适应性强的文件格式,通过在文件头中包含“标签”它能够在一个文件中处理多幅图像和数据。标签能够标明图像的如图像大小这样的基本几何尺寸或者定义图像数据是如何排列的并且是否使用了各种各样的图像压缩选项。例如,TIFF可以包含JPEG和行程长度编码压缩的图像。TIFF文件也可以包含基于矢量的裁剪区域(剪切或者构成主体图像的轮廓)。使用无损格式存储图像的能力使TIFF文件成为图像存档的有效方法。与JPEG不同,TIFF文件可以编辑然后重新存储而不会有压缩损失。其它的一些TIFF文件选项包括多层或者多页。

  尽管现今它是一种被广泛接受的标准格式,当TIFF最初出现的时候,它的可扩展性带来了很多兼容问题。程序员可以随意定义新的标签和选项,但是并不是所有的实现程序都能支持这些所有这些创造出的标签。作为结果,它的一个最小特性集成为了“这个”TIFF,即使是在今天大量的TIFF文件和读取它们的代码都是基于简单的32位非压缩图像。

  TIFF有一个使用LZW压缩的选项,这是一种减小文件大小的无损技术,但是这项技术在不同的司法权限内为几个专利所涵盖。到了2005年,除了一个之外这些专利都已经到期,其中包括Unisys所拥有的广为人知又有很多争议的专利。另外一个著名的专利是IBM拥有的将在2006年8月11日到期的专利,IBM也没有要加强它的意思(who has shown no interest to date in enforcing it)。

  每个TIFF文件都是从指示字节顺序的两个字节开始的。“II”表示小字节在先、“MM”表示大字节在先字节顺序。后面的两个字节表示数字42。数字42是“为了其深刻的哲学意义"而选择的。 42的读法取决于头两个字节所表示的字节顺序。整个文件根据所指出的字节顺序进行读取。

  字节顺序在Apple Macintosh和微软视窗程序之间可能产生兼容性的问题,它们通常为TIFF文件使用不同的字节顺序。一些程序提供了保存为Mac或者是Windows字节顺序的选项以使文件能在交叉平台使用。

二TIF图像的读取

    1.TIF的存储结构

       TIFF文件以.tif为扩展名。其数据格式是一种3级体系结构,从高到低依次为:文件头、一个或多个称为IFD的包含标记指针的目录和数据。

    1.文件头(IFH)

  在每一个TIFF文件中第一个数据结构称为图像文件头或IFH,它是图像文件体系结构的最高层。这个结构在一个TIFF文件中是惟一的,有固定的位置。它位于文件的开始部分,包含了正确解释TIFF文件的其他部分所需的必要信息。

   

        IFH数据结构包含3个成员共计8个字节,Byte order成员可能是“MM”(0x4d4d)或“II”(0x4949),0x4d4d表示该TIFF图是摩托罗拉整数格式 0x4949表示该图是Intel整数格式;Version成员总是包含十进制42(0x2a),它用于进一步校验该文件是否为TIF格式,42这个数并 不是一般人 想象中的那样认为是tif软件的版本,实际上,42这个数大概永远不会变化;第三个成员是IFD(接下来要说的第二个数据结构)相对文件开始处的偏移量。

    2.图像文件目录(IFD)

  IFD是TIFF文件中第2个数据结构,它是一个名为标记(tag)的用于区分一个或多个可变长度数据块的表,标记中包含了有关于图像的所有信息。IFD提供了一系列的指针(索引),这些指针告诉我们各种有关的数据字段在文件中的开始位置,并给出每个字段的数据类型及长度。这种方法允许数据字段定位在文件的任何地方,且可以是任意长度,因此文件格式十分灵活。

     IFD是TIF图中最重要的数据结构,它包含了一个TIF文件中最重要的信息,一个TIF图可能有多个IFD,这说明文件中有多个图像,每个IFD标 识1个图像的基本属性。 IFD结构中包含了三类成员,Directory Entry Count指出该结构里面有多少个目录入口;接下来就是N个线性排列的DE序列,数量不定(这就是 为什么称TIF格式文件为可扩充标记的文件,甚至用户可以添加自定义的标记属性),每个DE标识了图像的某一个属性;最后就是一个偏移量, 标识下一个文件目录相对于文件开始处的位置,当然,如果该TIF文件只包含了一幅图像,那么就只有一个IFD,显然,这个偏移量就等于0;


共12个字节,见图二。简单说,一个DE就是一幅图像的某一个属性。例如图像的大小、分辨率、是否压缩、像素的行列数、一个像素由几位 表示(1位代表黑白两色,8位代表256色等等)等。其中:tag成员是该属性的编号,在图像文件目录中,它是按照升序排列的。我们可以通过读 这些编号,然后到TIF格式官方白皮书中查找相应的含义。属性是用数据来表示的,那么type就是代表着该数据的类型,TIF官方指定的有5种数据类型。type=1就是BYTE类型(8位无标记整数)、type=2是ASCII类型(7位ASCII码加1位二进制0)、type=3是SHORT类型 (16位无标记整数)、type=4是LONG 类型(32位无标记整数)、type=5是RATIONAL类型(2个LONG,第一个是分子,第二个是分母)。length成员是数据的数量而不是数据类型的长度。 第4个成员valueOffset很重要,它是tag标识的属性代表的变量值相对文件开始处的偏移量。如果变量值占用的空间小于4个字节,那么该值就存放在 valueOffset中即可,没必要再另外指向一个地方了。


      3.图像数据

  根据IFD所指向的地址.存储相关的图像信息

以下是在6.0下运行的代码

[cpp] view plaincopy
  1. #ifndef TIFREADER_H   
  2. #define TIFREADER_H   
  3.   
  4. #include <stdio.h>  
  5. #include <string.h>  
  6.   
  7. #ifndef NULL  
  8. #define NULL   0  
  9. #endif  
  10.   
  11. #ifndef TRUE  
  12. #define TRUE   1  
  13. #define FALSE  0  
  14. #endif  
  15.   
  16.   
  17. typedef struct    
  18. {  
  19.   unsigned short Byte_order;//  
  20.   unsigned short Version;//校验文件是否是TIF文件  
  21.   unsigned   int OffsetToFirstFID;//相对对文件开始处的偏移量  
  22.  // unsigned short wDECount;//多少目录入口  
  23. }IFH;  
  24.   
  25. typedef struct  
  26. {  
  27.     unsigned short tag;//属性的编号  
  28.     unsigned short type;//数据类型  
  29.     unsigned long length;//数据的数量  
  30.     unsigned long valueOffset;//tag标识的属性代表的变量值相对文件开始处的偏移量  
  31. }DE;  
  32. typedef struct    
  33. {  
  34.     int width;  
  35.     int height;  
  36.       
  37. }Size;  
  38. typedef struct   
  39.  {  
  40.     int *data;  
  41. }DATA;  
  42. typedef struct   
  43. {  
  44.     DE *pde;  
  45.     int wDECount;  
  46. }PDE;  
  47. bool readTIF(char* path,IFH &ifh,PDE &de,Size &size,DATA &Data)  
  48. {  
  49.     unsigned char *data;  
  50.     int *dat;  
  51.     unsigned short wDECount;//多少目录入口  
  52.       
  53. //  ZeroMemory(&ifh, sizeof(IFH));  
  54. //  ZeroMemory(&de, sizeof(DE));  
  55.   
  56.     FILE *fp;  
  57.     fp=fopen(path,"rb");  
  58.     if(fp == NULL)  
  59.     {  
  60.         cout<<"open file error"<<endl;  
  61.         return false;  
  62.     }  
  63.     if(sizeof(IFH) != fread(&ifh, 1,sizeof(IFH),fp))  
  64.     {  
  65.         cout<<"读TIF文件头失败";  
  66.         return FALSE;  
  67.     }  
  68.      if(0x2a != ifh.Version)  
  69.      {  
  70.           cout<<"该文件不是TIF格式,读文件失败";  
  71.           return FALSE;  
  72.      }  
  73.   
  74.      if(0x4949 != ifh.Byte_order)  
  75.      {  
  76.           cout<<"该TIF文件不是IBMPC字节序,读文件失败";  
  77.           return FALSE;  
  78.      }  
  79.   
  80.      fseek(fp,ifh.OffsetToFirstFID,SEEK_SET);//将文件指针定位到IFD  
  81.      //读文件有多少个目录入口   
  82.     if(2 != fread(&wDECount,1,sizeof(unsigned short),fp))  
  83.      {  
  84.           cout<<"无法获得TIF文件目录入口数量";  
  85.           return FALSE;  
  86.      }  
  87.     cout<<"该TIF文件有"<<wDECount<<"个目录入口"<<endl;  
  88.     //创建DE数组,接收信息,数组中有wDECount个元素   
  89.     de.pde = new DE[wDECount];  
  90.     DE* pTemp = de.pde;  
  91.     de.wDECount = wDECount;  
  92.     memset(de.pde, 0, sizeof(DE)*wDECount);  
  93.      if(sizeof(DE)*wDECount != fread(de.pde,1, sizeof(DE)*wDECount,fp))  
  94.      {  
  95.           cout<<"读图象文件目录失败";  
  96.           delete []de.pde;  
  97.           return false ;  
  98.      }  
  99.     //把图像的大小和图像数据的容量保存到成员变量中  
  100.      int m_size_x;  
  101.      int m_size_y;  
  102.      int m_size;  
  103.      int i;  
  104.     for(i=0; i<wDECount; i++)  
  105.      {  
  106.           pTemp =de.pde + i;  
  107.           if(256 == pTemp->tag) //tag为256的目录入口中的变量标识了图象宽度  
  108.           {  
  109.               m_size_x = pTemp->valueOffset;  
  110.           }  
  111.           if(257 == pTemp->tag) //图象高度  
  112.           {  
  113.               m_size_y = pTemp->valueOffset;  
  114.           }  
  115.           if(273 == pTemp->tag) //计算图象数据占用字节数  
  116.           {  
  117.            //m_dwBmSize = pTemp->valueOffset - sizeof(IFH);  
  118.            //或者把tag=256的valueOffset乘以tag=257的valueOffset  
  119.               m_size = m_size_x * m_size_y;  
  120.           }  
  121.     }  
  122.     //填充所有像素数据, 颠倒图象数据从最后一行开始读起  
  123.      int j = 0;  
  124.     // int i = 0;  
  125.      data = (unsigned char*)malloc(m_size*sizeof(BYTE));  
  126.      dat = (int*)malloc(m_size*sizeof(int));  
  127.      for(i=m_size_y-1; i>=0; i--)  
  128.      {  
  129.          fseek(fp,sizeof(IFH) + i*m_size_x, SEEK_SET);  
  130.          fread((BYTE*)(data + 1) + j*m_size_x,sizeof(BYTE), m_size_x,fp);  
  131.           j++;  
  132.      }  
  133.      cout<<"width:"<<m_size_x<<endl;  
  134.      cout<<"height:"<<m_size_y<<endl;  
  135.      unsigned char* p;  
  136.      p = data;  
  137.      int *ptr;  
  138.      ptr = dat;  
  139.      for (i=0;i<m_size;i++,p++,ptr++)  
  140.      {  
  141.          *ptr = (int)(*p);  
  142.          int  h= *ptr;  
  143.         // cout<<h<<" ";  
  144.      }  
  145.      size.width = m_size_x;  
  146.      size.height = m_size_y;  
  147.      Data.data = dat;  
  148.      return TRUE;  
  149.     
  150. }  
  151. bool saveTIF(char* path,IFH ifh,PDE de,Size size,DATA Data)  
  152. {  
  153.     unsigned char *data;  
  154. //    unsigned short wDECount;//多少目录入口  
  155.       
  156. //  ZeroMemory(&ifh, sizeof(IFH));  
  157. //  ZeroMemory(&de, sizeof(DE));  
  158.   
  159.     FILE *fp;  
  160.     fp=fopen(path,"wb");  
  161.     if(fp == NULL)  
  162.     {  
  163.         cout<<"open file error"<<endl;  
  164.         return false;  
  165.     }  
  166.     if(sizeof(IFH) != fwrite(&ifh, 1,sizeof(IFH),fp))  
  167.     {  
  168.         cout<<"写TIF文件头失败";  
  169.         return FALSE;  
  170.     }  
  171.      fseek(fp,ifh.OffsetToFirstFID,SEEK_SET);//将文件指针定位到IFD  
  172.      //读文件有多少个目录入口   
  173.     if(2 != fwrite(&de.wDECount,1,sizeof(unsigned short),fp))  
  174.      {  
  175.           cout<<"无法获得TIF文件目录入口数量";  
  176.           return FALSE;  
  177.      }  
  178.     //创建DE数组,接收信息,数组中有wDECount个元素   
  179.    
  180.      if(sizeof(DE)*de.wDECount != fwrite(de.pde,1, sizeof(DE)*de.wDECount,fp))  
  181.      {  
  182.           cout<<"读图象文件目录失败";  
  183.           return false ;  
  184.      }  
  185.     //填充所有像素数据, 颠倒图象数据从最后一行开始读起  
  186.      int j = 0;  
  187.      int i = 0;  
  188.      data = (unsigned char*)malloc(size.width*size.height*sizeof(BYTE));  
  189.      int *ptr = Data.data;  
  190.   
  191.      unsigned char* p;  
  192.      p = data;  
  193.      for (i=0;i<size.width*size.height;i++,p++,ptr++)  
  194.      {  
  195.          *p = (unsigned char)(*ptr);  
  196.         // int  h= *ptr;  
  197.         // cout<<h<<" ";  
  198.      }    
  199.      for(i=size.height-1; i>=0; i--)  
  200.      {  
  201.          fseek(fp,sizeof(IFH) + i*size.width, SEEK_SET);  
  202.          fwrite((BYTE*)(data + 1) + j*size.width,sizeof(BYTE), size.width,fp);  
  203.           j++;  
  204.      }  
  205.   
  206.      return TRUE;  
  207.     
  208. }  
  209. #endif  


这个代码这是对特定的TIF进行读取,在保存时不知道为什么,不能再windows下正常显示,期待大侠解决呀

转载于:https://www.cnblogs.com/ainima/p/6331168.html

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

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

相关文章

g menu i meun_长沙话读“这里”,到底是阁(gó)里还是该(gái)里

“带笼子”、“打抱秋”……这些地道的长沙话&#xff0c;长沙人&#xff0c;你有多久没听过了&#xff1f;/ 长沙人&#xff0c;你还记得长沙话吗 / “去了很多地方&#xff0c;最后还是回到了长沙”“我听见了一句长沙话&#xff0c;就想回长沙了。”逗霸妹听过很多人回长沙的…

git使用---工作区和暂存区

转载于:https://www.cnblogs.com/momo-unique/articles/4380551.html

偶数哥德巴赫猜想

已知不小于6的偶数都可以分成两个素数之和。请编写6到100000的所有偶数的分解&#xff0c;若有一个偶数可以分解成多个素数之和&#xff0c;只需写出一种即可。 #include <iostream> #include <algorithm> using namespace std;bool isprime(int n)//判断素数{int …

电脑常见故障处理_彩超常见故障及维修

彩超是医学检测手段中重要的环节之一&#xff0c;是对产妇以及对病人进行内部组织和结构观察的重要方式之一&#xff0c;彩超应用得当可以及早的诊断出病人的疾病&#xff0c;为患者解除疾病的困扰。彩超设备是一种极为先进的诊断系统&#xff0c;一般彩超系统包括以下几个部分…

用基本信号画出如下的信号_股市入门基本知识丨下跌时期可以抄底的安全信号有哪些...

点击蓝色字体 关注我们 带来更多精彩股票市场中的秘籍其实就是“低买高卖”&#xff0c;不过我们不能在大盘一开始下跌的时候就进行买入&#xff0c;因为不清楚下跌的时间&#xff0c;太早介入&#xff0c;只有在反弹幅度超出我们介入的点的时候才可以进行高卖。那么什么时候才…

Flume数据传输事务分析[转]

本文基于ThriftSource,MemoryChannel,HdfsSink三个组件&#xff0c;对Flume数据传输的事务进行分析&#xff0c;如果使用的是其他组件&#xff0c;Flume事务具体的处理方式将会不同。一般情况下&#xff0c;用MemoryChannel就好了&#xff0c;我们公司用的就是这个&#xff0c;…

yii2中的rules验证规则

2019独角兽企业重金招聘Python工程师标准>>> Rules验证规则&#xff1a;required : 必须值验证属性||CRequiredValidator 的别名, 确保了特性不为空.[[字段名],required,requiredValue>必填值,message>提示信息];email : 邮箱验证||CEmailValidator 的别名,确…

opengl 如何加阴影_动漫嘴唇厚涂如何绘制?厚涂嘴唇正确画法

动漫嘴唇厚涂如何绘制&#xff1f;厚涂嘴唇正确画法&#xff01;嘴巴怎么画&#xff1f;画嘴巴真的很考验一个画师功力&#xff0c;好看的嘴巴生动而丰满&#xff0c;可以给整幅画作添上亮点&#xff0c;而画的不好的嘴巴呢&#xff0c;就容易把画面整体的风格打破。那么零基础…

详解JMeter函数和变量

详解JMeter函数和变量&#xff08;1&#xff09; JMeter函数可以被认为是某种特殊的变量&#xff0c;它们可以被采样器或者其他测试元件所引用。函数调用的语法如下&#xff1a; ${__functionName(var1,var2,var3)} 其中&#xff0c;__functionName匹配被调用的函数名称。用圆括…

hdu 5199 map或二分或哈希

题目描述&#xff1a;给出n棵树的高度&#xff0c;每棵树上都站着一只鸟&#xff0c;枪手Jack站在最左边那棵树的左边对鸟进行射击&#xff0c;当Jack在高度为H的地方向右发射一颗子弹的时候&#xff0c;高度为H的树上的鸟儿就会掉落&#xff08;注&#xff1a;其他树上的鸟儿不…

数字电路实验怎么接线视频讲解_家庭影院中音箱、功放、投影机、4K播放机不知道怎么连接?手把手教你...

家庭影院中音箱、功放、投影机、4K播放机不知道怎么连接&#xff1f;手把手教你有不少用户收到从家庭影院器材之后&#xff0c;表示完全不会连接。翻看说明书也觉得头大&#xff0c;知识太多&#xff0c;然而却很难找到要点。今天主要跟大家讲讲如何连接音箱、功放、投影机和影…

.NET开发过程中的全文索引使用技巧之Solr

前言&#xff1a;相信许多人都听说过.net开发过程中基于Lucene.net实现的全文索引&#xff0c;而Solr是一个高性能&#xff0c;基于Lucene的全文搜索服务器。同时对其进行了扩展&#xff0c;提供了比Lucene更为丰富的查询语言&#xff0c;同时实现了可配置、可扩展并对查询性能…

auto.js停止所有线程_Java线程与并发编程实践:深入理解volatile和final变量

同步有两种属性&#xff1a;互斥性和可见性。synchronized关键字与两者都有关系。Java同时也提供了一种更弱的、仅仅包含可见性的同步形式&#xff0c;并且只以volatile关键字关联。假设你自己设计了一个停止线程的机制(因为无法使用Thread不安全的stop()方法))。清单1中Thread…

项目实例改编:利用structs2的action 实时显示图片、pdf和其他内容的框架抽取。(转)...

转自&#xff1a;http://www.verydemo.com/demo_c167_i1382.html 针对&#xff1a;预览文件&#xff08;图片&#xff0c;PDF&#xff09;文件来源为action中的inputStream 重点&#xff1a; structs2的action的配置 action的写法和结果类型 resulttype的写法 网页上实…

开始Go开发之旅-Golang架构师之路系列实战

2019独角兽企业重金招聘Python工程师标准>>> 作者: gomaster.me(冯琪超) 系列:Golang架构师之路 巧妇难做无米之炊&#xff0c;golang sdk就是gopher的大米 下载golang 点击 官网下载golang sdk 根据不同系统&#xff0c;官网下载链接会选择相应的平台进行链接跳转&…

安卓9.0官方系统升级包_华为、荣耀公布可升级安卓10.0机型,你的手机在名单之内吗?...

在近两个月以前&#xff0c;美方将华为关进了小黑屋&#xff0c;随后谷歌也将华为旗下的机型移出了安卓10.0升级名单&#xff0c;这一波操作之后&#xff0c;引起了不小的“恐慌”&#xff0c;许多华为用户也在担心是否还能正常使用安卓系统服务&#xff0c;不过&#xff0c;让…

2. Mysql数据库的入门知识

2. Mysql数据库的入门知识 &#xff08;1&#xff09;打开Windows系统提供的服务查看相应的服务。 &#xff08;2&#xff09;在Windows任务管理器的进程中查看 &#xff08;3&#xff09;使用命令行管理windows的Mysql数据库服务。 Net start 服务名 Net stop 服务名 mysql -h…

nginx php-fpm 输出php错误日志(转)

nginx是一个web服务器&#xff0c;因此nginx的access日志只有对访问页面的记录&#xff0c;不会有php 的 error log信息。 nginx把对php的请求发给php-fpm fastcgi进程来处理&#xff0c;默认的php-fpm只会输出php-fpm的错误信息&#xff0c;在php-fpm的errors log里也看不到ph…

win7优化设置_win7蓝牙怎么打开?

当电脑需要连接蓝牙设备的时候&#xff0c;就需要打开蓝牙设置才行。鉴于一些win7的用户还不知道蓝牙功能在哪&#xff0c;win7蓝牙怎么打开&#xff0c;故系统圣地分享本篇教程。1、win7蓝牙怎么打开?首先要你的电脑支持蓝牙功能。如果你的电脑有蓝牙功能的话那么在电脑的右下…

Doxygen从零学起———安装和配置

Doxygen可以为多种语言生成说明文档&#xff08;从程序的源代码中提取其中按照约定格式写的注释中提取信息&#xff09; 例如C, Objective-C, C#, C, PHP, Python, IDL (Corba, Microsoft, and UNO/OpenOffice flavors), Fortran, VHDL, Tcl, D ,从这期开始&#xff0c;我将系…