正如您链接到的另一个答案中所解释的那样,它不是严格等效的,因为您必须 catch 或 throw Exception
fromAutoCloseable.close()
并且您必须确保不要context
在块之后做任何事情,try
因为它不像InitialDirContext
直接实现的那样超出范围AutoCloseable
。我仍然同意其他人的观点,这种解决方法非常好。
当然,您也可以直接扩展InitialDirContext
并使其实现AutoCloseable
,或者(对于最终类)使用委托模式并包装目标对象。
package de.scrum_master.stackoverflow;
import javax.naming.NamingException;
import javax.naming.directory.InitialDirContext;
import java.util.Hashtable;
import java.util.Properties;
public class TryWithResourcesAutoCloseableWrapper {
public static void main(String[] args) throws NamingException {
final Properties props = new Properties();
props.put("java.naming.factory.initial", "com.sun.jndi.dns.DnsContextFactory");
variant1(props);
variant2(props);
variant3(props);
}
public static void variant1(Properties props) throws NamingException {
final InitialDirContext context = new InitialDirContext(props);
try (final AutoCloseable dummy = context::close) {
lookupMX(context);
}
catch (NamingException ne) {
throw ne;
}
catch (Exception e) {
e.printStackTrace();
}
}
public static void variant2(Properties props) throws NamingException {
final InitialDirContext context = new InitialDirContext(props);
try (final MyCloseable dummy = context::close) {
lookupMX(context);
}
}
public static void variant3(Properties props) throws NamingException {
try (final MyInitialDirContext context = new MyInitialDirContext(props)) {
lookupMX(context);
}
}
private static void lookupMX(InitialDirContext context) throws NamingException {
System.out.println(context.getAttributes("scrum-master.de", new String[] { "MX" }));
}
public interface MyCloseable extends AutoCloseable {
void close() throws NamingException;
}
public static class MyInitialDirContext extends InitialDirContext implements AutoCloseable {
public MyInitialDirContext(Hashtable<?, ?> environment) throws NamingException {
super(environment);
}
}
}
关于如何使用这些解决方法的更多想法:
- 两者
variant1
都以你永远不会使用的块内的对象variant2
为代价,除非你先将它们投射到。相反,您当然可以直接使用外部对象,这也是您的建议。dummy
try
InitialDirContext
context
- 在
variant3
自动关闭context
对象中直接具有正确的(子)类型,因此您实际上可以无缝地使用它而无需强制转换或虚拟对象。这是以特殊子类为代价的。