Coffeescript 等到从回调中设置标志,然后继续代码的下一部分

Posted

技术标签:

【中文标题】Coffeescript 等到从回调中设置标志,然后继续代码的下一部分【英文标题】:Coffeescript wait until flag is set from a callback and then move on to the next part of the code 【发布时间】: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()

【问题讨论】:

【参考方案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) 或类似事件。

【讨论】:

以上是关于Coffeescript 等到从回调中设置标志,然后继续代码的下一部分的主要内容,如果未能解决你的问题,请参考以下文章

如何在 ngCookies 中设置 httpOnly 标志?

在android的sqlite数据库中设置标志

等到当前位置在 didFinishLaunchingWithOptions 中设置

在 AngularJS 中设置 JSONP 回调函数

如何在 Spring Boot 中设置同站点 cookie 标志?

如何使用 jQuery 在 css 中设置重要标志? [复制]