Java实现远程服务生产与消费(RPC)的4种方法

阅读: 评论:0

Java实现远程服务生产与消费(RPC)的4种方法

Java实现远程服务生产与消费(RPC)的4种方法

目录

  • 一. 通过rmi实现远程服务的生产与消费
    • 远程服务提供者实现.
      • 创建rmi-provider项目(Maven)
    • 远程服务消费者实现
      • 创建rmi-consumer项目
  • 二. 通过WebService实现远程服务的生产与消费
    • 远程服务提供者实现.
    • 查看远程服务文档wdsl
    • 远程服务消费者实现.
  • 三. 通过HttpClient实现远程服务的生产与消费
    • 远程服务提供者实现
    • 远程服务消费者实现
  • 四. 通过spring提供的RestTemplate实现远程服务的生产与消费
    • 远程服务消费者实现.

本文将通过具体的远程服务发布与消费案例展示4种RPC远程调用方法.

一. 通过rmi实现远程服务的生产与消费

  • Java自身提供了包, 方便开发者进行远程服务的部署与消费, 下面将通过具体案例进行讲解.

远程服务提供者实现.

创建rmi-provider项目(Maven)

  1. 创建UserService接口.
//将要发布的服务的接口
public interface UserService extends Remote {public String helloRmi(String name) throws RemoteException;
}
  1. 创建UserServiceImpl实现类
  • 注意, UserServiceImpl除了实现UserService接口外, 还要继承UnicastRemoteObject类, 你可以理解为它是一个发布出去供他人调用的类, 当UserServiceImpl实现了这个类后, UserServiceImpl就能被发布出去供别人调用.
//将要发布的服务的实现类
public class UserServiceImpl extends UnicastRemoteObject implements UserService {public UserServiceImpl() throws RemoteException {super();}public String helloRmi(String name) throws RemoteException {return "hello " + name;}
}
  1. 发布远程服务
public static void main(String[] args) {try {//完成远程服务的发布ateRegistry(8888);//将远程服务发布在本地的8888端口String name = "rmi://localhost:8888/rmi";//发布的远程服务被访问的urlUserService userService = new UserServiceImpl();//创建一个提供具体服务的远程对象Naming.bind(name, userService);//给远程服务绑定一个urlSystem.out.println("--- 已发布rmi远程服务 ---");} catch (Exception e) {e.printStackTrace();}
}

远程服务消费者实现

创建rmi-consumer项目

  1. rmi-provider项目种的UserService接口与UserServiceImpl实现类复制到本rmi-consumer项目中.(这一步可以进行优化解耦, 我们可以多创建一个rmi-resource项目, 让rmi-providerrmi-consumer共同依赖rmi-resource项目, 然后把资源文件比如远程服务所用到的UserService等放入rmi-resource项目中)
  2. 远程服务消费者对远程服务发起调用.
public static void main(String[] args) {try {//发布远程服务的访问urlString name = "rmi://localhost:8888/rmi";//通过发布远程服务的url, 获取远程服务的代理对象UserService userService = (UserService) Naming.lookup(name);System.out.println("获得的远程服务的代理对象:" + Class().getName());String result = userService.helloRmi("rmi");//拿到远程方法调用的结果System.out.println("result: " + result);}catch (Exception e) {e.printStackTrace();}
}//最后输出
获得的远程服务的代理对象:com.sun.proxy.$Proxy0
result: hello rmi
  • 通过最后的输出我们看到获得的远程服务对象是动态代理产生的.

 

二. 通过WebService实现远程服务的生产与消费

  • WebService协议是RPC的一种具体实现, 服务提供方和消费方通过http + xml进行通信.

远程服务提供者实现.

  1. 首先创建远程服务接口UserService及其实现类UserServiceImpl.
  • 注意, 使用WebService时需要对远程服务加上注解@WebService
@WebService
public interface UserService {public String sayHello(String name);
}@WebService
public class UserServiceImpl implements UserService {@Overridepublic String sayHello(String name) {return "hello " + name + "~";}
}
  1. 发布远程服务, 过程和rmi差不多, 需要提供远程服务的访问地址和具体的远程服务实现类, 使用Endpoint类的publish()方法进行发布, 这都是JDK封装好的.
public class WsProviderApp {public static void main(String[] args) {//发布的WebService的被访问地址String address = "localhost:9999/ws";//创建远程服务对象UserService userService = new UserServiceImpl();//发布服务Endpoint.publish(address, userService);System.out.println("远程服务已经发布...");}
}

查看远程服务文档wdsl

  • rmi不同的是, WebService发布后, 调用者可以通过查看它的文档对远程服务发起调用.
  • 查看的方法是在浏览器中输入远程服务的访问地址加上?wdsl, 比如本案例中是localhost:9999/ws?wsdl
  • 注意, 在客户端调用远程方法时需要用工具对wdsl文档进行解析, 并获得调用远程方法的工具类. 具体操作见下一段.

远程服务消费者实现.

  1. 首先根据文档获得调用远程服务的工具类, JDK已经为我们封装好了获取的工具, 它在bin目录下, 名字是wsimport
  2. 打开命令行, 在命令行中输入解析命令
wsimport -keep -d C:githubRepositoriesshoppingws-consumersrcmainjava -p com.shenghao.client localhost:9999/ws?wsdl解释:
1. wsimport 是命令的名字
2. -keep 用于保留生成的类, 如果没有该指令会只生成class文件
3. -d 后面接项目中存放这些工具类的包, 填绝对路径
4. -p 填wdsl文档的地址

 

  1. 可以看到命令执行完后, 指定的包中出现一堆相关的类, 最直接调用到的类是UserServiceImplService. 下面演示对远程方法进行调用.
public static void main(String[] args) {//创建服务类对象UserServiceImplService service = new UserServiceImplService();//获得远程服务的代理对象UserServiceImpl userService = UserServiceImplPort();System.out.Class().getName());//对远程服务对象的方法进行调用String result = userService.sayHello("炭烧生蚝");System.out.println(result);
}//结果输出
com.sun.proxy.$Proxy32
hello 炭烧生蚝~

三. 通过HttpClient实现远程服务的生产与消费

  • 这里我们换一个案例进行演示. 假设现在有一套用户系统和一套订单系统, 要实现用户系统访问订单系统以获得某个用户的订单信息.

远程服务提供者实现

  • 提供远程服务的过程和响应web请求很相似, 只不过响应的不是<html>标签, 而是json字符串. 微信小程序前后端通信也是这个原理.
  1. 创建名为order-sys的Maven项目, 指定打包为war包.
点击这里查看l文件, 常规操作

4.3.18.RELEASE1.22.52.02.9.0jstljstl${jstl.version}javax.servletservlet-api${servlet-api.version}providedjavax.servletjsp-api${jsp-api.version}providedorg.springframeworkspring-webmvc${spring.version}com.jackson-databind${jackson.version}at.maventomcat7-maven-plugin2.2/order7070

 

  1. 创建订单类
public class Order {private String id;private Double total;private String date;//get / set ...
}
  1. 对外提供服务, 发布时打包发布到Tomcat
@Controller
public class OrderController {/*** 接收http请求, 响应订单集合, 异步响应* 将list集合序列化为json串响应* @param uid* @return*/@RequestMapping("/loadOrderList2")@ResponseBodypublic List<Order> loadOrderList2(String uid){System.out.println("uid: " + uid);//模拟订单数据Order o1 = new Order();o1.setId("111");o1.setTotal(333.33);o1.setDate("2019-4-29");Order o2 = new Order();o2.setId("222");o2.setTotal(444.44);o2.setDate("2019-5-29");Order o3 = new Order();o3.setId("333");o3.setTotal(555.55);o3.setDate("2019-6-29");List<Order> list = new ArrayList<>();list.add(o1);list.add(o2);list.add(o3);return list;}
}

远程服务消费者实现

  1. 在服务消费端使用HttpClient发送请求, 可以理解为模拟浏览器发送post/get请求. HttpClient为我们封装了拼接一个请求的细节, 使得发送一个请求变得容易.
public static void main(String[] args) throws IOException {//发送远程的http请求的地址String url = "localhost:7070/order/loadOrderList2";//创建HttpClient对象CloseableHttpClient client = ateDefault();//创建HttpPost对象, 发送post请求HttpPost method = new HttpPost(url);//封装发送到服务提供者的参数NameValuePair id = new BasicNameValuePair("uid", "10001");List<NameValuePair> params = new ArrayList<>();params.add(id);//封装请求体数据method.setEntity(new UrlEncodedFormEntity(params, "UTF-8"));//发送具体的http请求HttpResponse response = ute(method);//获得服务提供者响应的具体数据HttpEntity entity = Entity();//获得http的响应体InputStream is = Content();int len = 0;char[] buf = new char[1024];//使用字符流读InputStreamReader reader = new InputStreamReader(is);StringBuffer sb = new StringBuffer();while((len = ad(buf)) != -1){sb.append(String.valueOf(buf, 0, len));}System.out.println(sb);//将响应回来的json字符串解析为Order集合List<Order> list = JSON.String(), Order.class);for(Order o : list){System.out.Id() + "t" + o.getTotal() + "t" + o.getDate());}
}

 

四. 通过spring提供的RestTemplate实现远程服务的生产与消费

  • 通过一个红包系统和订单系统进行演示, 红包系统访问订单系统, 获得某个用户的订单信息, 派发红包.
  • 订单系统继续沿用HttpClient中的订单系统, 通过访问loadOrderList2方法能返回一个订单集合Json字符串.

远程服务消费者实现.

@Controller
public class RedController {//注入由spring提供的RestTemplate对象@Autowiredprivate RestTemplate restTemplate;/*** 发送远程的http请求, 消费http服务* 获得订单对象的集合*/@RequestMapping("/loadOrderList3")@ResponseBodypublic List<ResponseEntity<Order[]>> loadOrderList3(String uid){//发送远程http请求的urlString url = "localhost:7070/order/loadOrderList2";//发送到远程服务的参数MultiValueMap<String, Object> params = new LinkedMultiValueMap<>();params.add("uid", uid);//通过RestTemplate对象发送post请求ResponseEntity<Order[]> entitys = restTemplate.postForEntity(url, params, Order[].class);//查看响应的状态码System.out.StatusCodeValue());//查看响应头HttpHeaders headMap = Headers();for(Map.Entry<String, List<String>> m : Set()){System.out.Key() + ": " + m.getValue());}return Arrays.asList(entitys);}
}

转载于:.html

本文发布于:2024-01-27 19:15:31,感谢您对本站的认可!

本文链接:https://www.4u4v.net/it/17063541312110.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

标签:种方法   实现远程   Java   RPC
留言与评论(共有 0 条评论)
   
验证码:

Copyright ©2019-2022 Comsenz Inc.Powered by ©

网站地图1 网站地图2 网站地图3 网站地图4 网站地图5 网站地图6 网站地图7 网站地图8 网站地图9 网站地图10 网站地图11 网站地图12 网站地图13 网站地图14 网站地图15 网站地图16 网站地图17 网站地图18 网站地图19 网站地图20 网站地图21 网站地图22/a> 网站地图23