【问题标题】:Coffeescript wait until flag is set from a callback and then move on to the next part of the codeCoffeescript 等到从回调中设置标志,然后继续代码的下一部分
【发布时间】:2021-11-12 19:05:05
【问题描述】:

我这里有一个使用 coffeescript 的简单程序,它在 Raspberry pi 上创建一个服务器,它从在同一网络上的 Windows 机器上运行的 python 脚本接收数据,对这些数据进行一些计算并发回结果。

我想做的是让服务器保持打开、运行和接受数据,直到 python 脚本发送的参数要求它关闭。

为了尝试这个,我创建了一个循环来阻塞代码,直到这些标志被设置;但是,这会阻塞代码,因此没有设置标志(python 脚本甚至无法连接)。所以我想我需要在这里/promises 使用某种异步控制。

# Functions
process_data = (data) ->
    double_value: data * 2
    triple_value: data * 3

clean_up_actions = () ->
    console.log "Cleaning up..."

# Create server
net = require('net')
domain = '10.1.1.28'
port = 9001

program_complete = false
clean = false

console.log("Creating server...")
server = net.createServer (socket) ->
    console.log "#{socket.remoteAddress} has connected"

    # When data is received, do something and send back a response
    socket.on 'data', (data) ->
        console.log("Data received...")
        if not program_complete
            incoming_data = JSON.parse(data)
            
            some_useful_value = incoming_data["some_useful_value"]
            result = process_data(some_useful_value)
            socket.write JSON.stringify(result)
            
            # Check if end of program
            program_complete = incoming_data["end_program"]
            if program_complete
                clean_up_actions()
                clean = true

console.log "Listening to #{domain}:#{port}"
server.listen port, domain
    
console.log("Waiting until complete...")
loop
    break if program_complete and clean
console.log "--- PROGRAM FINISHED ---"

为了测试这一点,我使用了Q.delay(10000) 而不是循环,它按照我期望的方式工作(我得到了一个连接并且可以来回传递数据没有问题),除了当@ 987654323@设置为True,10秒后退出(如预期)。

...
# console.log("Waiting until complete...")
# loop
#     break if program_complete and clean
# console.log "--- PROGRAM FINISHED ---"

Q = require('q')
Q.delay(10000).then ->
    console.log "Done"
.finally ->
  console.log('--- PROGRAM END ---')
  process.exit(0)

我的问题是,我如何使用 Promise 或其他异步控件(如 Q.delay(n)),而不是延迟一段时间,而是等到设置这些标志。

如果有兴趣,或者想测试一下,python代码如下

import socket
import json
import time

REMOTE_HOST = '10.1.1.28'
REMOTE_PORT = 9001

# Create a socket
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# Connect to the remote host and port
sock.connect((REMOTE_HOST, REMOTE_PORT))

# Send some data
data_dictionary = {
    'some_useful_value' : 21,
    'end_program' : False
}
data_to_send = json.dumps(data_dictionary).encode('utf-8')
sock.send(data_to_send)
print(sock.recv(1024))

time.sleep(0.5)

# Send some data, but this time, end
data_dictionary = {
    'some_useful_value' : 13,
    'end_program' : True
}
data_to_send = json.dumps(data_dictionary).encode('utf-8')
sock.send(data_to_send)
print(sock.recv(1024))

time.sleep(0.5)

# Terminate
sock.close()

【问题讨论】:

    标签: sockets promise coffeescript


    【解决方案1】:

    您可能有几种方法可以实现这一点,我们等到您的代码中的标志 end_programclean 被您的 socket.on 'data' 方法设置为 true。

    在脚本结束时,您可以使用一个函数,该函数不断返回一个承诺,直到满足某个条件(在这种情况下,设置了标志 clean)。

    您使用的 Q 库应该足够了,因为我相信它的 .delay 函数是异步的。例如,在下面的代码中,该函数每秒检查一次标志,如果未设置标志,该函数将再次调用自身,并在另一秒钟后检查。如果设置了标志,我们就可以继续代码的下一部分。

    这可能看起来像这样,使用变量 clean 作为您的标志:

    waitForFlag = () -> 
      Q.delay(1000).then -> 
        if (clean)
          console.log "Condition met! We can move on"
        else
          # if condition isn't met, check again in one second
          waitForFlag() 
    
    # Call function
    waitForFlag().then ->
      process.exit(0)
    

    【讨论】:

      【解决方案2】:

      我的问题是,我如何使用 Promise 或其他异步控制(如 Q.delay(n))而不是延迟一段时间,而是等到这些标志被设置。

      我认为简短的回答是否定的,或者至少在不深入研究一些更深奥的东西(如中断)的情况下不会。但是您可以通过设置短暂等待来近似“手动”,检查值是否已设置,然后再次等待或退出等待状态并根据检查结果继续执行其余逻辑。

      我不熟悉您在此处导入的 Q 库(并使用 Q.delay 调用),但我认为您走在正确的道路上。在一般情况下,我认为您想要做的是使用setIntervalsetTimeout 基本上“让”执行一段相对较短的时间,然后检查该值是否已设置。我想Q.delay 在内部使用setTimeout 或类似的。

      这是我的意思的粗略草图:

      # here is the flag we'll wait to be set
      the_flag = null 
      
      # here is an arbitrary callback method that represents
      # the "action" to take once the flag is set
      the_callback = (flag)->
        console.log "#{(new Date()).toISOString()}: the_flag is now set to:", flag
      
      # we'll keep a handle to the value returned by setInterval
      # so we can cancel it later.
      interval_id = null
      
      # here's the method that checks if the flag has been set
      # and responds accordingly
      check_if_flag_is_set = ()->
        if the_flag?
          # it *is* set now, so clear the interval and callback
          clearInterval(interval_id)
          interval_id = null
          the_callback(the_flag)
        else
          console.log "#{(new Date()).toISOString()}: the flag is not yet set, still waiting..."
      
      # now just kick off the interval to invoke the "check"
      # method every 3 seconds
      interval_id = setInterval( check_if_flag_is_set, 3000 )
      

      现在,如果/当 the_flag 的值在某些异步代码中设置时,check_if_flag_is_set 方法将检测到它并将结果报告给 the_callback。请注意,setInterval 等待期是非阻塞的 - 在我们等待下一次调用 check_if_flag_is_set 时,其他代码可以并且将会执行。

      也就是说,网络 i/o 交互也是非阻塞的,所以如果标志是由正在写入套接字的一些数据设置的,那么您实际上不需要额外的一层异步、非阻塞等待 via setInterval。您可以等待网络侦听器发出 on('data',handler) 或类似事件。

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 2015-10-04
        • 2017-02-22
        • 1970-01-01
        • 2012-06-08
        • 1970-01-01
        • 2016-10-29
        • 1970-01-01
        相关资源
        最近更新 更多