5

我正在开发一个能够为 PACS 创建结构化报告的系统。

显然,为了创建一个 DICOM 实例(包含报告数据的文件),我需要三个 UID 用于 Study、Series 和 Instance。StudyUID 和 SeriesUID 必须与为其创建报​​告的研究和系列的相同。但是对于 SOPInstanceUID,我需要生成新的 UID。

getNewSOPInstanceUID在 Pixelmed 文档中看到了方法,但我不熟悉 Pixelmed 源。我需要一个算法或 Python 源代码。

4

4 回答 4

12

DICOM 中有两种创建 UID 的方法。一种基于已注册的 UID 根,另一种基于 UUID。后一种方法于 2012 年通过 CP-1156 添加到 DICOM 标准中。可以通过将 UUID 转换为 DICOM UID 来创建学习 UID、系列 UID、SOP 实例 UID 等内容的 UID。

大多数编程语言都支持创建 UUID。下面的示例代码代码基于 GUID 值在 C# 中创建有效的 DICOM UID。

public static string GuidToUidStringUsingStringAndParse(Guid value)
{
    var guidBytes = string.Format("0{0:N}", value);
    var bigInteger = BigInteger.Parse(guidBytes, NumberStyles.HexNumber);
    return string.Format(CultureInfo.InvariantCulture, "2.25.{0}", bigInteger);
}

下面的方法做同样的事情,但快了大约 5 倍:

public static string ConvertGuidToUuidInteger(ref Guid value)
{
    // ISO/IEC 9834-8, paragraph 6.3 (referenced by DICOM PS 3.5, B.2) defines how
    // to convert a UUID to a single integer value that can be converted back into a UUID.

    // The Guid.ToByteArray Method returns the array in a strange order (see .NET docs),
    // BigInteger expects the input array in little endian order.
    // The last byte controls the sign, add an additional zero to ensure
    // the array is parsed as a positive number.
    var octets = value.ToByteArray();
    var littleEndianOrder = new byte[]
    { octets[15], octets[14], octets[13], octets[12], octets[11], octets[10], octets[9], octets[8],
        octets[6], octets[7], octets[4], octets[5], octets[0], octets[1], octets[2], octets[3], 0 };

    return "2.25." + new BigInteger(littleEndianOrder).ToString(CultureInfo.InvariantCulture);
}
于 2013-01-04T22:20:18.257 回答
6

有关 DICOM UID 的更多详细信息,请参阅答案。

A] 增加计数器 [不推荐]

一个简单的逻辑是获取您的 SeriesInstanceUID 并将其递增1。假设您的 SeriesInstanceUID 是“1.1.1.1.1”,那么您的 SOPInstanceUID 可能是“ 1.1.1.1.2 ”或“1.1.1.1.1.1 ”。

问题:

  • 当实例被删除并创建下一个实例时,不应使用较早的计数器。
  • 在多线程环境中,应该足够小心。
  • 不保证跨不同系统/应用程序的唯一性。

B] 日期时间 [不推荐]

通常使用的其他技术是将时间戳(带有刻度)附加到组织根。

问题:

  • 多线程环境是个问题。
  • 系统时钟可能会关闭。
  • 无法保证跨不同系统/应用程序的唯一性。

C] 更复杂 [推荐]

1.2.840.xxxxx.30.152.99999.235.20.100.yyyyMMddHHmmss.zzzzzz

在哪里:

1.2.840.xxxxx:组织根
30:应用程序 ID
152:应用程序版本
99999:安装/位置 ID
235:研究 ID
20:序列号
100:图像编号
yyyyMMddHHmmss:日期时间
zzzzzz:线程安全计数器/随机数

问题:

  • 如果系统时钟关闭,算法可能会失败;这进一步受到线程安全计数器/随机数的保护。遥远的可能性;需要小心。

D] UUID 派生的 UID [推荐]

UID 可以从根“2.25”生成。后跟通用唯一标识符 (UUID) 的十进制表示。

问题:

  • 这可能适用于动态创建的 UID,例如 SOP 实例 UID,但不适用于在设计期间确定的 UID,例如私有 SOP 类或传输语法 UID,或实现类 UID。
  • UID 仅限于 128 位。DICOM UID 支持范围更广。
于 2017-09-20T08:12:22.830 回答
2

根据DICOM 标准(PS 3.5-2011 第 61 页),您需要一个 orgroot 和一个后缀。可以在此处找到示例(PS 3.5-2011 第 83 页)。

也不要忘记 UI 字段必须用 '\0' 字节而不是空格填充(如果它们没有偶数长度)。

我建议像这样创建 UID:

YOUR_ORGROOT.Year.Month.Day.Hour.Minute.Second.Milliseconds.StaticCounter

请注意,限制为 64 个字符!

于 2012-04-25T12:45:38.080 回答
-1

我真的建议您不要自己实施它。如今,大多数语言都提供了 UUID 库,不要重新发明轮子。特别是。如果你要编写代码来提取 MAC 地址,用可移植的C语言编写它可能非常复杂。

UUID并不完全符合 DICOM 定义,因此您需要注册自己的 Organization Root UID,然后只需使用生成的 UUID 填充即可,该 UUID 会带来空间和时间唯一性条件。

YOUR_ORG_ROOT.CONVERTED_UUID

请注意,您有 64 个字节(已经足够了,请参见此处)用于存储在值表示 UI 中:

  • 将 UUID 的十六进制表示法转换为 VR:UI 定义([0-9.]+)
  • 小心修剪(您可能会在此操作期间引入冗余)
  • 选择一个短Org Root
  • 如果需要,用\0(0 二进制)填充。

最后,由于您使用的是 python,请使用 uuid lib python-uuid


以上应被视为标准中正式定义的替代实现:

当直接将 UUID 转换为 UID 时,必须使用“2.25”。根。

于 2013-01-07T12:10:54.660 回答