爪哇。重用 try-finally 构造并返回

Yeg*_*xim 3 java code-reuse try-catch

看看这段代码:

IDfSessionManager manager = DfcSessionManager.getSessionManager();
    try {
        IDfSession session = manager.getSession(DfsUtils.getCurrentRepository());
        ...
        return somewhat; //May be without return statement
    } finally {
        if (session != null) {
            manager.release(session);
        }
    }
Run Code Online (Sandbox Code Playgroud)

这种结构重复多次并围绕不同的代码。这可以是带有或不带有 return 语句的方法。我想为这个 try-finally 块制作一些可重用的东西。

我想出了这样的认识。

public abstract class ISafeExecute<T> {

private IDfSession session = null;

protected abstract T execute() throws DfException;

public T executeSafely() throws Exception {
    IDfSessionManager manager = DfcSessionManager.getSessionManager();
    try {
        session = manager.getSession(DfsUtils.getCurrentRepository());
        return execute();
    } finally {
        if (session != null) {
            manager.release(session);
        }
    }
}

public IDfSession getSession() {
    return session;
}
Run Code Online (Sandbox Code Playgroud)

}

会话字段是用公共 getter 制作的。

我们可以像这样使用这个类(带有返回的对象):

return new ISafeExecute<String>() {
        @Override
        public String execute() throws DfException {
            return getSession().getLoginTicket();
        }
    }.executeSafely();
Run Code Online (Sandbox Code Playgroud)

或者没有返回对象:

    new ISafeExecute() {
        @Override
        public Object execute() {
            someMethod();
            return null;
        }
    }.executeSafely();
Run Code Online (Sandbox Code Playgroud)

Mat*_*eid 5

您可以使用Runnable<T>构建一种机制来执行此操作(类似于将一个函数注入另一个函数):

public void runInSession(Runnable<IDfSession> runnable) {

    IDfSession session = null;
    try {

        session = manager.getSession(DfsUtils.getCurrentRepository());
        runnable.run(session);        

    } finally {
        if (session != null) {
            manager.release(session);
        }
    }

}
Run Code Online (Sandbox Code Playgroud)

您也可以使用更多泛型来使您能够返回值。我这里缺少 Java 编译器,但我对语法有点不确定。

编辑,因为我看到你的编辑:

使用自定义ISafeExecute界面可能比使用 更简洁Runnable<T>,但想法保持不变。您可以构建它,以便可以优雅地放置返回值(或错误):

interface ISafeExecute<T> {

  void execute(IDfSession session);

  T getResult();

  Exception getException();

}

mySafeExecute.execute(session);

if(mySafeExecute.getException() == null) {
    return mySafeExecute.getResult();
} else {
    // runtime exception or declaration in method
    // signature
    throw new RuntimeException(mySafeExecute.getException());
}
Run Code Online (Sandbox Code Playgroud)

  • 太棒了!但是 Callable&lt;T&gt; 而不是 Runnable。 (2认同)