[英]How can I express *finally* equivalent for a Scala's Try?
如何使用新的Try
API将以下Java代码转换为Scala?
public byte[] deflate(byte[] data) {
ByteArrayOutputStream outputStream = null;
GZIPOutputStream gzipOutputStream = null;
try {
outputStream = new ByteArrayOutputStream();
gzipOutputStream = new GZIPOutputStream(outputStream);
gzipOutputStream.write(data);
return outputStream.toByteArray();
catch (Exception e) {
...
} finally {
if (gzipOutputStream != null) gzipOutputStream.close();
}
}
Scala版本应该是这样的......
def deflate(data Array[Byte]): Try[Array[Byte]] = Try {
ByteArrayOutputStream outputStream = new ByteArrayOutputStream()
new GZIPOutputStream(outputStream).write(data)
outputStream.toByteArray
}
...但是我如何实现Java的finally
等价?
Scala的Try
API不应该直接等同于Java中的try-catch-finally
构造。 的确,它应该是什么原因? Scala也内置了try-catch-finally
构造。 您可以像在Java中一样直接使用它。 当您需要组合可能失败的多个操作时,需要Try
。
实际上,您提出了更复杂的问题 - 资源管理。 事实上,你的代码应该是这样的:
try (ByteArrayOutputStream os = new ByteArrayOutputStream(data);
GZIPOutputStream gzos = new GZIPOutputStream(os)) {
gzipOutputStream.write(data);
return outputStream.toByteArray();
} catch (Exception e) {
...
}
Java中的“try-with-resources”语言功能会在try
后自动关闭您在括号中指定的所有资源。
不幸的是,Scala在库或语言中没有它的直接等价物。 但是因为Scala更具表现力,你可以手动编写这个结构或使用第三方库,其中我推荐使用scala-arm
。 有关更多信息,请参阅以下链接
由于Try {}块永远不会抛出异常,因此不需要finally语句。 此外,对于这种特殊情况,您可能应该使用scala-arm,就像其他海报所建议的那样。
但是,您可以轻松地将一个finally方法添加到Try中,以便在成功或失败的情况下执行副作用。
像这样的东西:
implicit class TryHasFinally[T](val value:Try[T]) extends AnyVal {
import scala.util.control.NonFatal
def Finally(action: => Unit) : Try[T] =
try {
action;
value
} catch {
case NonFatal(cause) => Failure[T](cause)
}
}
请注意,在Try的所有方法的精神中,如果您的操作抛出非致命异常,但只是将其捕获为失败,则不会抛出异常。
你会像这样使用它:
import java.io._
import java.util.zip._
def deflate(data: Array[Byte]): Try[Array[Byte]] = {
var outputStream : ByteArrayOutputStream = null
Try {
outputStream = new ByteArrayOutputStream()
new GZIPOutputStream(outputStream).write(data)
outputStream.toByteArray
} Finally {
outputStream.close()
}
}
请注意,您不必在Finally中检查null,因为如果由于某些不可思议的原因,outputStream为null,您将只获得一个Failure(NullPointerException)。 此外,如果close抛出IOException,您将收到Failure(IOException)。
Choppy的Lazy TryClose monad适用于这种需要try-with-resources的场景。 另外,它很懒,所以你可以撰写东西。
以下是如何使用它的示例:
val output = for {
outputStream <- TryClose(new ByteArrayOutputStream())
gzipOutputStream <- TryClose(new GZIPOutputStream(outputStream))
_ <- TryClose.wrap(gzipOutputStream.write(data))
} yield wrap(outputStream.toByteArray())
// Does not actually run anything until you do this:
output.resolve.unwrap match {
case Success(bytes) => // do something with bytes
case Failure(e) => // handle exception
}
更多信息: https : //github.com/choppythelumberjack/tryclose
(只需确保导入tryclose._
和tryclose.JavaImplicits._
)
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.