Java RMI
Remote method invocation(RMI) allow a java object to invoke method on an object running on
another machine. RMI provide remote communication between java program. RMI is used for
building distributed application.
Concept of RMI application
A RMI application can be divided into two part,Client program and Server program.
A Server program creates some remote object, make their references available for the client to
invoke method on it. A Client program make request for remote objects on server and invoke
method on them. Stub and Skeleton are two important object used for communication with remote
object.
Stub and Skeleton
Stub act as a gateway for Client program. It resides on Client side and communicate
with Skeleton object. It establish the connection between remote object and transmit request to it.
Skeleton object resides on server program. It is responsible for passing request from Stub to remote
object.
     Steps to write the RMI program
     The is given the 6 steps to write the RMI program.
        1. Create the remote interface
        2. Provide the implementation of the remote interface
        3. Compile the implementation class and create the stub and skeleton objects using the
           rmic tool
        4. Start the registry service by rmiregistry tool
        5. Create and start the remote application
        6. Create and start the client application
     1) create the remote interface
     For creating the remote interface, extend the Remote interface and declare the
     RemoteException with all the methods of the remote interface. Here, we are creating a
     remote interface that extends the Remote interface. There is only one method named add()
     and it declares RemoteException.
1.   import java.rmi.*;
2.   public interface Adder extends Remote{
3.   public int add(int x,int y)throws RemoteException;
4.   }
     2) Provide the implementation of the remote
     interface
     Now provide the implementation of the remote interface. For providing the implementation
     of the Remote interface, we need to
           Either extend the UnicastRemoteObject class,
           or use the exportObject() method of the UnicastRemoteObject class
     In case, you extend the UnicastRemoteObject class, you must define a constructor that
     declares RemoteException.
1.   import java.rmi.*;
2.   import java.rmi.server.*;
3.   public class AdderRemote extends UnicastRemoteObject implements Adder{
4.   AdderRemote()throws RemoteException{
5.   super();
6.   }
7.   public int add(int x,int y){return x+y;}
8.   }
   3) create the stub and skeleton objects using the rmic tool.
   Next step is to create stub and skeleton objects using the rmi compiler. The rmic tool
   invokes the RMI compiler and creates stub and skeleton objects.
1. rmic AdderRemote
   4) Start the registry service by the rmiregistry tool
   Now start the registry service by using the rmiregistry tool. If you don't specify the port
   number, it uses a default port number. In this example, we are using the port number
   5000.
1. Start rmiregistry 5000
   5) Create and run the server application
   Now rmi services need to be hosted in a server process. The Naming class provides methods
   to get and store the remote object. The Naming class provides 5 methods.
      1. public static java.rmi.Remote lookup(java.lang.String) throws
         java.rmi.NotBoundException, java.net.MalformedURLException,
         java.rmi.RemoteException; it returns the reference of the remote object.
      2. public static void bind(java.lang.String, java.rmi.Remote) throws
         java.rmi.AlreadyBoundException, java.net.MalformedURLException,
         java.rmi.RemoteException; it binds the remote object with the given name.
      3. public static void unbind(java.lang.String) throws
         java.rmi.RemoteException, java.rmi.NotBoundException,
         java.net.MalformedURLException; it destroys the remote object which is bound
         with the given name.
      4. public static void rebind(java.lang.String, java.rmi.Remote) throws
         java.rmi.RemoteException, java.net.MalformedURLException; it binds the
         remote object to the new name.
      5. public static java.lang.String[] list(java.lang.String) throws
         java.rmi.RemoteException, java.net.MalformedURLException; it returns an
         array of the names of the remote objects bound in the registry.
   In this example, we are binding the remote object by the name sonoo.
1. import java.rmi.*;
2. import java.rmi.registry.*;
3. public class MyServer{
4. public static void main(String args[]){
5. try{
6. Adder stub=new AdderRemote();
7. Naming.rebind("rmi://localhost:5000/sonoo",stub);
8. }catch(Exception e){System.out.println(e);}
9. }
10. }
     6) Create and run the client application
     At the client we are getting the stub object by the lookup() method of the Naming class and
     invoking the method on this object. In this example, we are running the server and client
     applications, in the same machine so we are using localhost. If you want to access the
     remote object from another machine, change the localhost to the host name (or IP address)
     where the remote object is located.
1.   import java.rmi.*;
2.   public class MyClient{
3.   public static void main(String args[]){
4.   try{
5.   Adder stub=(Adder)Naming.lookup("rmi://localhost:5000/sonoo");
6.   System.out.println(stub.add(34,4));
7.   }catch(Exception e){}
8.   }
9.   }
1. For running this rmi example,
2.
3. 1) compile all the java files
4.
5. javac *.java
6.
7. 2)create stub and skeleton object by rmic tool
8.
9. rmic AdderRemote
10.
11. 3)start rmi registry in one command prompt
12.
13. Start rmiregistry 5000
14.
15. 4)start the server in another command prompt
16.
17. java MyServer
18.
19. 5)start the client application in another command prompt
20.
21. java MyClient