1)什麼是rmi
2)簡單的實現rmi
3)rmi原理
4)手寫rmi框架java
Remote procedure call protocal 遠程過程調用協議
不用知道具體細節,調用遠程系統中類的方法,就跟調用本地方法同樣。
RPC協議實際上是一種規範。
包括Dubbo,Thrift,rmi,webservice,hessainweb
網絡協議和網絡IO對於調用端和服務端來講是透明的。安全
一個RPC框架應該包含的要素:bash
rmi(remote method invocation) 遠程方法調用
能夠認爲是RPC的java版本服務器
RMI使用的是JRMP(JAVA Remote Messageing Protocol),能夠說JRMP是專門爲java定製的通訊協議,因此它是純java的分佈式解決方案。網絡
package com.llf.rmidemo;
import java.rmi.Remote;
import java.rmi.RemoteException;
public interface SayHello extends Remote{
public String sayHello(String name)throws RemoteException;
}
複製代碼
package com.llf.rmidemo;
import java.rmi.RemoteException;
import java.rmi.server.UnicastRemoteObject;
public class SayHelloImpl extends UnicastRemoteObject implements SayHello{
protected SayHelloImpl() throws RemoteException {
}
@Override
public String sayHello(String name) throws RemoteException {
return "Hello LLF -->"+name;
}
}
複製代碼
package com.llf.rmidemo;
import java.net.MalformedURLException;
import java.rmi.AlreadyBoundException;
import java.rmi.Naming;
import java.rmi.RemoteException;
import java.rmi.registry.LocateRegistry;
public class HelloServer {
public static void main(String[] args) {
try {
SayHello hello=new SayHelloImpl();
LocateRegistry.createRegistry(8888);
try {
Naming.bind("rmi://localhost:8888/sayhello", hello);
System.out.println("Server start success!");
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (AlreadyBoundException e) {
e.printStackTrace();
}
} catch (RemoteException e) {
e.printStackTrace();
}
}
}
複製代碼
package com.llf.rmidemo;
import java.net.MalformedURLException;
import java.rmi.Naming;
import java.rmi.NotBoundException;
import java.rmi.RemoteException;
public class HelloClient {
public static void main(String[] args) {
try {
SayHello hello=(SayHello) Naming.lookup("rmi://localhost:8888/sayhello");
System.out.println(hello.sayHello("FXP"));
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (RemoteException e) {
e.printStackTrace();
} catch (NotBoundException e) {
e.printStackTrace();
}
}
}
複製代碼
1)編寫服務器程序,暴露監聽,能夠使用socket 2)編寫客戶端程序,經過IP和端口鏈接到指定的服務,而且把咱們的數據作封裝(序列化) 3)服務器端收到請求先反序列化在進行業務邏輯處理,把返回結果序列化返回框架
咱們近乎的能夠以以下的圖來理解: a) stub和skeleton這倆個身份都是做爲代理存在,客戶端的稱爲stub,服務端的稱爲skeleton,經過這倆個對象屏蔽了遠程方法調用的具體細節,這倆個是必不可少的。 b)Registry:註冊所,提供了服務名到服務的映射。socket
客戶端分佈式
用代碼來模擬RMI底層過程以下: 新建一個User的對象ide
package com.llf.rmi;
public class User {
private int age;
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
複製代碼
編寫一個Skeleton類供客戶端調用【這塊是rmi定義出來屏蔽底層序列化及流鏈接的,這邊模擬寫了下底層的序列化及流】
package com.llf.rmi;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.net.ServerSocket;
import java.net.Socket;
//server程序
public class User_Skeleton extends Thread {
private UserServer userServer;
public User_Skeleton(UserServer userServer) {
this.userServer = userServer;
}
public void run() {
ServerSocket serverSocket = null;
ObjectInputStream read = null;
ObjectOutputStream oos = null;
Socket socket=null;
try {
serverSocket = new ServerSocket(8888);
socket = serverSocket.accept();
while (socket != null) {
read = new ObjectInputStream(socket.getInputStream());
String method = (String) read.readObject();
if (method.equals("age")) {
int age = userServer.getAge();
oos = new ObjectOutputStream(socket.getOutputStream());
oos.writeInt(age);
oos.flush();
}
}
} catch (Exception e) {
e.printStackTrace();
} finally {
if (serverSocket != null) {
try {
oos.close();
read.close();
socket.close();
serverSocket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
複製代碼
寫一個stub
package com.llf.rmi;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.net.Socket;
import java.net.UnknownHostException;
public class User_Stub extends User {
private Socket socket;
public User_Stub() {
try {
socket=new Socket("localhost", 8888);
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
public int getAge(){
ObjectOutputStream oos=null;
ObjectInputStream ois=null;
try {
oos=new ObjectOutputStream(socket.getOutputStream());
oos.writeObject("age");
oos.flush();
ois=new ObjectInputStream(socket.getInputStream());
return ois.readInt();
} catch (IOException e) {
e.printStackTrace();
}finally {
try {
ois.close();
oos.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return 0;
}
}
複製代碼
編寫服務器代碼
package com.llf.rmi;
public class UserServer extends User{
public static void main(String[] args) {
UserServer server=new UserServer();
server.setAge(18);
//模擬rmi生成的skeleton代理對象
User_Skeleton user_Skeleton=new User_Skeleton(server);
user_Skeleton.start();
}
}
複製代碼
編寫客戶端代碼
package com.llf.rmi;
public class UserClient {
public static void main(String[] args) {
User user=new User_Stub();
int age=user.getAge();
System.out.println(age);
}
}
複製代碼