簡介
scala.sys.process提供了shell的和系統交互的DSL,包括執行命令, 邏輯操作, 重定向, 管道等操作。
啟動流程
要執行與ProcessBuilder關聯的所有外部命令,scala.sys.process提供了四組方法中。 這些方法中的每一種都具有各種過載和變化,以實現對I/O的進一步控制。
這四種方法如下:
- run:最通用的方法,它立即返回一個scala.sys.process.Process,並且外部命令同時執行。
"find /tmp -name *.jar".run()
- !:阻塞,直到所有外部命令退出,並返回執行鏈中最后一個的退出代碼
"find /tmp -name *.jar" ! 如果沒有參數,直接輸出到標准輸出或者標准錯誤輸出. 同時可以傳遞Logger參數給!. import scala.sys.process._ var normalLines = 0 var errorLines = 0 val countLogger = ProcessLogger(line => normalLines += 1,line => errorLines += 1) "find /etc" ! countLogger 保存到變量中(返回最后命令運行的狀態) val result: Int = "find /tmp -name *.jar" !
- !!:阻塞直到所有外部命令都退出,並返回一個生成輸出的String。
"find project -name *.jar".!! 保存到變量中(保存輸出內容) val result :String = "find project -name *.jar" !!
- lineStream:像run一樣立即返回,生成的輸出通過Stream [String]提供。
獲取該流的下一個元素可能會阻塞,直到它變為可用。 如果返回碼不為零,此方法將拋出異常
val contents: Stream[String] = Process("ls").lineStream
注:如果不需要,請使用lineStream_! 方法。
val etcFiles = "find /etc" lines_! ProcessLogger(line => ())
處理輸入和輸出
- 如果未指定,則使用run或!執行外部命令的輸入。 不會綁定任何東西,輸出將被重定向到Scala進程的stdout和stderr。
- 對於方法!! 和lines,不提供輸入,輸出將根據這些方法的語義進行定向。
Combining ProcessBuilder
這些ProcessBuilder以三種不同的方式組合。
- #| 將第一個命令的輸出傳遞給第二個命令的輸入。 它鏡像一個shell管(|)
- #&& 有條件地執行第二個命令,如果前一個命令以退出值0結束。它鏡像shell的&&
- #|| 如果前一個命令的退出值不為零,則有條件地執行第三個命令。 它鏡像shell的||
例如:
它們可以並行執行,第一個輸出作為輸入提供給第二個,就像Unix管道一樣。 這是通過#|實現的方法。
"ls" #| "grep linux"
它們可以按順序執行,第二次結束后第二次啟動。 這是通過###方法完成的。
"ls" ### "grep linux"
第二個的執行可以通過第一個的返回代碼(退出狀態)來調節,或者僅在它為零時,或者僅在它不為零時。 方法 #&& 和 #|| 完成這些任務
import scala.sys.process._ "find src -name *.scala -exec grep null {} ;" #| "xargs test -z" #&& "echo null-free" #|| "echo null detected" !
重定向
scala的重定向不僅可以從定向普通的文件,還可以把網絡上的文件進行重定向到本地:
a #< url or url #> a 例子: url("http://fuliang.iteye.com") #> file("blog.html") ! 或者 file("blog.html") #< url("http://fuliang.iteye.com") ! 注: 示例中的url和file均需進行封裝,具體如下: new java.net.URL("http://www.baidu.com") #> new java.io.File("/tmp/baidu.html") !
注意,重定向必須用 new java.io.File("")封裝,否則會當作命令,比如"ls" #> "/tmp/a" !將會出錯,必須scala> "ls" #> new java.io.File("/tmp/a") !
文件:
a #< file or file #> a a可以是一個文件或者一個命令,比如: file("blog.html") #> file("fuliang_blog.html") ! 或者 file("fuliang_blog.html") #< file("blog.html") !
追加操作 #>>, #<<
a #>> file or file #<< url("http://fuliang.iteye.com") #> "grep -i ruby" #>> file("Ruby") ! 或者 file("Ruby") #<< ( "grep ruby" #< url(http://fuliang.iteye.com") ) !
與cat一起使用
val spde = url("http://technically.us/spde/About") val dispatch = url("http://databinder.net/dispatch/About") val build = file("project/build.properties") cat(spde, dispatch, build) #| "grep -i scala" !
最后:! 最后執行命令,並返回退出值。 無論打印什么都將發送到Scala過程標准輸出。 如果我們想捕捉它,我們可以用 !! 代替
注意:lines方法雖然已棄用,但可能與同名的StringLike方法沖突。 為避免這種情況,可能希望在進程中調用構建器而不是導入scala.sys.process._。 上面的例子是
import scala.sys.process.Process
Process("find src -name *.scala -exec grep null {} ;") #| Process("xargs test -z") #&& Process("echo null-free") #|| Process("echo null detected") !
