背景:

专栏之前对Transfer Syntax(暂定中文翻译为传输语义8月初博客中提到的DICOM3.0标准中文版开源书籍计划顺利启动。兴许会面临诸多专有名词的翻译工作,欢迎广大博友提意见)进行过多次的介绍。在DICOM医学图像处理:DICOM网络传输中差别过Abstract Syntax与Transfer Syntax,在DICOM:dcmqrscp.exe与storescu.exe中C-STORE服务的差别中介绍过在网络服务中Transfer Syntax的作用。以及上一篇DICOM:dcm4che工具包怎样压缩dcm文件探讨(续篇)中介绍对dcm文件进行压缩时提到的JPEG LossLess压缩语义以及Implicit VR Little Endian。

Transfer Sytanx在DICOM标准中占有重要的一席之地,既作为必要元素写入到DCM文件元信息(MetaInformation)中,又是DICOM网络服务中两方传输数据的前提(如博文DICOM医学图像处理:DICOM网络传输描写叙述TransferSyntaxx是PresentationContext必备元素)。

题记:

最近收到了GE一款设备的数据,使用dcm4che3提供的StoreSCP服务起初无法识别,在开启“–accept-known”后顺利完毕接收,但数据打开后图像“失真”——与原始图像全然不同。因此特意研究了一下GE私有的Transfer Syntax。即1.2.840.113619.5.2。以下让我们看看这奇葩的私有协议。

dcm4che3的StoreSCP:

dcm4che3提供的StoreSCP服务在未开启“–accept-known”选项时,仅仅支持sop-classes.properties文件里的标准,开启后能够接受其它未知协议。

通过查看StoreSCP.java源代码能够看到对未知协议相同採用直接将数据流存储到文件里。

    private void storeTo(Association as, Attributes fmi,
PDVInputStream data, File file) throws IOException {
LOG.info("{}: M-WRITE {}", as, file);
file.getParentFile().mkdirs();
DicomOutputStream out = new DicomOutputStream(file);
try {
out.writeFileMetaInformation(fmi);
data.copyTo(out);
} finally {
SafeClose.close(out);
}
}

将StoreSCP的“–accept-known”选项开启后。在本地顺利接收到了GE设备的数据。依照上述代码所看到的直接存储到了文件里。可是图像显示结果失真。

图像失真:

使用DICOM Viewer打开数据。结果例如以下所看到的:



即使调节窗宽窗位,也无法顺利显示各种组织。通过查看DICOM相关信息并未找出问题。



相同採用直接查看二进制的方法。定位到PixelData元素能够看到真实像素数据数值为FA 24。依照GE Private TransferSyntax=1.2.840.113619.5.2的说明,该私有语义是Implicit VR - Big Endian (G.E Private)假设依照Big Endian来解析像素数据为0xFA24=-1500,假设依照Little Endian像素数据为0x24FA=9466.在Sante Editor查看背景数据显示为9466.由此能够推測问题出如今PixelData数据读取有误。即Sante Editor对GE Private TransferSyntax=1.2.840.113619.5.2理解有误。





奇葩之GE Private TS:

利用Wireshark抓取GE设备发送到StoreSCP的数据包。依据Wireshark中对DICOM协议数据包的提示。发现GE Private TransferSyntax的确非常奇葩,例如以下所看到的:



言外之意,GE Private TransferSyntax私有传输语义仅仅对PixelData元素採用Big Endian进行处理。对于其它非PxielData元素依旧採用Implicit VR Little Endian,即GE Private TransferSyntax对标准Implicit VR Little Endian语义所做的改动仅限于PixelData数据。



结合之前StoreSCP.java中的源代码可知。dcm4che3的StoreSCP通过开启“–accept-known”选项尽管能够接受GE Private TransferSyntax私有语义。可是并未真正理解当中的含义。而是简单的将1.2.840.113619.5.2写入到MetaInformation元信息中。而将PixelData直接复制到文件流中。加之多数DICOM Viewer无法顺利理解GE Private TransferSyntax。因此导致在解析PixelData时依照之前大多数元素的方式直接以Implicit VR Little Endian语义读。导致图像失真

解决方式:

參照GE的说明文档,了解GE Private TransferSyntax私有语义后可知,除了PixelData的存储顺序是Big Endian以外,其它元素GE私有语义与Implicit VR Little Endian标准默认语义没有差别。

因此考虑到DICOM Viewer的兼容性问题。在StoreSCP.java的storeTo函数中对GE Private TransferSyntax私有语义进行单独处理,对PixelData进行转序处理就可以。在完毕PixelData的Big Endian到Little Endian转序后。也就能够将Transfer Syntax直接由GE Private TransferSyntax改成Implicit VR Little Endian

示范代码例如以下所看到的:

    private void storeTo(Association as, Attributes fmi, PDVInputStream data,
File file) throws IOException {
LOG.info("{}: M-WRITE {}", as, file);
file.getParentFile().mkdirs(); boolean bExchange=false;
// TransferSyntax=1.2.840.113619.5.2,is GE Private TS,
// Implicit VR Little Endian for all elements except pixel Data, which is Big Endian
if(fmi.getString(Tag.TransferSyntaxUID)=="1.2.840.113619.5.2")
{
fmi.setString(Tag.TransferSyntaxUID, VR.UI, "1.2.840.10008.1.2");
bExchange=true;
}
DicomOutputStream out = new DicomOutputStream(file);
try {
out.writeFileMetaInformation(fmi);
data.copyTo(out); } finally {
SafeClose.close(out);
if(bExchange)
{
//这里应该对于GE Private进行单独推断。将Pixel Data数据由Big Endian转换成Little Endian try {
DicomInputStream input=new DicomInputStream(file);
Attributes attrs=input.readDataset(-1, -1);
byte[] bytes=attrs.getBytes(2145386512);
byte[] newBytes=new byte[bytes.length];
for(int i=0;i<bytes.length/2;++i)
{
newBytes[2*i]=bytes[2*i+1];
newBytes[2*i+1]=bytes[2*i];
}
// //或者直接交换
// for(int i=0;i<bytes.length;i+=2)
// {
// byte swap=bytes[i];
// bytes[i]=bytes[i+1];
// bytes[i+1]=swap;
// }
attrs.setBytes(2145386512, VR.OW, newBytes);
File file2=new File("c:\\GE2.dcm");
DicomOutputStream out2=new DicomOutputStream(file2);
out2.writeDataset(input.getFileMetaInformation(), attrs);
input.close();
out2.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} }
}

改动后又一次打开新接收的数据,能够看到正确的图像例如以下所看到的:



至此,GE Private TransferSyntax私有语义的奇葩问题就攻克了。

作者:zssure@163.com

时间:2015/08/03

DICOM:Transfer Syntax传输语义之奇葩GE Private TS的更多相关文章

  1. DICOM:DICOM标准学习路线图(初稿)

    题记: DICOM医学图像处理专栏撰写已有两个年头,积累了近百篇文章.起初只是用于记录自己科研.工作中遇到的疑难问题,专注于图像处理(主要是医学图像,这也正是专栏名称最初的由来):后来逐渐延伸到了DI ...

  2. Dicom格式文件解析器

    转自:http://www.cnblogs.com/assassinx/archive/2013/01/09/dicomViewer.html Dicom全称是医学数字图像与通讯,这里讲的暂不涉及通讯 ...

  3. Dicom格式文件解析器[转]

    Dicom格式文件解析器   Dicom全称是医学数字图像与通讯,这里讲的暂不涉及通讯那方面的问题 只讲*.dcm 也就是diocm格式文件的读取,读取本身是没啥难度的 无非就是字节码数据流处理.只不 ...

  4. Dicom Conformance

    Platform Compatibility of DICOM Transfer Syntax   1.2.840.10008.1.2 Implicit VR - Little Endian yes ...

  5. DICOM:DICOM3.0网络通信协议

    转载:http://blog.csdn.net/zssureqh/article/details/41016091 背景: 专栏取名为DICOM医学图像处理原因是:博主是从医学图像处理算法研究时开始接 ...

  6. DICOM医学图像处理:DIMSE消息发送与接收“大同小异”之DCMTK fo-dicom mDCM

    背景: 从DICOM网络传输一文开始,相继介绍了C-ECHO.C-FIND.C-STORE.C-MOVE等DIMSE-C服务的简单实现,博文中的代码给出的实例都是基于fo-dicom库来实现的,原因只 ...

  7. DICOM医学图像处理:开源库mDCM与DCMTK的比較分析(一),JPEG无损压缩DCM图像

    背景介绍: 近期项目需求,须要使用C#进行最新的UI和相关DICOM3.0医学图像模块的开发.在C++语言下,我使用的是应用最广泛的DCMTK开源库,在本专栏的起初阶段的大多数博文都是对DCMTK开源 ...

  8. DICOM简介

    背景: DICOM分为两大类(这里只是从DICOM相关从业者日常工作角度出发来分类的):1)DICOM医学图像处理,即DCM文件中具体数据的处理,说图像可能有些狭隘,广义上还包括波形(心电).视频(超 ...

  9. 闲话Dicom

    最近在准备一场有关DICOM应用的讲座,整理了一下思路.想了几个问题,发现挺有意思的,想与大家共同分享.接触过DICOM,应该了解普通DICOM 文件包含的四级属性,病人,检查,序列,影像.每一级别需 ...

随机推荐

  1. python内置的一些模块

    logging模块: 默认情况下,logging将日志打印到屏幕,日志级别为WARNING:日志级别大小关系为:CRITICAL > ERROR > WARNING > INFO & ...

  2. HDU-4221 Greedy? 贪心 从元素的相对位置开始考虑

    题目链接:https://cn.vjudge.net/problem/HDU-4221 题意 给n个活动,每个活动需要一段时间C来完成,并且有一个截止时间D 当完成时间t大于截止时间完成时,会扣除t- ...

  3. Android APP弱网测试问题和解决分析

    最近做了一次移动APP的弱网和中断测试,接下来分享一下遇到的一些问题: 1.现象:用户登录应用时下载初始化数据,下载过程中因网速太慢点击取消并重新登录,数据初始化完成后出现重复,造成数据不一致. 原因 ...

  4. 【BZOJ 1293】[SCOI2009]生日礼物

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 显然的滑动窗口题. (尺取法 如果l..i这一段已经有k种珍珠了. 那么就尝试把l++; (即把l这个影响尝试去掉一下 如果不足k种 ...

  5. 【codeforces 22C】 System Administrator

    [题目链接]:http://codeforces.com/problemset/problem/22/C [题意] 给你n个点; 要求你构造一个含m条边的无向图; 使得任意两点之间都联通; 同时,要求 ...

  6. maven跳过单元测试-maven.test.skip和skipTests的区别以及部分常用命令

    -DskipTests,不执行测试用例,但编译测试用例类生成相应的class文件至target/test-classes下. -Dmaven.test.skip=true,不执行测试用例,也不编译测试 ...

  7. ios开发经常使用到的第三方库

    由于iOS SDK相对照较底层,所以开发人员就得受累多做一些体力活.只是幸运的是,有非常多第三方的类库能够用来简化非常多不必要的工作.经过作者团队的谨慎讨论.他们 评选出了10款可以极大提高iOS开发 ...

  8. hdu 1978 记忆化搜索

    注意: dp[i][j] 表示(i,j)这个点有多少种方式       mark[i][j]表示这个点是否走过  假设有直接返回dp[i][j]    dp的求法为全部梦走到点的dp的和 注意mark ...

  9. android 自己定义dialog并实现失去焦点(背景透明)的功能

    前言:因为在项目中须要用到更新显示动画的需求,所以想到了dialog,自己定义dialog不难.网上教程非常多,可是在实现dialog背景透明的需求时,遇到了一点问题.网上的一些方法在我的机器上并没有 ...

  10. NHibernate概括

    什么是?NHibernate?NHibernate是一个面向.NET环境的对象/关系数据库映射工具. 对象/关系数据库映射(object/relational mapping,ORM)这个术语表示一种 ...