1

在这个优秀的页面上发现了以下功能:

function os.capture(cmd, raw)
  local f = assert(io.popen(cmd, 'r'))
  local s = assert(f:read('*a'))
  f:close()
  if raw then return s end
  s = string.gsub(s, '^%s+', '')
  s = string.gsub(s, '%s+$', '')
  s = string.gsub(s, '[\n\r]+', ' ')
  return s
end

如果我使用 cmd = "ping myknowngoodip -c 1" 执行此函数,那么一切都很好,它会立即返回结果。

但是,如果 ping 超时(服务器宕机左右),它会阻塞整个程序至少 5 秒

对我来说可能的解决方案:

  • 使用协程(但如何正确设计它们以使其正常工作?)

  • 使用某些东西来检测文件是否被写入或是否有写锁(我该怎么做?)

目标是我确实通过 copas ( http://keplerproject.github.io/copas/ ) “查找”请求来 ping 不同的 IP 并报告 IP 是否存在。Copas 已启动并运行,工作至今

非常感谢您的帮助

4

1 回答 1

1

io.popen 不适用于 copas。Copas 用于套接字,而不是管道。

最快的解决方案是使用轮询;例如:

local socket = require "socket"
local waitTime, sleepTime = 5, 0.1
local endT = os.time() + waitTime
local result
os.capture("ping myknowngoodip -c 1 > myoutfile")
while os.time() <= endT do
    local pcall_ok, fHand = pcall(io.open, "myoutfile", "r")
    if pcall_ok then
        result = assert(fHand:read("*all"))
        assert(fHand:close())
    end
    socket.sleep(sleepTime)
end
if result then
    -- Success!
else
    -- Error :(
end

另一种可能性是单独的线程;LuaLanes是目前最好的多线程工具包,但在内存消耗方面也是最大的。然而,这有点涉及更多。

另一种选择是使用 Copas 或LOOP 的 SocketScheduler实现您自己的 ping 套接字(推荐,它比 Copas 好得多);这将是异步的开始。LOOP.SocketScheduler 也是一个很棒的协程管理框架,对各种事情都很有用,并且可以与上面的示例代码一起使用(将 socket.sleep 替换为 scheduler:sleep,并在新创建和注册的协程中运行代码)。

如果你是一个称职的 C 程序员,你也可以让 C 来完成工作;例如,调用一个C 函数,传递一个Lua 回调函数,并在ping 完成后从新创建的pthread 调用回调。

免责声明:代码未经测试。

于 2013-06-27T09:16:06.697 回答