使用 Netty 实现 RPC 通信框架
使用 Netty 实现 RPC 通信框架
远程过程调用(RPC,Remote Procedure Call) 是分布式系统中非常重要的通信机制。它允许客户端调用远程服务器上的方法,就像调用本地方法一样。RPC 的核心在于屏蔽底层通信细节,使开发者关注业务逻辑。
Netty 作为一个高性能的网络通信框架,非常适合实现 RPC 框架。本篇文章将介绍如何使用 Netty 实现一个简单的 RPC 通信框架。
1. RPC 通信框架基本原理
1.1 核心组成
RPC 框架的核心模块通常包括:
- 服务注册与发现:
- 将服务接口及其实现类的地址注册到中心(如注册中心或简单的服务端映射)。
- 序列化与反序列化:
- 将方法调用、参数等序列化成字节流,传输到远程服务器,服务器再反序列化进行处理。
- 网络通信:
- 使用 Netty 实现客户端和服务端之间的数据传输。
- 动态代理:
- 使用动态代理拦截客户端对接口的调用,将调用信息发送到服务端并返回结果。
1.2 RPC 调用流程
- 客户端:
- 客户端调用代理对象的方法。
- 代理对象将方法、参数打包成 RPC 请求,发送到服务器。
- 服务器:
- 服务器解析 RPC 请求,定位到具体的方法和参数。
- 调用本地方法,获取结果后返回给客户端。
- 客户端:
- 接收服务器的响应,将结果返回给调用者。
2. Netty 实现 RPC 通信框架
2.1 项目结构设计
src/main/java/
├── common/ // 通用模块
│ ├── RpcRequest.java // RPC 请求封装
│ ├── RpcResponse.java // RPC 响应封装
│ ├── Serializer.java // 序列化接口
│ ├── JsonSerializer.java // JSON 序列化实现
├── server/ // 服务端模块
│ ├── RpcServer.java // RPC 服务端
│ ├── ServiceRegistry.java // 服务注册表
├── client/ // 客户端模块
│ ├── RpcClient.java // RPC 客户端
│ ├── RpcProxy.java // 客户端动态代理
2.2 核心代码实现
2.2.1 通用模块
(1) RPC 请求与响应类
RpcRequest 和 RpcResponse 用于封装客户端发送的请求和服务器的响应。
public class RpcRequest {private String methodName; // 方法名private String className; // 类名private Object[] parameters; // 参数private Class<?>[] paramTypes; // 参数类型// Getters and setters
}public class RpcResponse {private Object result; // 方法调用结果private String error; // 错误信息(如果有)// Getters and setters
}
(2) 序列化接口
为确保传输的数据可以跨网络传递,定义序列化与反序列化的接口。
public interface Serializer {byte[] serialize(Object obj); // 序列化<T> T deserialize(byte[] bytes, Class<T> clazz); // 反序列化
}
(3) JSON 序列化实现
使用 Jackson 实现简单的 JSON 序列化。
import com.fasterxml.jackson.databind.ObjectMapper;public class JsonSerializer implements Serializer {private static final ObjectMapper objectMapper = new ObjectMapper();@Overridepublic byte[] serialize(Object obj) {try {return objectMapper.writeValueAsBytes(obj);} catch (Exception e) {throw new RuntimeException("Serialization failed", e);}}@Overridepublic <T> T deserialize(byte[] bytes, Class<T> clazz) {try {return objectMapper.readValue(bytes, clazz);} catch (Exception e) {throw new RuntimeException("Deserialization failed", e);}}
}
2.2.2 服务端模块
(1) 服务注册表
ServiceRegistry 用于存储服务接口与实现类的映射。
import java.util.HashMap;
import java.util.Map;public class ServiceRegistry {private final Map<String, Object> services = new HashMap<>();public void register(String className, Object serviceImpl) {services.put(className, serviceImpl);}public Object getService(String className) {return services.get(className);}
}
(2) RPC 服务端
服务端接收 RPC 请求并调用对应的服务实现。
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.*;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.LengthFieldBasedFrameDecoder;
import io.netty.handler.codec.LengthFieldPrepender;public class RpcServer {private final int port;private final ServiceRegistry serviceRegistry;public RpcServer(int port, ServiceRegistry serviceRegistry) {this.port = port;this.serviceRegistry = serviceRegistry;}public void start() throws InterruptedException {EventLoopGroup bossGroup = new NioEventLoopGroup();EventLoopGroup workerGroup = new NioEventLoopGroup();try {ServerBootstrap bootstrap = new ServerBootstrap();bootstrap.group(bossGroup, workerGroup).channel(NioServerSocketChannel.class).childHandler(new ChannelInitializer<Channel>() {@Overrideprotected void initChannel(Channel ch) throws Exception {ChannelPipeline pipeline = ch.pipeline();pipeline.addLast(new LengthFieldBasedFrameDecoder(65536, 0, 4, 0, 4));pipeline.addLast(new LengthFieldPrepender(4));pipeline.addLast(new RpcServerHandler(serviceRegistry));}});ChannelFuture future = bootstrap.bind(port).sync();System.out.println("RPC Server started on port " + port);future.channel().closeFuture().sync();} finally {bossGroup.shutdownGracefully();workerGroup.shutdownGracefully();}}
}class RpcServerHandler extends SimpleChannelInboundHandler<RpcRequest> {private final ServiceRegistry serviceRegistry;public RpcServerHandler(ServiceRegistry serviceRegistry) {this.serviceRegistry = serviceRegistry;}@Overrideprotected void channelRead0(ChannelHandlerContext ctx, RpcRequest request) throws Exception {Object service = serviceRegistry.getService(request.getClassName());if (service == null) {ctx.writeAndFlush(new RpcResponse(null, "Service not found"));return;}// 调用服务实现Object result = service.getClass().getMethod(request.getMethodName(), request.getParamTypes()).invoke(service, request.getParameters());ctx.writeAndFlush(new RpcResponse(result, null));}
}
2.2.3 客户端模块
(1) RPC 客户端
import io.netty.bootstrap.Bootstrap;
import io.netty.channel.*;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioSocketChannel;public class RpcClient {private final String host;private final int port;public RpcClient(String host, int port) {this.host = host;this.port = port;}public RpcResponse send(RpcRequest request) throws InterruptedException {EventLoopGroup group = new NioEventLoopGroup();RpcResponse response = new RpcResponse();try {Bootstrap bootstrap = new Bootstrap();bootstrap.group(group).channel(NioSocketChannel.class).handler(new ChannelInitializer<Channel>() {@Overrideprotected void initChannel(Channel ch) throws Exception {ChannelPipeline pipeline = ch.pipeline();pipeline.addLast(new RpcClientHandler(response));}});Channel channel = bootstrap.connect(host, port).sync().channel();channel.writeAndFlush(request).sync();channel.closeFuture().sync();} finally {group.shutdownGracefully();}return response;}
}class RpcClientHandler extends SimpleChannelInboundHandler<RpcResponse> {private final RpcResponse response;public RpcClientHandler(RpcResponse response) {this.response = response;}@Overrideprotected void channelRead0(ChannelHandlerContext ctx, RpcResponse msg) throws Exception {response.setResult(msg.getResult());response.setError(msg.getError());}
}
(2) 动态代理
import java.lang.reflect.Proxy;public class RpcProxy {private final RpcClient client;public RpcProxy(RpcClient client) {this.client = client;}@SuppressWarnings("unchecked")public <T> T create(Class<T> serviceClass) {return (T) Proxy.newProxyInstance(serviceClass.getClassLoader(), new Class<?>[]{serviceClass}, (proxy, method, args) -> {RpcRequest request = new RpcRequest();request.setClassName(serviceClass.getName());request.setMethodName(method.getName());request.setParameters(args);request.setParamTypes(method.getParameterTypes());RpcResponse response = client.send(request);if (response.getError() != null) {throw new RuntimeException(response.getError());}return response.getResult();});}
}
2.3 测试示例
-
定义服务接口和实现:
public interface HelloService {String sayHello(String name); }public class HelloServiceImpl implements HelloService {@Overridepublic String sayHello(String name) {return "Hello, " + name;} } -
服务端注册服务并启动:
ServiceRegistry registry = new ServiceRegistry(); registry.register(HelloService.class.getName(), new HelloServiceImpl());RpcServer server = new RpcServer(8080, registry); server.start(); -
客户端调用服务:
RpcClient client = new RpcClient("localhost", 8080); RpcProxy proxy = new RpcProxy(client);HelloService service = proxy.create(HelloService.class); String result = service.sayHello("Netty"); System.out.println(result); // 输出: Hello, Netty
3. 总结
通过上述代码,我们实现了一个简单的基于 Netty 的 RPC 通信框架,涵盖了服务注册、序列化、网络通信和动态代理等核心模块。
关键点回顾:
- 服务端:通过
ServiceRegistry注册服务,并使用 Netty 接收和处理 RPC 请求。 - 客户端:通过动态代理封装 RPC 调用,简化客户端使用。
- 序列化:使用 JSON 进行数据的序列化和反序列化。
该框架可以作为一个简单的模板,在实际应用中可扩展为支持注册中心(如 Zookeeper)、负载均衡、异步调用等高级功能的完整 RPC 框架。
相关文章:
使用 Netty 实现 RPC 通信框架
使用 Netty 实现 RPC 通信框架 远程过程调用(RPC,Remote Procedure Call) 是分布式系统中非常重要的通信机制。它允许客户端调用远程服务器上的方法,就像调用本地方法一样。RPC 的核心在于屏蔽底层通信细节,使开发者关…...
【机器学习06--贝叶斯分类器】
文章目录 基础理解01 贝叶斯决策论02 极大似然估计03 朴素贝叶斯分类器04 半朴素贝叶斯分类器05 贝叶斯网06 EM算法 补充修正1. 贝叶斯定理与分类的基本概念2. 贝叶斯决策论3. 极大似然估计4. 朴素贝叶斯分类器5. 半朴素贝叶斯分类器6. 贝叶斯网7. EM算法 面试常考 基础理解 本…...
创建vue3项目步骤以及安装第三方插件步骤【保姆级教程】
🎙座右铭:得之坦然,失之淡然。 💎擅长领域:前端 是的,我需要您的: 🧡点赞❤️关注💙收藏💛 是我持续下去的动力! 目录 一. 简单汇总一下创建…...
[146 LRU缓存](https://leetcode.cn/problems/lru-cache/)
分析 维护一个双向链表保存缓存中的元素。 如果元素超过容量阈值,则删除最久未使用的元素。为了实现这个功能,将get(), put()方法获取的元素添加到链表首部。 为了在O(1)时间复杂度执行get()方法,再新建一个映射表,缓存key与链表…...
【Java Nio Netty】基于TCP的简单Netty自定义协议实现(万字,全篇例子)
基于TCP的简单Netty自定义协议实现(万字,全篇例子) 前言 有一阵子没写博客了,最近在学习Netty写一个实时聊天软件,一个高性能异步事件驱动的网络应用框架,我们常用的SpringBoot一般基于Http协议࿰…...
【JavaWeb后端学习笔记】Redis常用命令以及Java客户端操作Redis
redis 1、redis安装与启动服务2、redis数据类型3、redis常用命令3.1 字符串String3.2 哈希Hash3.3 列表List3.4 集合Set(无序)3.5 有序集合zset3.6 通用命令 4、使用Java操作Redis4.1 环境准备4.2 Java操作字符串String4.3 Java操作哈希Hash4.4 Java操作…...
pdb调试器详解
文章目录 1. 启动 pdb 调试器1.1 在代码中插入断点1.2 使用命令行直接调试脚本 2. 常用调试命令2.1 基本命令2.2 高级命令2.3 断点操作 3. 调试过程示例4. 调试技巧4.1 条件断点4.2 自动启用调试4.2.1 运行程序时指定 -m pdb4.2.2在代码中启用 pdb.post_mortem4.2.3 使用 sys.e…...
项目15:简易扫雷--- 《跟着小王学Python·新手》
项目15:简易扫雷 — 《跟着小王学Python新手》 《跟着小王学Python》 是一套精心设计的Python学习教程,适合各个层次的学习者。本教程从基础语法入手,逐步深入到高级应用,以实例驱动的方式,帮助学习者逐步掌握Python的…...
Flink CDC实时同步mysql数据
官方参考资料: https://nightlies.apache.org/flink/flink-cdc-docs-master/zh/docs/connectors/flink-sources/mysql-cdc/ Apache Flink 的 Change Data Capture (CDC) 是一种用于捕获数据库变化(如插入、更新和删除操作)的技术。Flink CDC…...
题解 - 自然数无序拆分
题目描述 美羊羊给喜羊羊和沸羊羊出了一道难题,说谁能先做出来,我就奖励给他我自己做的一样礼物。沸羊羊这下可乐了,于是马上答应立刻做出来,喜羊羊见状,当然也不甘示弱,向沸羊羊发起了挑战。 可是这道题目…...
dfs_bool_void 两种写法感悟
dfs 的两种写法 在看之前实现图的遍历 dfs 和拓扑排序 dfs 实现的代码的时候的感悟 图的遍历 dfs 和拓扑排序 dfs 的区别 0 → 1 ↓ ↓ 2 → 3图的邻接表表示: adjList[0] {1, 2}; adjList[1] {3}; adjList[2] {3}; adjList[3] {};正常的 DFS 遍历&#x…...
MySQL 主从复制与 Binlog 深度解析
目录 1. Binlog的工作原理与配置2. 主从复制的设置与故障排除3. 数据一致性与同步延迟的处理 小结 MySQL的binlog(二进制日志)和主从复制是实现数据备份、容灾、负载均衡以及数据同步的重要机制。在高可用性架构和分布式数据库设计中,binlog同…...
大连理工大学《2024年845自动控制原理真题》 (完整版)
本文内容,全部选自自动化考研联盟的:《大连理工大学845自控考研资料》的真题篇。后续会持续更新更多学校,更多年份的真题,记得关注哦 目录 2024年真题 Part1:2024年完整版真题 2024年真题...
Java性能调优 - 多线程性能调优
锁优化 Synchronized 在JDK1.6中引入了分级锁机制来优化Synchronized。当一个线程获取锁时 首先对象锁将成为一个偏向锁,这样做是为了优化同一线程重复获取锁,导致的用户态与内核态的切换问题;其次如果有多个线程竞争锁资源,锁…...
行为树详解(4)——节点参数配置化
【分析】 行为树是否足够灵活强大依赖于足够丰富的各类条件节点和动作节点,在实现这些节点时,不可避免的,节点本身需要有一些参数供配置。 这些参数可以分为静态的固定值的参数以及动态读取设置的参数。 静态参数直接设置为Public即可&…...
计算机网络中的三大交换技术详解与实现
目录 计算机网络中的三大交换技术详解与实现1. 计算机网络中的交换技术概述1.1 交换技术的意义1.2 三大交换技术简介 2. 电路交换技术2.1 理论介绍2.2 Python实现及代码详解2.3 案例分析 3. 分组交换技术3.1 理论介绍3.2 Python实现及代码详解3.3 案例分析 4. 报文交换技术4.1 …...
《杨辉三角》
题目描述 给出 n(1≤n≤20)n(1≤n≤20),输出杨辉三角的前 nn 行。 如果你不知道什么是杨辉三角,可以观察样例找找规律。 输入格式 无 输出格式 无 输入输出样例 输入 #1复制 6 输出 #1复制 1 1 1 1 2 1 1 3 3 1 1 4 6 4 1 1 5 10 10 5 1 C语言…...
ARM学习(35)单元测试框架以及MinGW GCC覆盖率报告
单元测试框架以及MinGW GCC覆盖率报告 1、单元测试与覆盖率简介 随着代码越写越多,越来越需要注意自测的重要性,基本可以提前解决90%的问题,所以就来介绍一下单元测试,单元测试是否测试充分,需要进行评价,覆盖率就是单元测试是否充分的评估工具。 例如跑过单元测试后,…...
边缘计算+人工智能:让设备更聪明的秘密
引言:日常生活中的“智能”设备 你是否发现,身边的设备正变得越来越“聪明”? 早上醒来时,智能音箱已经根据你的日程播放舒缓音乐;走进厨房,智能冰箱提醒你今天的食材库存;而在城市道路上&…...
neo4j知识图谱AOPC的安装方法
AOPC下载链接:aopc全版本github下载 APOC,全称为Awesome Procedures On Cypher,是Neo4j图数据库的一个非常强大和流行的扩展库。它极大地丰富了Cypher查询语言的功能,提供了超过450个过程(procedures)和函数…...
基于服务器使用 apt 安装、配置 Nginx
🧾 一、查看可安装的 Nginx 版本 首先,你可以运行以下命令查看可用版本: apt-cache madison nginx-core输出示例: nginx-core | 1.18.0-6ubuntu14.6 | http://archive.ubuntu.com/ubuntu focal-updates/main amd64 Packages ng…...
【磁盘】每天掌握一个Linux命令 - iostat
目录 【磁盘】每天掌握一个Linux命令 - iostat工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景 注意事项 【磁盘】每天掌握一个Linux命令 - iostat 工具概述 iostat(I/O Statistics)是Linux系统下用于监视系统输入输出设备和CPU使…...
如何在看板中有效管理突发紧急任务
在看板中有效管理突发紧急任务需要:设立专门的紧急任务通道、重新调整任务优先级、保持适度的WIP(Work-in-Progress)弹性、优化任务处理流程、提高团队应对突发情况的敏捷性。其中,设立专门的紧急任务通道尤为重要,这能…...
基础测试工具使用经验
背景 vtune,perf, nsight system等基础测试工具,都是用过的,但是没有记录,都逐渐忘了。所以写这篇博客总结记录一下,只要以后发现新的用法,就记得来编辑补充一下 perf 比较基础的用法: 先改这…...
从零实现STL哈希容器:unordered_map/unordered_set封装详解
本篇文章是对C学习的STL哈希容器自主实现部分的学习分享 希望也能为你带来些帮助~ 那咱们废话不多说,直接开始吧! 一、源码结构分析 1. SGISTL30实现剖析 // hash_set核心结构 template <class Value, class HashFcn, ...> class hash_set {ty…...
CMake 从 GitHub 下载第三方库并使用
有时我们希望直接使用 GitHub 上的开源库,而不想手动下载、编译和安装。 可以利用 CMake 提供的 FetchContent 模块来实现自动下载、构建和链接第三方库。 FetchContent 命令官方文档✅ 示例代码 我们将以 fmt 这个流行的格式化库为例,演示如何: 使用 FetchContent 从 GitH…...
成都鼎讯硬核科技!雷达目标与干扰模拟器,以卓越性能制胜电磁频谱战
在现代战争中,电磁频谱已成为继陆、海、空、天之后的 “第五维战场”,雷达作为电磁频谱领域的关键装备,其干扰与抗干扰能力的较量,直接影响着战争的胜负走向。由成都鼎讯科技匠心打造的雷达目标与干扰模拟器,凭借数字射…...
如何理解 IP 数据报中的 TTL?
目录 前言理解 前言 面试灵魂一问:说说对 IP 数据报中 TTL 的理解?我们都知道,IP 数据报由首部和数据两部分组成,首部又分为两部分:固定部分和可变部分,共占 20 字节,而即将讨论的 TTL 就位于首…...
ABAP设计模式之---“简单设计原则(Simple Design)”
“Simple Design”(简单设计)是软件开发中的一个重要理念,倡导以最简单的方式实现软件功能,以确保代码清晰易懂、易维护,并在项目需求变化时能够快速适应。 其核心目标是避免复杂和过度设计,遵循“让事情保…...
人机融合智能 | “人智交互”跨学科新领域
本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...
