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

相关文章

Flutter实现文字镂空效果的详细步骤

《Flutter实现文字镂空效果的详细步骤》:本文主要介绍如何使用Flutter实现文字镂空效果,包括创建基础应用结构、实现自定义绘制器、构建UI界面以及实现颜色选择按钮等步骤,并详细解析了混合模... 目录引言实现原理开始实现步骤1:创建基础应用结构步骤2:创建主屏幕步骤3:实现自定义绘制器步骤4:构建U

SpringBoot中四种AOP实战应用场景及代码实现

《SpringBoot中四种AOP实战应用场景及代码实现》面向切面编程(AOP)是Spring框架的核心功能之一,它通过预编译和运行期动态代理实现程序功能的统一维护,在SpringBoot应用中,AO... 目录引言场景一:日志记录与性能监控业务需求实现方案使用示例扩展:MDC实现请求跟踪场景二:权限控制与

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel

Java注解之超越Javadoc的元数据利器详解

《Java注解之超越Javadoc的元数据利器详解》本文将深入探讨Java注解的定义、类型、内置注解、自定义注解、保留策略、实际应用场景及最佳实践,无论是初学者还是资深开发者,都能通过本文了解如何利用... 目录什么是注解?注解的类型内置注编程解自定义注解注解的保留策略实际用例最佳实践总结在 Java 编程

一文教你Python如何快速精准抓取网页数据

《一文教你Python如何快速精准抓取网页数据》这篇文章主要为大家详细介绍了如何利用Python实现快速精准抓取网页数据,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以了解下... 目录1. 准备工作2. 基础爬虫实现3. 高级功能扩展3.1 抓取文章详情3.2 保存数据到文件4. 完整示例

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

解决IDEA报错:编码GBK的不可映射字符问题

《解决IDEA报错:编码GBK的不可映射字符问题》:本文主要介绍解决IDEA报错:编码GBK的不可映射字符问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录IDEA报错:编码GBK的不可映射字符终端软件问题描述原因分析解决方案方法1:将命令改为方法2:右下jav

Python实现微信自动锁定工具

《Python实现微信自动锁定工具》在数字化办公时代,微信已成为职场沟通的重要工具,但临时离开时忘记锁屏可能导致敏感信息泄露,下面我们就来看看如何使用Python打造一个微信自动锁定工具吧... 目录引言:当微信隐私遇到自动化守护效果展示核心功能全景图技术亮点深度解析1. 无操作检测引擎2. 微信路径智能获

使用Java将各种数据写入Excel表格的操作示例

《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

Python中pywin32 常用窗口操作的实现

《Python中pywin32常用窗口操作的实现》本文主要介绍了Python中pywin32常用窗口操作的实现,pywin32主要的作用是供Python开发者快速调用WindowsAPI的一个... 目录获取窗口句柄获取最前端窗口句柄获取指定坐标处的窗口根据窗口的完整标题匹配获取句柄根据窗口的类别匹配获取句