multithreading - 我的应用程序需要使用 TCL 中的线程吗?

标签 multithreading sockets tcl

我正在尝试构建的应用程序如下。

Java 客户端 <---> TCL 服务器(比如 Server.tcl)<---> 一些其他 TCL 脚本(比如 ABC.tcl)

我正在使用 Socket 编程来实现 Java 客户端和 TCL 服务器之间的通信。现在我的要求是在运行时从 Server.tcl 调用 ABC.tcl。 ABC.tcl 在执行时会通过server.tcl 向Java 客户端发送一些问题。坐在 Java 客户端的用户应该发送对所问问题的答复,并且答复应该返回到 ABC.tcl。现在,当回复返回时,ABC.tcl 应该等待。但是我不能在 ABC.tcl 中有一个繁忙的等待循环,因为控件不会到达 Server.tcl,因为它是与 Java 客户端交谈的人。

ABC.tcl 将使用 Server.tcl 中定义的一些程序。如果需要,可以更改设计。

我正在考虑在 TCL 中使用线程。我打算将 ABC.tcl 作为一个单独的线程调用,然后让它等待回复。这样 Server.tcl 可以独立于 ABC.tcl 执行。

征求意见,他们是否有更好的方法来做到这一点? 如果 Threads 是唯一更好的方法,我可能会遇到哪些挑战,我该如何应对?

编辑:

我没有在 Server.tcl 和 ABC.tcl 之间使用套接字通信。 在 Server.tcl 中,我正在采购 ABC.tcl,当一些通信请求来自 Java 客户端时,我正在调用 ABC.tcl 的过程(比如 proc someProc{})。 现在在 someProc 过程中,一些处理已经完成,一个问题(比如“你想继续吗?”)通过 Server.tcl 发送回 Java 客户端。 现在在 ABC.tcl 中,我希望过程 someProc 等待用户输入内容。当用户输入答案(是/否)时, 我希望程序从等待用户输入的位置继续执行。

现在发生的事情是我无法根据需要正确设置等待条件。我尝试使用一个标志变量,该变量将在用户输入答案时更新,并尝试根据标志变量在 someProc 中保持一个繁忙的等待循环。但是当用户输入到来时,它不会从等待循环开始恢复执行。它像正常请求一样启动,然后进入在 Server.tcl 中注册的过程以处理传入数据。

我有“vwait events”调用,而且我在 Server.tcl 中有“fileevent $sock readable [list svcHandler $sock]”。

我无法获得,我究竟如何才能在 ABC.tcl 中的过程 someProc 中等待,并根据用户输入从相同的繁忙等待中恢复处理。

标题##Server.tcl 文件:

source "<path>/serverTest.tcl"


set svcPort [lindex $argv 0]
set filePath <path where all related files are kept>
set replyReady 0
set packet ""

proc writeToFile {fileName data mode} {
set fileId [open $fileName $mode]
puts -nonewline $fileId $data
close $fileId
}

proc writeJavaUTF {sock msg} {
  set date [exec date]
  set msg $date$msg
  set data [encoding convertto utf-8 $msg]
  puts -nonewline $sock [binary format "S" [string length $data]]$data
}

proc readJavaUTF {sock} {
  binary scan [read $sock 2] "S" len
  set data [read $sock [expr {$len & 0xFFFF}]]
  return [encoding convertfrom utf-8 $data]
}

proc sendMessageToUser {sock msg} {
      writeToFile [exec pwd]/CTS/log "\nSending Message Back to Client" "a"
     writeJavaUTF $sock $msg\n
}

proc setReplyReady {} {
        global replyReady
        set replyReady 1
        writeToFile [exec pwd]/CTS/log "\nSetReplyReady" "a"

}

proc resetReplyReady {} {
        global replyReady
        set replyReady 0
        writeToFile [exec pwd]/CTS/log "\nResetReplyReady" "a"
}

proc getReplyReadyStatus {} {
        global replyReady
       writeToFile [exec pwd]/CTS/log "\nReply Ready is : $replyReady" "a"
    return $replyReady
}


proc executeCommand {sock msg} {
    writeToFile [exec pwd]/CTS/log "\nexecuteCommand" "a"
    runServerTest $sock
}

# Handles the input from the client and  client shutdown
proc  svcHandler {sock} {

  global packet replyReady
  set packet [readJavaUTF $sock] ;# get the client packet
  writeToFile [exec pwd]/CTS/log "\nThe packet from the client is $packet" "w"

set packet [string range $packet 0 [expr {[string first "\n" $packet] - 1}]]

set endChar "EOC"

  if {[eof $sock] || ![string compare -nocase $packet $endChar]} {    ;# client gone or finished
      writeToFile [exec pwd]/CTS/log "Closing the socket" "a"
      writeToFile [exec pwd]/CTS/flag "0" "w"
     close $sock        ;# release the servers client channel
     exit
  } else {
    #doService $sock $ipacket
     set typeReply "reply"
     set typeExecute "exec"
     set type [string range $packet 0 [expr {[string first ":" $packet] - 1}]]
     writeToFile [exec pwd]/CTS/log "\nThe type is : $type" "a"
     # If it is not a reply for a request
     if {![string compare -nocase $type $typeExecute]} {
       executeCommand $sock $packet
     } else {
      writeToFile [exec pwd]/CTS/log "\nExecuting the ELSE" "a"
       setReplyReady
     }

  }
}

proc accept {sock addr port}  {

  # Setup handler for future communication on client socket
  fileevent $sock readable [list svcHandler $sock]

  # Read client input in lines, disable blocking I/O
  fconfigure $sock -buffering line -blocking 0 -translation binary

  return $sock
}

set sock [socket -server accept $svcPort]
vwait events    ;# handle events till variable events is set

标题## serverTest.tcl(作为ABC.tcl)

proc runServerTest {sock} {
global replyReady
writeToFile [exec pwd]/CTS/log "\nInside serverTest.tcl." "a"

resetReplyReady
sendMessageToUser $sock "From Server : The socket is working. Do you want to continue ?"
writeToFile [exec pwd]/CTS/log "\nWaiting for user input" "a"
#Loop untill the reply is read from the socket
while {![getReplyReadyStatus]} {
after 1000
}

set retMessage [getPacket]
resetReplyReady
writeToFile [exec pwd]/CTS/log "\nThe reply from user is : $retMessage" "a"
}

标题##代码中的挑战

我以“exec:Hello”的形式发送了一个新请求 我发送回复为“回复:是”

正如您在 proc svcHandler{} 中看到的那样,如果它是一个新请求,我将执行 executeCommand{} proc。否则,我设置 replyReady 标志。当我执行 executeCommand{} proc 时,proc runServerTest{} 被调用,它位于 serverTest.tcl 中。现在在此过程中,我将问题发回给用户,我可以在客户端的 JAVA UI 屏幕上看到该问题。当我尝试发送回复时,svcHandler{} 中的 proc setReplyready{} 根本没有执行。当我看到日志文件时,我可以看到 proc getReplyReadyStatus{} 的输出,它每秒从 proc runServerTest{} 的 while 循环中调用。这意味着 while 循环无限期地执行。由于 proc setReplyReady{} 没有从 svcHandler{} 调用,可能是 while 循环没有退出。真的不知道进程在哪里等待。任何解决方法?

谢谢, 皮尤什

最佳答案

不要使用线程。使用事件循环。

阅读 fileevent:http://www.tcl.tk/man/tcl8.6/TclCmd/fileevent.htm

通常,您会为 Java 客户端套接字和 ABC 脚本套接字设置事件处理程序,以便在它们可读时触发,并在数据到达时执行您需要执行的操作。

如果您需要实现定时操作,例如看门狗定时器,请查看 after 命令。

关于multithreading - 我的应用程序需要使用 TCL 中的线程吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16536026/

相关文章:

java - 线程安全树

javascript - 套接字.io : How do I call upon my socket methods dynamically from an object?

http - 在tcl 8.0中使用http post方法获取url

java - 使用并发运行的线程将单词添加到数组列表

java - 在java中的两个线程之间传递字符串

c# - 限制单次执行一个方法的线程数

java - 向所有客户端发送消息(客户端 - 服务器通信)

java - 使用同一个客户端套接字向服务器套接字发送多条消息

postgresql - 如何在 PL/TCL 中访问 PostgreSQL 数组

linux - 如何用Tcl/Tk语言创建一个带有文本模式控件的定时器