您可以使用-re 标志直接在expect 中使用regexp。感谢 Donal 指出单引号和双引号问题。我已经使用两种方式给出了解决方案。
我创建了一个文件,内容如下,
Id Name
-------------------
abcd 12 John Smith
这不过是您的 java 程序的控制台输出。我已经在我的系统中对此进行了测试。即我刚刚用cat 模拟了你的程序的输出。您只需用您的程序命令替换cat 代码。简单的。 :)
双引号:
#!/bin/bash
expect -c "
spawn ssh user@domain
expect \"password\"
send \"mypassword\r\"
expect {\\\$} { puts matched_literal_dollar_sign}
send \"cat input_file\r\"; # Replace this code with your java program commands
expect -re {-\r\n(.*?)\s\s}
set output \$expect_out(1,string)
#puts \$expect_out(1,string)
puts \"Result : \$output\"
"
单引号:
#!/bin/bash
expect -c '
spawn ssh user@domain
expect "password"
send "mypasswordhere\r"
expect "\\\$" { puts matched_literal_dollar_sign}
send "cat input_file\r"; # Replace this code with your java program commands
expect -re {-\r\n(.*?)\s\s}
set output $expect_out(1,string)
#puts $expect_out(1,string)
puts "Result : $output"
'
如您所见,我使用了{-\r\n(.*?)\s\s}。这里的大括号防止任何变量替换。在您的输出中,我们有一个充满连字符的第二行。然后换行。然后是您的第 3 行内容。让我们解码使用的正则表达式。
-\r\n 是将一个文字连字符和一个新行匹配在一起。这将匹配第二行中的最后一个连字符和换行符,而换行符现在又进入第三行。因此,.*? 将匹配所需的输出(即 abcd 12),直到遇到与 \s\s 匹配的双倍空格。
您可能想知道为什么我需要用于获取子匹配模式的括号。
一般情况下,expect 会将 expect 的整个匹配字符串保存在 expect_out(0,string) 中,并将所有匹配/不匹配的输入缓冲到 expect_out(buffer)。每个子匹配都会保存在后续的字符串编号中,如expect_out(1,string)、expect_out(2,string)等。
正如 Donal 所指出的,最好使用单引号的方法,因为它看起来不那么凌乱。 :)
在双引号的情况下,不需要用反斜杠转义\r。
更新:
我已将 regexp 从 -\r\n(\w+\s+\w+)\s\s 更改为 -\r\n(.*?)\s\s。
通过这种方式-您的要求-例如match any number of letters and single spaces until you encounter first occurrence of double spaces in the output
现在,让我们来回答您的问题。您提到您已经尝试过-\r\n(\w+)\s\s。但是,\w+ 存在问题。请记住 \w+ 不会匹配空格字符。您的输出中有一些空格,直到双倍空格。
正则表达式的使用取决于您对要匹配的输入字符串的要求。您可以根据需要自定义正则表达式。
更新版本 2:
.*?有什么意义。如果您单独询问,我将重复您的评论。在正则表达式中,* 是一个贪心运算符,? 是我们的救命稻草。让我们将字符串视为
Stackoverflow is already overflowing with number of users.
现在,看看正则表达式.*flow的效果如下。
* 匹配任意数量的字符。 更准确地说,它匹配尽可能长的字符串,同时仍然允许模式本身匹配。 因此,模式中的.* 匹配模式匹配中的字符Stackoverflow is already over 和flow字符串中的文本flow。
现在,为了防止.* 仅匹配字符串flow 的第一次出现,我们将? 添加到它。它将帮助模式以非贪婪的方式表现。
现在,再次回到您的问题。如果我们使用了.*\s\s,那么它将匹配整行,因为它试图尽可能地匹配。这是正则表达式的常见行为。
更新版本 3:
按以下方式编写您的代码。
x=$(expect -c "
spawn ssh user@host
expect \"password\"
send \"password\r\"
expect {\\\$} { puts matched_literal_dollar_sign}
send \"cat input\r\"
expect -re {-\r\n(.*?)\s\s}
if {![info exists expect_out(1,string)]} {
puts \"Match did not happen :(\"
exit 1
}
set output \$expect_out(1,string)
#puts \$expect_out(1,string)
puts \"Result : \$output\"
")
y=$?
# $x now contains the output from the 'expect' command, and $y contains the
# exit status
echo $x
echo $y;
如果流程正常,则退出代码的值为0。否则,它将为1。这样,您可以在bash脚本中检查返回值。
查看here 以了解info exists 命令。