本地主机拒绝了 Rmi 连接。

15

我有一个使用Java RMI的问题:

当我尝试运行我的服务器时,我遇到了connectException(见下文)。

在执行重新绑定方法时发生异常:

Runtime.getRuntime().exec("rmiregistry 2020");
MyServer server = new MyServer();
Naming.rebind("//localhost:2020/RemoteDataPointHandler", server);

当使用rmi://localhost:2020/RemoteDataPointHandler时也不起作用。使用默认端口也不起作用。我还尝试使用127.0.0.1 IP地址,但效果相同。

我的运行时参数:

-Djava.security.policy=java.security.AllPermission
在主线程中发生异常: java.rmi.ConnectException: 拒绝连接到主机: localhost;嵌套异常是: 
    java.net.ConnectException: 拒绝连接
    at sun.rmi.transport.tcp.TCPEndpoint.newSocket(TCPEndpoint.java:574)
    at sun.rmi.transport.tcp.TCPChannel.createConnection(TCPChannel.java:185)
    at sun.rmi.transport.tcp.TCPChannel.newConnection(TCPChannel.java:171)
    at sun.rmi.server.UnicastRef.newCall(UnicastRef.java:306)
    at sun.rmi.registry.RegistryImpl_Stub.rebind(Unknown Source)
    at java.rmi.Naming.rebind(Naming.java:160)
    at be.fortega.knx.server.Main.(Main.java:25)
    at be.fortega.knx.server.Main.main(Main.java:16)
原因是:java.net.ConnectException: 拒绝连接
    at java.net.PlainSocketImpl.socketConnect(Native Method)
    at java.net.PlainSocketImpl.doConnect(PlainSocketImpl.java:333)
    at java.net.PlainSocketImpl.connectToAddress(PlainSocketImpl.java:195)
    at java.net.PlainSocketImpl.connect(PlainSocketImpl.java:182)
    at java.net.SocksSocketImpl.connect(SocksSocketImpl.java:433)
    at java.net.Socket.connect(Socket.java:524)
    at java.net.Socket.connect(Socket.java:474)
    at java.net.Socket.(Socket.java:371)
    at java.net.Socket.(Socket.java:184)
    at sun.rmi.transport.proxy.RMIDirectSocketFactory.createSocket(RMIDirectSocketFactory.java:22)
    at sun.rmi.transport.proxy.RMIMasterSocketFactory.createSocket(RMIMasterSocketFactory.java:128)
    at sun.rmi.transport.tcp.TCPEndpoint.newSocket(TCPEndpoint.java:569)
    ... 7 more

1
我认为你在这里有一个时间问题。当你重新绑定时,注册表还没有完全启动。 - user207421
5个回答

13

我也遇到了连接异常的相似问题。当注册表尚未启动(就像你的情况一样)或者注册表已经取消导出(就像我的情况一样)时,就会抛出该异常。

但是有一个简短的注释来说明启动注册表的两种方式之间的区别:

Runtime.getRuntime().exec("rmiregistry 2020");

在新的进程中运行javas bin目录中的rmiregistry.exe,并与您的java代码并行继续。

LocateRegistry.createRegistry(2020);

RMI方法调用启动注册表,返回该注册表远程对象的引用,然后继续执行下一条语句。

在您的情况下,当您尝试绑定对象时,注册表没有及时启动。


9

当我替换掉

时,它似乎可以工作。
Runtime.getRuntime().exec("rmiregistry 2020");

by

LocateRegistry.createRegistry(2020);

有人知道为什么吗?有何区别?

4
第二个选项将其作为当前JVM中的远程对象启动。第一个选项启动一个单独的进程。 - user207421

6

在尝试连接(注册)RMI服务之前,您需要运行一个rmiregistry

LocateRegistry.createRegistry(2020)方法调用将在指定的端口号上创建并导出一个注册表。

请参阅LocateRegistry文档。


1

我们可以在Windows中注意到一个区别:

如果您使用Runtime.getRuntime().exec("rmiregistry 1024");

您可以看到rmiregistry.exe进程将在任务管理器中运行

而如果您使用Registry registry = LocateRegistry.createRegistry(1024);

您无法在任务管理器中看到该进程正在运行,

我认为Java以不同的方式处理它。

这是我的server.policy文件

在运行应用程序之前,请确保您已关闭所有现有的javaw.exe和与您的rmi程序相对应的rmiregistry.exe。

以下代码对我有效,使用Registry.LocateRegistry()

Runtime.getRuntime.exec("");


// Standard extensions get all permissions by default

grant {
    permission java.security.AllPermission;
};

虚拟机参数

-Djava.rmi.server.codebase=file:\C:\Users\Durai\workspace\RMI2\src\

代码:

package server;    

import java.rmi.Naming;
import java.rmi.RMISecurityManager;
import java.rmi.Remote;
import java.rmi.registry.LocateRegistry;
import java.rmi.registry.Registry;

public class HelloServer 
{
  public static void main (String[] argv) 
  {
    try {

        if(System.getSecurityManager()==null){
            System.setProperty("java.security.policy","C:\\Users\\Durai\\workspace\\RMI\\src\\server\\server.policy");
            System.setSecurityManager(new RMISecurityManager());
        }

 Runtime.getRuntime().exec("rmiregistry 1024");

 //     Registry registry = LocateRegistry.createRegistry(1024);
   //   registry.rebind ("Hello", new Hello ("Hello,From Roseindia.net pvt ltd!"));
   //Process process = Runtime.getRuntime().exec("C:\\Users\\Durai\\workspace\\RMI\\src\\server\\rmi_registry_start.bat");

        Naming.rebind ("//localhost:1024/Hello",new Hello ("Hello,From Roseindia.net pvt ltd!")); 
      System.out.println ("Server is connected and ready for operation.");
    } 
    catch (Exception e) {
      System.out.println ("Server not connected: " + e);
      e.printStackTrace();
    }
  }
}

0

看起来你应该将你的命令设置为 String[],例如:

String[] command = new String[]{"rmiregistry","2020"};
Runtime.getRuntime().exec(command);

它就像是 main(String[] args) 的风格。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接