4 DICOM成像协议编码实现-元数据组解析

2024-04-04 10:32

本文主要是介绍4 DICOM成像协议编码实现-元数据组解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

以下链接是本系列文章,不足之处,可在评论区讨论:
系列文章

以下链接中的代码是完整的且可运行的,链接如下,可按需下载:
dicom成像程序

  本篇文章对应 专栏 从零讲解DICOM协议-成像协议中的文章DICOM成像协议剖析和DICOM成像协议实现思路,建议先看以上两篇文章以了解DICOM底层协议,有助于理解代码实现。

上篇文章DICOM成像协议编码实现-文件头解析讲解了DICOM解析引擎的实现思路和整体代码框架,并完成

  1. 读取DICOM文件至内存中
  2. 读取文件头到文件头对象中

本篇文章将继续进行以下几部分的代码思路讲解和实现:

  1. 按照元数据组特性读取元数据组中的各个DataElement
  2. 按照数据组特性读取数据组中的各个DataElement
  3. 如果PixelData是压缩格式,则用相应的解压算法解压
  4. CT值转BMP,保存BMP图像

依赖:

class DcmRead : public FileRead
{
public:DcmRead(bool iscompress = false);DcmRead(string path, bool iscompress = false);DcmRead(char *buffer, int len, bool iscompress = false);~DcmRead();
public:virtual bool ReadFile(string path) override;vector<ElementData> ParseFile(TagName tagname);ElementFormat GetTransferSyntax();//获得除了pixeldata以外的元素ElementValue GetElement(TagName tagname);ElementData GetPixelData();DcmToBmpTag GetBMPInfo();void SaveBMP(string path);virtual void Clear() override;bool IsCompress();
public:DcmTags * GetDcmTags();DcmFile *GetDcmFile();void SetClassify(DcmFilePeriod *dcmclassify);
private://dicom文件前128个0和4个"DICM"virtual void GetHead() override;//dicom文件0x0002组,元数据组virtual void GetMetadata() override;//dicom文件信息组virtual void GetInfodata(void *dcmelementset) override;//dicom所有元素//virtual void GetElements(void *dcmelementset) override;private://压缩图像数据ElementData UnCompressPixel(vector<ElementData> pixelset);vector<ElementData> GetUnCompressPixel();void TransferUnCompress();
private:void MemCopy(int len, int &datalen, char **data);void TagClassify(int startindex,vector<DcmElement> *dcmelementset);//判断是否是元数据组0x0002(显式小端)bool IsMetaData(char *data);//判断是否是SQbool IsSQ(char * data);int GetDataLen(char *data);//是否完整获取SQ序列bool IsSQEnd();//函数是否递归操作bool IsEmbed();void SetVrDataLen(char *data);void DeletePartPointer(DcmElement dcmelement);void DeletePointer(DcmElement dcmelement);void Init(bool iscompress);
private:void SetBinding();bool(DcmRead::*CompareModel)(char *source, char *des);int(DcmRead::*ElementIntValue)(char *buffer, int len);bool BigCompareModel(char *source, char *des);bool LittleCompareModel(char *source, char *des);int BigElementLen(char *buffer, int len);int LittleElementLen(char *buffer, int len);
private:void GetDcmClassifyInfo();void FileClassify();
private:bool status;bool isbigmodel;bool isin;bool iscompress;DcmFile dcmFile;int offset;//SQ序列嵌套层数int embednum;int vrlen;int datalen;ElementFormat ef;CompressFormat cf;ElementValueType et;vector<DcmElement> sqelementset;//Pipe *jpeg2k;SocketClient *sc;private:DcmTags *dcmtags;DcmFilePeriod *dcmclassify;
};

  前面文章讲过元数据组的特点是:
  元数据组是小端显式格式
  元数据组和数据组都是由DICOM协议中的DataElement结构组成
  DataElement由Tag,VR,Length,Value组成
  Tag由group和elment组成,VR有显式和隐式之分,Length所在字节由显隐式和VR类型决定,Value长度是偶数等
  元数据组中最重要的元素是传输语法(0002,0010) Transfer Syntax UID
根据以上特点,从DcmRead类中实现GetMetadata()函数

void DcmRead::GetMetadata()
{//元数据都是显式小端ef = ElementFormat::ExplicitLittle;int startindex = sizeof(dcmFile.Head) + sizeof(dcmFile.DcmFlag);TagClassify(startindex, &dcmFile.DcmElementSet);GetTransferSyntax();
}

  startindex为从元数据组在buffer内存指针的偏移地址
  TagClassify函数的主要内容为:
  for循环内判断是否是元数据组0002,循环执行

memcpy(dcmelement.tag, buffer + offset, sizeof(dcmelement.tag));
SetVrDataLen(dcmelement.tag);
MemCopy(vrlen, dcmelement.vr.len, &dcmelement.vr.data);
MemCopy(datalen, dcmelement.datalen.len, &dcmelement.datalen.data);
int len = (this->*ElementIntValue)(dcmelement.datalen.data, dcmelement.datalen.len);
MemCopy(len, dcmelement.data.len, &dcmelement.data.data);

  显式,VR长度根据27种类型判断

class ElementBytes
{
public:static const int ImplicitLenBytes = 4;//隐式VR,vr长度为0字节,len长度为4字节static const int ExplicitLenComBytes = 2;//显式VR,普通类型vr长度为2字节,len长度为2字节static const int ExplicitVrComBytes = 2;//显式VR,OB,OW,OF,SQ,UT,UN的vr长度4字节,len长度为4字节static const int ExplicitLenOtherBytes = 4;//显式VR,OB,OW,OF,SQ,UT,UN的vr长度4字节,len长度为4字节static const int ExplicitVrOtherBytes = 4;//显式VR,OB,OW,OF,SQ,UT,UN的vr长度4字节,len长度为4字节};
void DcmRead::SetVrDataLen(char *data)
{if (!IsSQ(data)){switch (ef){case ElementFormat::ImplicitLittle:vrlen = 0;datalen = ElementBytes::ImplicitLenBytes;break;case ElementFormat::ExplicitLittle:case ElementFormat::ExplicitBig:case ElementFormat::CompressPixel:char vrtype[2];memcpy(vrtype, buffer + offset, sizeof(vrtype));for (int j = 0; j < TagType::VrType.size(); j++){if (memcmp(vrtype, TagType::VrType[j], sizeof(vrtype)) == 0){vrlen = ElementBytes::ExplicitVrOtherBytes;datalen = ElementBytes::ExplicitLenOtherBytes;break;}if (j == TagType::VrType.size() - 1){vrlen = ElementBytes::ExplicitVrComBytes;datalen = ElementBytes::ExplicitLenComBytes;}}break;default:break;}}
}

  得到VR的长度和Length的长度后,从buffer内存指针中获取对应的数据,偏移地址累加

void DcmRead::MemCopy(int len, int &datalen, char **data)
{datalen = len;*data = new char[datalen];memcpy(*data, buffer + offset, datalen);offset += datalen;
}

  小端,ElementIntValue绑定小端转换函数

void DcmRead::SetBinding()
{if (ef == ElementFormat::ExplicitLittle || ef == ElementFormat::ImplicitLittle || ef == ElementFormat::CompressPixel){CompareModel = &DcmRead::LittleCompareModel;ElementIntValue = &DcmRead::LittleElementLen;}else if (ef == ElementFormat::ExplicitBig){CompareModel = &DcmRead::BigCompareModel;ElementIntValue = &DcmRead::BigElementLen;}
}
int DcmRead::LittleElementLen(char *buffer, int len)
{int data = 0;int flag = 0x000000FF;for (int i = 0; i < len; i++){data |= ((buffer[i] << i * 8) & (flag << i * 8));}return data;
}

传输语法获取

ElementFormat DcmRead::GetTransferSyntax()
{for (int i = 0; i < dcmFile.DcmElementSet.size(); i++){if (memcmp(dcmFile.DcmElementSet[i].tag, TagType::TransferSyntaxUID, sizeof(dcmFile.DcmElementSet[i].tag)) == 0){if (memcmp(dcmFile.DcmElementSet[i].data.data, TransferSyntaxType::ImplicitLittle.c_str(), dcmFile.DcmElementSet[i].data.len) == 0){ef = ElementFormat::ImplicitLittle;isbigmodel = false;}else if (memcmp(dcmFile.DcmElementSet[i].data.data, TransferSyntaxType::ExplicitLittle.c_str(), dcmFile.DcmElementSet[i].data.len) == 0){ef = ElementFormat::ExplicitLittle;isbigmodel = false;}else if (memcmp(dcmFile.DcmElementSet[i].data.data, TransferSyntaxType::ExplicitBig.c_str(), dcmFile.DcmElementSet[i].data.len) == 0){ef = ElementFormat::ExplicitBig;isbigmodel = true;}else //压缩格式都是显式小端{isbigmodel = false;ef = ElementFormat::CompressPixel;if (memcmp(dcmFile.DcmElementSet[i].data.data, TransferSyntaxType::Jpeg2000.c_str(), dcmFile.DcmElementSet[i].data.len) == 0){cf = CompressFormat::Jpeg2000;}else if (memcmp(dcmFile.DcmElementSet[i].data.data, TransferSyntaxType::Jpeglossless.c_str(), dcmFile.DcmElementSet[i].data.len) == 0){cf = CompressFormat::Jpeglossless;}}break;}}return ef;
}

  至此,元数据组解析框架已基本完成,为突出重点和整体处理流程,部分函数没有列出完整内容。
  下篇文章将进行DICOM数据组解析框架讲解。

这篇关于4 DICOM成像协议编码实现-元数据组解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/875543

相关文章

Python实现精确小数计算的完全指南

《Python实现精确小数计算的完全指南》在金融计算、科学实验和工程领域,浮点数精度问题一直是开发者面临的重大挑战,本文将深入解析Python精确小数计算技术体系,感兴趣的小伙伴可以了解一下... 目录引言:小数精度问题的核心挑战一、浮点数精度问题分析1.1 浮点数精度陷阱1.2 浮点数误差来源二、基础解决

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

Java实现远程执行Shell指令

《Java实现远程执行Shell指令》文章介绍使用JSch在SpringBoot项目中实现远程Shell操作,涵盖环境配置、依赖引入及工具类编写,详解分号和双与号执行多指令的区别... 目录软硬件环境说明编写执行Shell指令的工具类总结jsch(Java Secure Channel)是SSH2的一个纯J

使用Python实现Word文档的自动化对比方案

《使用Python实现Word文档的自动化对比方案》我们经常需要比较两个Word文档的版本差异,无论是合同修订、论文修改还是代码文档更新,人工比对不仅效率低下,还容易遗漏关键改动,下面通过一个实际案例... 目录引言一、使用python-docx库解析文档结构二、使用difflib进行差异比对三、高级对比方

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达

MyBatis-plus处理存储json数据过程

《MyBatis-plus处理存储json数据过程》文章介绍MyBatis-Plus3.4.21处理对象与集合的差异:对象可用内置Handler配合autoResultMap,集合需自定义处理器继承F... 目录1、如果是对象2、如果需要转换的是List集合总结对象和集合分两种情况处理,目前我用的MP的版本

HTTP 与 SpringBoot 参数提交与接收协议方式

《HTTP与SpringBoot参数提交与接收协议方式》HTTP参数提交方式包括URL查询、表单、JSON/XML、路径变量、头部、Cookie、GraphQL、WebSocket和SSE,依据... 目录HTTP 协议支持多种参数提交方式,主要取决于请求方法(Method)和内容类型(Content-Ty

深度解析Java @Serial 注解及常见错误案例

《深度解析Java@Serial注解及常见错误案例》Java14引入@Serial注解,用于编译时校验序列化成员,替代传统方式解决运行时错误,适用于Serializable类的方法/字段,需注意签... 目录Java @Serial 注解深度解析1. 注解本质2. 核心作用(1) 主要用途(2) 适用位置3

QT Creator配置Kit的实现示例

《QTCreator配置Kit的实现示例》本文主要介绍了使用Qt5.12.12与VS2022时,因MSVC编译器版本不匹配及WindowsSDK缺失导致配置错误的问题解决,感兴趣的可以了解一下... 目录0、背景:qt5.12.12+vs2022一、症状:二、原因:(可以跳过,直奔后面的解决方法)三、解决方

MySQL中On duplicate key update的实现示例

《MySQL中Onduplicatekeyupdate的实现示例》ONDUPLICATEKEYUPDATE是一种MySQL的语法,它在插入新数据时,如果遇到唯一键冲突,则会执行更新操作,而不是抛... 目录1/ ON DUPLICATE KEY UPDATE的简介2/ ON DUPLICATE KEY UP