Java RMI第一次调用缓慢

时间:2017-01-09 08:48:41

标签: java macos ubuntu rmi

我正在为学校开展个人项目,我必须使用RMI在服务器和客户端之间进行通信。

项目信息

我的项目的目标是在特定时间(纽约证券交易所关闭后)检索服务器上每一天的股票信息(来自纽约证券交易所)。每个库存对象都保存在数据库中。信息通过http检索,与RMI无关。

对于客户来说,也可以获取股票。当用户想要获取当天的股票对象时,它将直接从第3方服务获取。例如,当用户想要从上个月获取Google的股票时,会在服务器上通过RMI请求它。服务器将在数据库中查找stock对象并检索Stock对象并将其发送给客户端。

问题

当我启动客户端应用程序时,我必须登录。客户端将创建包含用户名和密码的User对象。 当我按下登录按钮时,将显示主屏幕前约2分钟。

在我设置RMI连接的源代码下面。

服务器(main.java)

public static void main(String[] args) throws UnknownHostException {

    InetAddress IP= InetAddress.getLocalHost();
    System.out.println("IP of my system is := "+IP.getHostAddress());

    if(args.length == 1 && args[0].toLowerCase().equals("local")) {
        System.out.println("Running on localhost");
        System.setProperty("java.rmi.server.hostname", IP.getHostAddress());
    } else {
        System.out.println("rmi hostname is set to 37.97.223.70");
        System.setProperty("java.rmi.server.hostname", "37.97.223.70");
    }

    try {
        Registry reg = LocateRegistry.createRegistry(1099);
        StockAppServer server = StockAppServer.getInstance();
        reg.rebind("StockApp", server);
        System.out.println("StockApp bound for StockAppServer object.");
    } catch (RemoteException e) {
        e.printStackTrace();
    }
}

根据应用程序启动时传递给它的参数,我将RMI主机名设置为当前的IP地址,或者设置为远程服务器地址。远程服务器地址是静态IP,因此不会更改。

服务器(StockAppServer.java)

此类实现客户端用于调用服务器上的方法的接口。所以这个类扩展了UnicastRemoteObject。当我启动服务器时,将调用registerStockTask()。此方法将获取股票代码(What are ticker symbols?),然后安排任务在特定时间获取所有股票对象。

private static StockAppServer _instance;
private List<User> loggedInUsers;
private List<Group> activeGroups;
private List<Notification> registeredNotifications;

private StockAppServer() throws IOException {
    _instance = this;

    this.loggedInUsers = new ArrayList<>();
    this.activeGroups = new ArrayList<>();
    this.registeredNotifications = new ArrayList<>();

    this.registerStockTask();

    clearActiveGroups();
    checkForCompletedNotifications();

    // Start the restful framework to allow incoming connections from the NodeJS server to manage new notification
    Router.getInstance();
}

public static StockAppServer getInstance() {
    try{
        return _instance == null ? new StockAppServer() : _instance;
    } catch (IOException e) {
        e.printStackTrace();
    }

    return null;
}

客户端(main.java)

public static void main(String[] arguments) throws Exception {
    args = arguments;
    Application.launch();
}

@Override
public void start(Stage primaryStage) throws Exception {
    InetAddress IP= InetAddress.getLocalHost();
    System.out.println("IP of my system is := "+IP.getHostAddress());



    if(args.length == 1 && args[0].toLowerCase().equals("local")) {
        // Program started with local command, expect that server is running on local host
        reg = LocateRegistry.getRegistry(IP.getHostAddress(), 1099);
        System.out.println("Attempting to connect to RMI server over 127.0.0.1");
    } else {
        // Program started without additional commands. Except that "the server" is available;
        reg = LocateRegistry.getRegistry("37.97.223.70", 1099);
        System.out.println("Attempting to connect to RMI server over 37.97.223.70");
    }

    try {
        StockApp.getInstance().setServerInterfaces((IStockSend) reg.lookup("StockApp"), (IUserHandling) reg.lookup("StockApp"));
    } catch(RemoteException e) {
        AlertMessage.showException("Unable to connect to server.", e);
    } catch (NotBoundException e) {
        AlertMessage.showException("No server has been found with the name \"StockApp\" on the remote host.\nPlease try again later", e);
    }

    LoginController.showMenu();

    //FileNotFoundException e = new FileNotFoundException("Couldn't find file blabla.txt");
    //AlertMessage.showException("Something went wrong. Please try again later.", e);
}

我是如何解决问题的

当我在本地测试我的应用程序时,没有问题。登录方法将在几毫秒内完成,我将代表主屏幕。

  • 我开始在Macbook上打开防火墙。没有结果,登录方法仍然需要大约2秒钟。
  • 我关闭了我的Ubuntu服务器的防火墙。没有结果,服务器和macbook上的防火墙都被关闭了。登录方法大约需要2秒钟。
  • 在服务器上运行(感谢jenkins)另一个(不相关的)程序。该程序使用套接字而不是RMI。当该程序未运行时,登录方法仍需要大约2分钟。
  • 在StockAppServer.java中,我调用了以下方法:

    超级(1099); 这与我采取的上述步骤具有相同的结果。

我不知道还有什么可以解决我的问题。

我试图为RMI部分提供尽可能多的代码。我需要任何其他源代码,只是问我可以更新这个问题。此外,源代码可通过github:https://github.com/juleskreutzer/GSO-Maatwerk获得。确保使用-remote param运行程序。

更新1(9-1-2017)

在评论中请求 yanys 时,我应该运行以下命令:

  

dscacheutil -q host -a name localhost

返回以下输出:

的Mac:

name: localhost
ip_address: 127.0.0.1

Ubuntu的:

dscacheutil: command not found

更新2(9-1-2017)

我查看了运行java服务器的VPS提供商。在他们身边一切都应该没问题。据他们说,它不应该是一个DNS问题。经过一些研究,我发现RMI使用DNS和反向DNS。在这种情况下,反向DNS就是问题所在。请查看我如何解决问题的答案。

1 个答案:

答案 0 :(得分:0)

正如EJP在该问题的评论中指出的那样, 是一个DNS问题。

我联系了我的托管服务提供商的支持,看看我是否有错误的设置。他们帮助我解决了这个问题。

首先我们测试了我的VPS的速度,这是大约1000mbit的下载和上传速度。在我们检查完之后,他们说他们身边没有任何问题。

经过一些研究,我发现RMI同时使用DNS和反向DNS。问题是我没有在我的服务器上设置反向DNS。我已经有一个用于反向DNS的域名。

我做了以下事情:

  1. 在我的网站上创建一个指向服务器IP地址的A记录。我把它命名为vps.mydomain.com
  2. 在我的服务器的控制面板中添加反向DNS
  3. 将我的服务器的主机名更改为vps.mydomain.com *
  4. *我的服务器运行Ubuntu 16.04,在带有systemd的ubuntu机器上,你可以使用命令

      

    sudo hostnamectl set-hostname new-name

    更改主机名