Dcm4Che - 从pacs获取图像

时间:2014-03-02 11:06:57

标签: java dicom

我遇到了以下问题。我必须编写连接到pacs并获取图像的小应用程序。我决定使用dcm4che工具包。我写了以下代码:    公共课Dcm4 {

/**
 * @param args the command line arguments
 */
public static void main(String[] args) {
    // TODO code application logic here
    DcmQR dcmqr = new MyDcmQR("server");

    dcmqr.setCalledAET("server", true);
    dcmqr.setRemoteHost("213.165.94.158");
    dcmqr.setRemotePort(104);
    dcmqr.getKeys();

    dcmqr.setDateTimeMatching(true);
    dcmqr.setCFind(true);
    dcmqr.setCGet(true);

    dcmqr.setQueryLevel(MyDcmQR.QueryRetrieveLevel.IMAGE);

    dcmqr.addMatchingKey(Tag.toTagPath("PatientID"),"2011");
    dcmqr.addMatchingKey(Tag.toTagPath("StudyInstanceUID"),"1.2.276.0.7230010.3.1.2.669896852.2528.1325171276.917");
    dcmqr.addMatchingKey(Tag.toTagPath("SeriesInstanceUID"),"1.2.276.0.7230010.3.1.3.669896852.2528.1325171276.916");


    dcmqr.configureTransferCapability(true);
    List<DicomObject> result=null;
    byte[] imgTab=null;
    BufferedImage bImage=null;
    try {
        dcmqr.start();
        System.out.println("started");
        dcmqr.open();
        System.out.println("opened");
        result = dcmqr.query();
        System.out.println("queried");
        dcmqr.get(result);
        System.out.println("List Size = " + result.size());

        for(DicomObject dco:result){
            System.out.println(dco);
            dcmTools.toByteArray(dco);
            System.out.println("end parsing");
        }

    } catch (Exception e) {
        System.out.println("error "+e);
    }

    try{
        dcmqr.stop();
        dcmqr.close();
    }catch (Exception e) {

    }

    System.out.println("done");
}
}

在我调用dcmTools.toByteArray(dco)之前,一切似乎都没问题。 调用toByteArray()之前的输出如下所示:

List Size = 1
(0008,0052) CS #6 [IMAGE] Query/Retrieve Level
(0008,0054) AE #6 [server] Retrieve AE Title
(0020,000E) UI #54 [1.2.276.0.7230010.3.1.3.669896852.2528.1325171276.916] Series Instance UID

ToByteArray的来源:

public static byte[] toByteArray(DicomObject obj) throws IOException {
    ByteArrayOutputStream baos = new ByteArrayOutputStream();
    BufferedOutputStream bos = new BufferedOutputStream(baos);
    DicomOutputStream dos = new DicomOutputStream(bos);
    dos.writeDicomFile(obj);
    dos.close();
    byte[] data = baos.toByteArray();
    return data;
}

在给ByteArray打电话后,我得到了输出:

error java.lang.IllegalArgumentException: Missing (0002,0010) Transfer Syntax UID

我在其他论坛中发现了一些信息,看起来DcmQR.get()方法似乎不发送imgage数据。是否可以强制DcmQR执行此操作。我写过问题是在DcmQR.createStorageService()方法中,但是我还没有找到解决方案。请帮帮我!!!

Hello cneller!

我做了一些你建议的修改:我添加了setMoveDest和setStoreDestination,DicomObject存储在我添加的目标中 - 它看起来很棒。然后我尝试编写基于FutureDimseRSP的响应处理程序,该处理程序在Association.cget方法中使用:

public class MyDimseRSP extends DimseRSPHandler implements DimseRSP{

private MyEntry entry = new MyEntry(null, null);

private boolean finished;
private int autoCancel;
private IOException ex;

@Override
public synchronized void onDimseRSP(Association as, DicomObject cmd,
                                    DicomObject data) {
    super.onDimseRSP(as, cmd, data);
    MyEntry last = entry;
    while (last.next != null)
        last = last.next;

    last.next = new MyEntry(cmd, data);
    if (CommandUtils.isPending(cmd)) {
        if (autoCancel > 0 && --autoCancel == 0)
            try {
                super.cancel(as);
            } catch (IOException e) {
                ex = e;
            }
    } else {
        finished = true;
    }
    notifyAll();
}

@Override
public synchronized void onClosed(Association as) {
    if (!finished) {
//            ex = as.getException();
        ex = null;
        if (ex == null) {
            ex = new IOException("Association to " + as.getRemoteAET()
                    + " closed before receive of outstanding DIMSE RSP");
        }
        notifyAll();
    }
}

public final void setAutoCancel(int autoCancel) {
    this.autoCancel = autoCancel;
}

@Override
public void cancel(Association a) throws IOException {
    if (ex != null)
        throw ex;
    if (!finished)
        super.cancel(a);
}

public DicomObject getDataset() {
    return entry.command;
}

public DicomObject getCommand() {
    return entry.dataset;
}

public MyEntry getEntry() {
    return entry;
}

public synchronized boolean next() throws IOException, InterruptedException {
    if (entry.next == null) {
        if (finished)
            return false;

        while (entry.next == null && ex == null)
            wait();

        if (ex != null)
            throw ex;
    }
    entry = entry.next;
    return true;
}
}

这是MyEntry代码:

public class MyEntry {

final DicomObject command;
final DicomObject dataset;
MyEntry next;

public MyEntry(DicomObject command, DicomObject dataset) {
    this.command = command;
    this.dataset = dataset;
}

public DicomObject getCommand() {
    return command;
}

public DicomObject getDataset() {
    return dataset;
}

public MyEntry getNext() {
    return next;
}

public void setNext(MyEntry next) {
    this.next = next;
}
}

然后我按照以下方式从Dmcqr重新获取get方法:

public void getObject(DicomObject obj, DimseRSPHandler rspHandler)throws IOException, InterruptedException{
    TransferCapability tc = selectTransferCapability(qrlevel.getGetClassUids());
    MyDimseRSP myRsp=new MyDimseRSP();
    if (tc == null)
        throw new NoPresentationContextException(UIDDictionary
                .getDictionary().prompt(qrlevel.getGetClassUids()[0])
                + " not supported by " + remoteAE.getAETitle());
    String cuid = tc.getSopClass();
    String tsuid = selectTransferSyntax(tc);
    DicomObject key = obj.subSet(MOVE_KEYS);
    assoc.cget(cuid, priority, key, tsuid, rspHandler);
    assoc.waitForDimseRSP();
}

在这个方法的第二个参数中,我使用了我的响应处理程序(MyDimseRSP)的实例。我运行我的代码我得到了我的响应处理程序的命令和数据集的null值。在“next”变量中,只有“command”不为null,而od当然不是我需要的DicomObject。我做错了什么!!!!

1 个答案:

答案 0 :(得分:0)

您将不得不逐步完成代码(包括DCM4CHE工具包代码)。我怀疑您使用的是默认响应处理程序,它只计算已完成操作的数量,并且实际上并不存储来自get命令的图像数据。

显然,下面的for循环遍历find操作的结果,而不是get(需要在响应处理程序中处理)。

for(DicomObject dco:result)

我希望您必须覆盖响应处理程序才能正确编写DICOM文件。另请参阅DcmRcv类,用于从您将收到的DicomObject中编写DICOM文件。

根据您上面的编辑,我假设您只是尝试获取原始DICOM实例数据(而不是存储它的命令)。响应处理程序大致如下:

List<DicomObject> dataList = new ArrayList<DicomObject>();

@Override
public void onDimseRSP(Association as, DicomObject cmd, DicomObject data) {
    if( shouldAdd(as, cmd) ) {
        dataList.add( data )
    }
}

注意大型列表,但它应该为您提供内存中的数据。