默认情况下,Ruby以缓冲模式打开$stdin和$stdout.这意味着您不能使用Ruby来执行类似grep的操作过滤文本.有没有办法强制Ruby使用面向行的模式?我看到了各种解决方案,包括popen3(仅限缓冲模式)和pty(不需要单独处理$stdout和$stderr,我需要).
我该如何做? Python似乎有同样的缺点.
解决方法
看起来你最好打赌是使用STDOUT.syswrite和STDOUT.sysread – 以下似乎有相当不错的性能,尽管是丑陋的代码:
- STDIN.sync = true
- STDOUT.syswrite "Looking for #{ARGV[0]}\n"
- def next_line
- mybuff = @overflow || ""
- until mybuff[/\n/]
- mybuff += STDIN.sysread(8)
- end
- overflow = mybuff.split("\n")
- out,*others = overflow
- @overflow = others.join("\n")
- out
- rescue EOFError => e
- false # NB: There's a bug here,see below
- end
- line = next_line
- while line
- STDOUT.syswrite "#{line}\n" if line =~ /#{ARGV[0]}/i
- line = next_line
- end
注意:不确定你需要#sysread #sync,但如果是这样,你也应该同步STDOUT.此外,它一次读取8个字节到mybuff中 – 你应该尝试这个值,它的效率很低/ cpu很重.最后,这段代码是hacky,需要一个重构,但它的工作原理 – 使用ls -l〜/ * |进行测试ruby rgrep.rb doc(其中’doc’是搜索词)
第二个注意事项:显然,我很忙,试图让它表现很好,我没能让它执行正常!由于Dmitry Shevkoplyas有noted,如果在EOFError被提出时,在@overflow中有文本,该文本将会丢失.我相信如果用以下代码替换catch,那么它应该解决问题:
- rescue EOFError => e
- return false unless @overflow && @overflow.length > 0
- output = @overflow
- @overflow = ""
- output
- end