我正在运行一个Web服务,该服务将替换docx模板中的文本,然后将其转换为pdf。
 我正在使用ubuntu 18.04和glassfish服务器进行部署
 当我对转换所有内容的服务发出单个请求时就可以了,但是
   当我像双击问题或并发请求一样快速提出双重请求时,出现以下异常:

com.sun.star.lang.DisposedException
    在com.sun.star.lib.uno.environments.remote.JobQueue.removeJob(JobQueue.java:201)


使用者:java.io.IOException:到达EOF-套接字,主机=本地主机,端口= 8100,本地主机=本地主机,本地端口= 58494,peerHost =本地主机,peerPort = 8100
在com.sun.star.lib.uno.bridges.java_remote.XConnectionInputStream_Adapter.read(XConnectionInputStream_Adapter.java:50)

我建立了以示例为指导的代码,我是openoffice和LibreOffice的初学者,我看到例外情况是拖延到xDesktop.terminate();。 ,所以我做了一个实验并删除了该语句,所以现在没有引发异常了,但是正如我提到的我是一个初学者,因此我不确定xDesktop.terminate();是什么。删除它有什么后果?

这是我正在运行的代码:

    public Response getFilePdf(Integer idqueja) {
        try {

                    // Initialise
       String oooExeFolder = "/opt/libreoffice6.1/program";
       XComponentContext xContext = BootstrapSocketConnector.bootstrap(oooExeFolder);
    //XComponentContext xContext = Bootstrap.bootstrap();

    XMultiComponentFactory xMCF = xContext.getServiceManager();

    Object oDesktop = xMCF.createInstanceWithContext(
         "com.sun.star.frame.Desktop", xContext);

    XDesktop xDesktop = (XDesktop) UnoRuntime.queryInterface(
         XDesktop.class, oDesktop);

    // Load the Document
    String workingDir = "/home/somePath/";
    String myTemplate = workingDir + "template.docx";

    if (!new File(myTemplate).canRead()) {
        throw new RuntimeException("Cannotix load template:" + new File(myTemplate));
    }

    XComponentLoader xCompLoader = (XComponentLoader) UnoRuntime
        .queryInterface(com.sun.star.frame.XComponentLoader.class, xDesktop);

    String sUrl = "file:///" + myTemplate;

    PropertyValue[] propertyValues = new PropertyValue[0];

    propertyValues = new PropertyValue[1];
    propertyValues[0] = new PropertyValue();
    propertyValues[0].Name = "Hidden";
    propertyValues[0].Value = new Boolean(true);

    XComponent xComp = xCompLoader.loadComponentFromURL(
        sUrl, "_blank", 0, propertyValues);


    // Manipulate
    XReplaceDescriptor xReplaceDescr = null;
    XReplaceable xReplaceable = null;

    XTextDocument xTextDocument = (XTextDocument) UnoRuntime
            .queryInterface(XTextDocument.class, xComp);

    xReplaceable = (XReplaceable) UnoRuntime
            .queryInterface(XReplaceable.class,
                    xTextDocument);

    xReplaceDescr = (XReplaceDescriptor) xReplaceable
            .createReplaceDescriptor();


        xReplaceDescr.setSearchString("<version>");
    xReplaceDescr.setReplaceString("1.x");
    xReplaceable.replaceAll(xReplaceDescr);
    // mail merge the date
    xReplaceDescr.setSearchString("<number>");
    xReplaceDescr.setReplaceString("12345677");
    xReplaceable.replaceAll(xReplaceDescr);


        OOoOutputStream output= new OOoOutputStream();

    // save as a PDF
    XStorable xStorable = (XStorable) UnoRuntime
            .queryInterface(XStorable.class, xComp);

    propertyValues = new PropertyValue[2];
    // Setting the flag for overwriting
    propertyValues[0] = new PropertyValue();
        propertyValues[1] = new PropertyValue();

    propertyValues[0].Name = "OutputStream";
    propertyValues[0].Value = output;
    // Setting the filter name

    propertyValues[1].Name = "FilterName";
    propertyValues[1].Value = "writer_pdf_Export";

    // Appending the favoured extension to the origin document name
    //String myResult = workingDir + "fileConverted.pdf";
    xStorable.storeToURL("private:stream", propertyValues);


    // shutdown
    xDesktop.terminate();

         ByteArrayInputStream inStream = new ByteArrayInputStream(output.toByteArray());



                  ResponseBuilder response = Response.ok((Object) inStream);
                            response.header("Content-Disposition", "attachment;filename=template.pdf");
                            return response.build();


        } catch (Exception e) {
            e.printStackTrace();
                        ResponseBuilder response = Response.serverError();
                        return response.build();
        }
    }


因此,该Web服务方法计划将文档提供给许多用户,因此,如果我同时或过于连续地请愿,除非我删除xDesktop.terminate();否则它将引发异常。但是我不知道它是否还会带来诸如覆盖内存或类似事情的进一步后果。
提前致谢。

最佳答案

问题是xDesktop.terminate()请求关闭基础的soffice进程(Java API基本上只是围绕它的包装器)。您有两种选择:


您可以自己预先启动libreoffice,因此您不必一直产生新的办公室程序,而只需连接到现有的办公室即可。这样做的好处是,如果您有很多小请求(无启动成本),则性能会很好,但是您的转换不太隔离(出于安全考虑),实际上转换不会并行发生。在这种情况下,您启动了办公室程序,因此不要调用xDesktop.terminate()。
或者,您可以使用专用的唯一UserInstallation目录(请参见libreoffice --help)启动每个办公室流程,然后xDesktop.terminate()不会有问题,因为一次转换是一个办公室流程。


DisposedException只是意味着您正在使用远程UNO协议与办公室进程进行交谈,而在您的请求正在进行时,其他人杀死了办公室进程。

您需要研究Java API如何允许传递自定义的办公室参数,但在最坏的情况下,您可以执行soffice "--accept=socket,host=localhost,port=9999;urp;StarOffice.ServiceManager" -env:UserInstallation=file:///tmp/test之类的操作,如果并行执行两次转换,则需要确保9999和/tmp/test是唯一的。 (同样,请参见文档,如果发现更好,可以使用unix套接字而不是TCP端口。)

因此,最重要的是:如果您在多个转换之间共享事务处理过程,则不要终止xDesktop单例,因为这会“崩溃”其他转换过程。

10-05 18:27