使用Groovy开发之I/O

Groovy提供了一些辅助方法来帮助开发者开发I/O程序,开发过程中你可以使用标准的java代码,但Groovy提供了更多简便的方法来处理files、streams、readers,一起来看一下。

1、读取文件

def baseDir = "/Users/jiangxiaoma/Documents"
new File(baseDir, 'test.txt').eachLine { line ->
    println line
}

eachLine方法是被Groovy自动添加到File类的方法并有不同版本,比如要加上行号,你可以这样:

new File(baseDir, 'test.txt').eachLine { line, nb ->
    println "Line $nb: $line"
}

eachLine闭包内发生任何异常,该方法都能确保源文件被正常关闭。

在某些情况下你可能使用Reader,你仍将从Groovy的自动资源管理中获益,下面的例子,如果发生异常,reader将自动被关闭:

def count = 0, MAXSIZE = 3
new File(baseDir,"test.txt").withReader { reader ->
    while (reader.readLine()) {
        if (++count > MAXSIZE) {
            throw new RuntimeException('Test.txt should only have 3 verses')
        }
    }
}

如果你想把文件内容各行放进一个list,你可以这样:

def list = new File(baseDir, 'test.txt').collect {it}

或者你可以使用as将文件内容放入一个array:

def array = new File(baseDir, 'test.txt') as String[]

如果想把文件内容直接放进一个byte[],可以这样:

def file = new File(baseDir, 'test.txt')
byte[] contents = file.bytes

在进行I/O操作的时候不仅限于文件,事实上,很多操作都依赖于输入输出流,下面例子是从File中获取InputStream

def is = new File(baseDir,'test.txt').newInputStream()
// do something ...
is.close()

这种方式需要自己手动关闭流,使用withInputStream方法将会自动关闭流:

new File(baseDir,'haiku.txt').withInputStream { stream ->
// do something ...
}

2、写入文件

在某些情况下,你可能不仅需要读取文件,也要写入文件,写入文件使用Writer

new File(baseDir,'test.txt').withWriter('utf-8') { writer ->
    writer.writeLine 'Into the ancient pond'
    writer.writeLine 'A frog jumps'
    writer.writeLine 'Water’s sound!'
}

更简单的方法是使用<<操作符:

new File(baseDir,'test.txt') << '''Into the ancient pond
    A frog jumps
    Water’s sound!'''

如果你的test.txt里有内容,上面两种方式将会覆盖test.txt的内容。

当然你也可以直接用输出流来写入文件:

def os = new File(baseDir,'data.bin').newOutputStream()
// do something ...
os.close()

同样的,withOutputStream方法将自动关闭输出流和处理异常:

new File(baseDir,'data.bin').withOutputStream { stream -> 
// do something ...
}

3、遍历文件树

遍历文件夹是一个经常用到的功能,Groovy提供了一些方法来遍历,比如列出根目录下的所有文件和文件夹、找到符合正则表达式的标题的文件:

def dir = new File("/")
//eachFile()方法返回该目录下的所有文件和子目录,不递归
dir.eachFile { file ->
    println file.name
}
dir.eachFileMatch(~/.*\.txt/) {file ->
    println file.name
}

可能你需要处理更深目录层次的文件,或者只显示文件或者文件夹,你可以使用eachFileResource

def dir = new File("/")
//dir.eachFileRecurse()方法会递归显示该目录下所有的文件和目录
dir.eachFileRecurse { file ->
    println file.name
}
dir.eachFileRecurse(FileType.FILES) { file ->
    println file.name
}

一些更复杂的遍历方法你可以使用traverse方法,但需要你设置一个特殊的标志指示如何遍历:

dir.traverse { file ->
    //如果当前文件是一个目录且名字是bin,则停止遍历
    if (file.directory && file.name=='bin') {
        FileVisitResult.TERMINATE
    //否则打印文件名字并继续
    } else {
        println file.name
        FileVisitResult.CONTINUE
   }
}

4、数据和类的序列化&反序列化

在java中使用java.io.DataOutputStreamjava.io.DataInputStream进行序列化和反序列化是非常常用的,使用Groovy将使之变得更容易,下面是序列化数据到文件和从文件读取数据进行反序列化:

boolean b = true
String message = 'Hello from Groovy'
def file = new File(baseDir, 'test.txt')
// 序列化数据到文件
file.withDataOutputStream { out ->
    out.writeBoolean(b)
    out.writeUTF(message)
}
// ...
// 从文件读取数据并反序列化
file.withDataInputStream { input ->
    assert input.readBoolean() == b
    assert input.readUTF() == message
}

同样的,如果一个类实现了Serializable接口,可以将对象序列化到文件:

def file = new File(baseDir, 'test.txt')
Person p = new Person(name:'Bob', age:76)
// 序列化对象到文件
file.withObjectOutputStream { out ->
    out.writeObject(p)
}
// ...
// 从文件读取数据进行反序列化
file.withObjectInputStream { input ->
    def p2 = input.readObject()
    assert p2.name == p.name
    assert p2.age == p.age
}

5、程序中执行shell命令

Groovy提供了简单的方法执行shell命令:

def process = "ls -l".execute()
println "Found text ${process.text}"

//逐行处理
def process = "ls -l".execute()
process.in.eachLine { line ->
    println line
}

execute()方法返回了一个java.lang.Process实例。

如果想执行windows下的dir命令:

def process = "dir".execute()
println "${process.text}"

将返回IOExcepton异常:“Cannot run program "dir": CreateProcess error=2, The system cannot find the file specified.
这是因为dir命令是windows下的shell命令(cmd.ext),不能执行,应该这样写:

def process = "cmd /c dir".execute()
println "${process.text}"

你可能感兴趣的:(使用Groovy开发之I/O)