Golang-inspired resource management library.
implementation "me.jason5lee:defer:2.0.0"
implementation("me.jason5lee:defer:2.0.0")
Open two file for reading and writing, print a message before attempting to close them.
deferScope {
val f1 = File("/path/to/read-file")
.bufferedReader()
.defer { close() }
val f2 = File("/path/to/write-file")
.bufferedWriter()
.defer { close() }
defer { println("Closing file ...") }
// Operating f1 and f2
}
Equivalent code using use
and try-finally
File("/path/to/read-file").bufferedReader().use { f1 ->
File("/path/to/write-file").bufferedWriter().use { f2 ->
try {
// Operating f1 and f2, with 3 indents instead of 1.
} finally {
println("Closing file ...")
}
}
}
Example of multiple resources and canceling: suppose you have many output targets, each target has a name. Now you want to implement an object that maintains multiple output target objects, which output information to the corresponding target according to the type of information. You can implement this object like this
data class Message(val type: String, ...)
class OutputTarget(name: String) {
fun close() { ... }
fun output(message: Message) { ... }
}
class Output(vararg names: String) {
private val outputMap: Map<String, OutputTarget>
init {
outputMap = deferScope {
val outMap = HashMap<String, OutputTarget>(names.size)
for (name in names) {
val target = OutputTarget(name)
.defer { close() } // This defer is to close the created target when an exception is thrown.
outMap[name] = target
}
outMap.toMap().also {
cancelDeferred() // cancel all defers because the construction was successful.
}
}
}
fun output(message: Message) {
outputMap[message.type]?.output(message)
}
fun close() {
// You can use deferScope to ensure all targets are closed even if some fail.
deferScope {
for (output in outputMap.values) {
output.defer { close() }
}
}
}
}
Compared to using the use function and try-finally, this library has the following advantages
- Less indent, especially if you have multiple resources to close.
- Easy to have a variable number of resources.
- Provide the
cancelDefer
method to transfer ownership, i.e. pass the resource to another object and let it close.
Use suspendDeferScope
if there may be suspend function calls in the defer block.
The defer code will be executed in the non-cancellable coroutine context.