RemoteActor取消注册actor [英] RemoteActor unregister actor

查看:106
本文介绍了RemoteActor取消注册actor的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我正在与RemoteActors一起玩。现在我想知道,如果我关闭RemoteActor,会发生什么情况。
该演员可以通过RemoteActor.alive和RemoteActor.register来使用。
我找不到两者的反函数:存活和注册。

I'm playing with RemoteActors. Now I wonder, what happens if I shut down an RemoteActor. The actor was made available with RemoteActor.alive and RemoteActor.register. I can't find the inverse of either of both: alive and register.

如何正确关闭RemoteActor?

How do I properly shut down an RemoteActor?

更新

为了使这一点更加明显,我举了一个小例子。以下2个程序均未终止,JVM继续运行。所有由用户创建的actor和main都已完成。

To make it more obvious, I made a 'small' example. Neither of the following 2 programs terminate, the JVM keeps running. All user created actors and main are finished.

package test

import scala.actors.{OutputChannel, AbstractActor, Actor} , Actor._
import scala.actors.remote.RemoteActor , RemoteActor._


object ActorTestA{
  def main(args :Array[String]) {
    RemoteActor.classLoader = getClass().getClassLoader()
    println("Start A")
    val a = new A().start
    println("Fin A")
  }
}

object ActorTestB{
  def main(args :Array[String]) {
    RemoteActor.classLoader = getClass().getClassLoader()
    println("Start B")
    val b = new B().start
    b !? 'start
    println("Fin B")
  }
}

case class M1(sendRef :AbstractActor, m2 :M2)
case class M2(v1:String, v2 :String, sendRef :AbstractActor)
case class M3(m2 :M2)

object A { val portToUse = 20000 }
class A extends Actor {
  alive(A.portToUse)
  register('A, this)
  val proxy = select(actors.remote.Node("localhost", A.portToUse), 'A)

  def act = { 
    loop { react {
      case M1(sendRef, m2) =>
        println("A receives M1")
        sendRef ! M3(m2)
        self ! 'exit
      case 'exit => 
        println("A exits")
        exit
      case any => println("Unknown Msg: "+any)
    } } 
  }
}


class B extends Actor {
  val portToUse = 20001 
  alive(portToUse)
  register('B, this)
  val proxy = select(actors.remote.Node("localhost", portToUse), 'B)

  var resultTo :OutputChannel[Any] = _
  def act = { 
    loop { react {
      case 'start => 
        println("B starts")
        val a = select(actors.remote.Node("localhost", A.portToUse), 'A)
        a ! M1(proxy, M2("some","val", proxy))
        resultTo = sender
      case M3(M2(v1,v2,sendRef))=>
        println("B receives M3")
        resultTo ! sendRef
        self ! 'exit
      case 'exit => 
        println("B exits")
        exit
      case any => println("Unknown Msg: "+any)
    } } 
  }
}

A的输出是:

Start A
Fin A
A receives M1
A exits

对于B,

Start B
B starts
B receives M3
B exits
Fin B

调试器对一个程序说,以下4个非守护线程仍在运行:

The debugger says for A Program, that following 4 non-daemon threads are still up:


  • PlainSocketImpl.socketAccept

  • SocketInputStream.socketRead0

  • ForkJoinScheduler.liftedTree1

  • DestroyJavaVM

  • PlainSocketImpl.socketAccept
  • SocketInputStream.socketRead0
  • ForkJoinScheduler.liftedTree1
  • DestroyJavaVM

推荐答案

演员 .link和演员 .trap退出您要找的东西吗?

Is Actor.link and Actor.trapExit what you are looking for?

编辑,请尝试此操作。我将所有内容都移到了act方法中。在远程角色之间发送代理似乎也会引起问题。这里有一些有关此可能错误的讨论: http://www.scala-lang.org/node / 6779 ;

EDIT Try this. I moved everything into the act methods. Also sending proxies between remote actors seemed to cause problems. There's some discussion about this possible bug here: http://www.scala-lang.org/node/6779 ; see the comment from Stefan Kuhn.

package test

import scala.actors.{OutputChannel, AbstractActor, Exit, Debug, Actor} , Actor._
import scala.actors.remote.RemoteActor , RemoteActor._


object ActorTestA{
  def main(args :Array[String]) {
    println("Start A")
    val a = new A().start
    println("Fin A")
  }
}

object ActorTestB{
  def main(args :Array[String]) {
    println("Start B")
    val b = new B().start
    b ! 'start
    println("Fin B")
  }
}

case class AA(hostname: String, port: Int, symbol: Symbol) {
  def proxy = select(actors.remote.Node(hostname, port), symbol)
}
case class M1(sendRef :AA, m2 :M2)
case class M2(v1:String, v2 :String, sendRef :AA)
case class M3(m2 :M2)

object A { val portToUse = 20000 }
class A extends Actor {
  def act = { 
    alive(A.portToUse)
    register('A, this)
    RemoteActor.classLoader = getClass().getClassLoader()
    val proxy = select(actors.remote.Node("localhost", A.portToUse), 'A)

    loop { react {
      case M1(sendRef, m2) =>
        println("A receives M1")
        sendRef.proxy ! M3(m2)
        self ! 'exit
      case 'exit => 
        println("A exits")
        exit
      case any => println("Unknown Msg: "+any)
    } } 
  }
}


class B extends Actor {
  def act = { 
    RemoteActor.classLoader = getClass().getClassLoader()
    val portToUse = 20001 
    alive(portToUse)
    register('B, this)
    var proxy = AA("localhost", portToUse, 'B)
    var resultTo :Option[OutputChannel[Any]] = None

    loop { react {
      case 'start => 
        println("B starts")
        val a = select(actors.remote.Node("localhost", A.portToUse), 'A)
        a ! M1(proxy, M2("some","val", proxy))
        resultTo = Some(sender)
      case M3(M2(v1,v2,sendRef))=>
        println("B receives M3") 
        resultTo match {  
          case Some(ch) => ch ! sendRef.proxy;  
          case None => println("ch missing!?")
        }
        self ! 'exit 
      case 'exit => 
        println("B exits")
        exit
      case any => println("Unknown Msg: "+any)
    } } 
  }
}

这篇关于RemoteActor取消注册actor的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆