网络编程 -- RPC实现原理 -- RPC -- 迭代版本V4 -- 远程方法调用 整合 Spring 自动注册
啦啦啦
V4——RPC -- 远程方法调用 + Spring 自动注册
服务提供商:
1. 配置 rpc04_server.xml 注入 服务提供商 rpcServiceProvider并指定初始化方法、销毁方法 及 服务实例 IUserService
2. 读取 服务消费者 请求的 MethodStaics ,通过反射获取服务端实例方法的返回值。返回值为null值,则映射为NullWritable实例返回。不为null,则不加以约束。
服务消费者:
1. 配置 rpc04_client.xml 注入 服务注册类 ProxyBeanRegistry 并指定 服务提供商地址 及 注册服务 。
2. 服务注册类 实现 BeanFactoryPostProcessor 接口,容器启动后 注入 RemoteRPCClient 实例 及 注册服务代理对象
服务提供商:
XML : rpc04_server.xml
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:p="http://www.springframework.org/schema/p" xsi:schemaLocation=" http://www.springframework.org/schema/beans classpath:/org/springframework/beans/factory/xml/spring-beans-4.1.xsd http://www.springframework.org/schema/context classpath:/org/springframework/context/config/spring-context-4.1.xsd http://mybatis.org/schema/mybatis-spring http://mybatis.org/schema/mybatis-spring.xsd http://www.springframework.org/schema/aop classpath:org/springframework/aop/config/spring-aop-4.1.xsd" default-lazy-init="false"> <bean id="rpcServiceProvider" class="lime.pri.limeNio.netty.rpc04.core.server.RPCServiceProvider" > <constructor-arg index="0" value="9999"/> </bean> <bean id="IUserService" class="lime.pri.limeNio.netty.rpc04.service.impl.UserService"/> </beans>
Class : RPCServiceProvider
package lime.pri.limeNio.netty.rpc04.core.server; import java.lang.reflect.Method; import java.util.List; import org.springframework.beans.BeansException; import org.springframework.beans.factory.DisposableBean; import org.springframework.beans.factory.InitializingBean; import org.springframework.context.ApplicationContext; import org.springframework.context.ApplicationContextAware; import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.serializer.SerializerFeature; import io.netty.bootstrap.ServerBootstrap; import io.netty.buffer.ByteBuf; import io.netty.buffer.Unpooled; import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelFutureListener; import io.netty.channel.ChannelHandlerAdapter; import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelPipeline; import io.netty.channel.EventLoopGroup; import io.netty.channel.nio.NioEventLoopGroup; import io.netty.channel.socket.SocketChannel; import io.netty.channel.socket.nio.NioServerSocketChannel; import io.netty.handler.codec.LengthFieldBasedFrameDecoder; import io.netty.handler.codec.LengthFieldPrepender; import io.netty.handler.codec.MessageToMessageCodec; import io.netty.util.CharsetUtil; import lime.pri.limeNio.netty.rpc04.core.assist.MethodStaics; import lime.pri.limeNio.netty.rpc04.core.assist.NullWritable; public class RPCServiceProvider implements ApplicationContextAware, InitializingBean, DisposableBean { private ServerBootstrap serverBootstrap; private EventLoopGroup boss; private EventLoopGroup worker; private int port; private ApplicationContext act; public RPCServiceProvider() { super(); } public RPCServiceProvider(int port) { this.serverBootstrap = new ServerBootstrap(); this.boss = new NioEventLoopGroup(); this.worker = new NioEventLoopGroup(); this.serverBootstrap.group(boss, worker); this.serverBootstrap.channel(NioServerSocketChannel.class); this.port = port; } public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { this.act = applicationContext; } public void afterPropertiesSet() throws Exception { serverBootstrap.childHandler(new ChannelInitializer<SocketChannel>() { @Override protected void initChannel(SocketChannel ch) throws Exception { ChannelPipeline pipeline = ch.pipeline(); pipeline.addLast(new LengthFieldPrepender(2)) .addLast(new LengthFieldBasedFrameDecoder(65535, 0, 2, 0, 2)) .addLast(new MessageToMessageCodec<ByteBuf, Object>() { @Override protected void encode(ChannelHandlerContext ctx, Object msg, List<Object> out) throws Exception { out.add(Unpooled.buffer() .writeBytes(JSON.toJSONString(msg, SerializerFeature.WriteClassName) .getBytes(CharsetUtil.UTF_8))); } @Override protected void decode(ChannelHandlerContext ctx, ByteBuf msg, List<Object> out) throws Exception { out.add(JSON.parse(msg.toString(CharsetUtil.UTF_8))); } }).addLast(new ChannelHandlerAdapter() { @Override public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception { MethodStaics methodStaics = (MethodStaics) msg; Object invoke = null; try { Object bean = act.getBean(methodStaics.getTargetInterface()); Method method = bean.getClass().getDeclaredMethod(methodStaics.getMethod(), methodStaics.getParameterTypes()); invoke = method.invoke(bean, methodStaics.getArgs());
if(null == invoke)
throw new NullPointException(); } catch (NullPointerException e) { //如果返回值为空,则返回NullWritable实例代替传输
invoke = new NullWritable(); } ChannelFuture channelFuture = ctx.writeAndFlush(invoke); channelFuture.addListener(ChannelFutureListener.FIRE_EXCEPTION_ON_FAILURE); channelFuture.addListener(ChannelFutureListener.CLOSE_ON_FAILURE); channelFuture.addListener(ChannelFutureListener.CLOSE); } @Override public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception { System.err.println(cause); } }); } }); /** * 绑定监听端口并启动服务 注意 : 启动的服务是阻塞的,防止阻塞Spring工厂采用异步启动 */ new Thread() { public void run() { try { System.out.println("服务启动@" + port + " ..."); ChannelFuture channelFuture = serverBootstrap.bind(port).sync(); channelFuture.channel().closeFuture().sync(); } catch (InterruptedException e) { System.out.println(e); } finally { } }; }.start(); } public void destroy() throws Exception { boss.shutdownGracefully(); worker.shutdownGracefully(); } }
Class : IUserService
package lime.pri.limeNio.netty.rpc04.service; import java.util.List; import lime.pri.limeNio.netty.rpc04.entity.User; public interface IUserService { User queryById(Integer id); List<User> queryByName(String name); List<User> queryAll(Integer pageSize, Integer pageNum); }
Class : UserService
package lime.pri.limeNio.netty.rpc04.service.impl; import java.util.ArrayList; import java.util.Date; import java.util.Iterator; import java.util.List; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import lime.pri.limeNio.netty.rpc04.entity.User; import lime.pri.limeNio.netty.rpc04.service.IUserService; public class UserService implements IUserService { private static Map<Integer, User> userMap = new ConcurrentHashMap<Integer, User>(); static { for (int i = 1; i <= 100; i++) { userMap.put(i, new User(i, "lime_" + i, new Date())); } } public User queryById(Integer id) { return userMap.get(id); } public List<User> queryAll(Integer pageSize, Integer pageNum) { int stNum = (pageNum - 1) * pageSize + 1; int enNum = pageNum * pageSize; List<User> result = new ArrayList<User>(); for (int i = stNum; i <= enNum; i++) { result.add(userMap.get(i)); } return result; } public static void main(String[] args) { UserService userService = new UserService(); for (User user : userService.queryAll(10, 2)) { System.out.println(user); } System.out.println(userService.queryById(100)); } public List<User> queryByName(String name) { List<User> result = null; Iterator<User> iterator = userMap.values().iterator(); while (iterator.hasNext()) { User user = iterator.next(); if (user.getName().equals(name)) { if (null == result) result = new ArrayList<User>(); result.add(user); } } return result; } }
服务消费者:
XML : rpc04_client.xml
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:p="http://www.springframework.org/schema/p" xsi:schemaLocation=" http://www.springframework.org/schema/beans classpath:/org/springframework/beans/factory/xml/spring-beans-4.1.xsd http://www.springframework.org/schema/context classpath:/org/springframework/context/config/spring-context-4.1.xsd http://mybatis.org/schema/mybatis-spring http://mybatis.org/schema/mybatis-spring.xsd http://www.springframework.org/schema/aop classpath:org/springframework/aop/config/spring-aop-4.1.xsd" default-lazy-init="false"> <bean class="lime.pri.limeNio.netty.rpc04.core.registry.ProxyBeanRegistry"> <constructor-arg index="0"> <bean class="lime.pri.limeNio.netty.rpc04.core.client.assist.HostAndPort"> <constructor-arg index="0" value="127.0.0.1" /> <constructor-arg index="1" value="9999" /> </bean> </constructor-arg> <property name="proxyBeanMap"> <map> <entry key="userService" value="lime.pri.limeNio.netty.rpc04.service.IUserService" /> <entry key="personService" value="lime.pri.limeNio.netty.rpc04.service.IUserService" /> </map> </property> </bean> </beans>
Class : ProxyBeanRegistry
package lime.pri.limeNio.netty.rpc04.core.registry; import java.util.Map; import org.springframework.beans.BeansException; import org.springframework.beans.factory.config.BeanFactoryPostProcessor; import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; import org.springframework.beans.factory.support.AbstractBeanDefinition; import org.springframework.beans.factory.support.BeanDefinitionBuilder; import org.springframework.beans.factory.support.DefaultListableBeanFactory; import org.springframework.util.CollectionUtils; import lime.pri.limeNio.netty.rpc04.core.client.assist.HostAndPort; import lime.pri.limeNio.netty.rpc04.core.client.proxy.RPCObjectProxy; import lime.pri.limeNio.netty.rpc04.core.client.rpcClient.impl.RemoteRPCClient; /** * 通过postProcessBeanFactory()向beanFactory注册实体Bean * * @author lime * */ public class ProxyBeanRegistry implements BeanFactoryPostProcessor { private Map<String, Class<?>> proxyBeanMap; private HostAndPort hap; public ProxyBeanRegistry() { super(); } public ProxyBeanRegistry(HostAndPort hap) { super(); this.hap = hap; } public void postProcessBeanFactory(ConfigurableListableBeanFactory listableBeanFactory) throws BeansException { DefaultListableBeanFactory defaultListableBeanFactory = (DefaultListableBeanFactory) listableBeanFactory; // 配置RemoteRPCClient实例:HostAndPort BeanDefinitionBuilder remoteRPCClient = BeanDefinitionBuilder.genericBeanDefinition(RemoteRPCClient.class); remoteRPCClient.addConstructorArgValue(hap); AbstractBeanDefinition remoteRPCClientBeanDefinition = remoteRPCClient.getBeanDefinition(); defaultListableBeanFactory.registerBeanDefinition("remoteRPCClient", remoteRPCClientBeanDefinition); // 配置RPCObjectProxy实例:注册的接口、服务消费者 if(!CollectionUtils.isEmpty(proxyBeanMap)){ for(Map.Entry<String, Class<?>> entry : proxyBeanMap.entrySet()){ String beanName = entry.getKey(); //注册Bean的名字 Class<?> beanClazz = entry.getValue(); //需要代理的接口 BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(RPCObjectProxy.class); beanDefinitionBuilder.addConstructorArgReference("remoteRPCClient"); beanDefinitionBuilder.addConstructorArgValue(beanClazz); AbstractBeanDefinition beanDefinition = beanDefinitionBuilder.getBeanDefinition(); defaultListableBeanFactory.registerBeanDefinition(beanName, beanDefinition); } } } public void setProxyBeanMap(Map<String, Class<?>> proxyBeanMap) { this.proxyBeanMap = proxyBeanMap; } }
Class : RPCClient
package lime.pri.limeNio.netty.rpc04.core.client.rpcClient; import org.springframework.beans.factory.DisposableBean; import lime.pri.limeNio.netty.rpc04.core.assist.MethodStaics; /** * 通过RPCClient实现对远程方法的调用 * * @author lime * */ public interface RPCClient extends DisposableBean { Object invokeMethod(MethodStaics methodStaics); }
Class : RemoteRPCClient
package lime.pri.limeNio.netty.rpc04.core.client.rpcClient.impl; import java.net.InetSocketAddress; import java.util.List; import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.serializer.SerializerFeature; import io.netty.bootstrap.Bootstrap; import io.netty.buffer.ByteBuf; import io.netty.buffer.Unpooled; import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelFutureListener; import io.netty.channel.ChannelHandlerAdapter; import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelPipeline; import io.netty.channel.EventLoopGroup; import io.netty.channel.nio.NioEventLoopGroup; import io.netty.channel.socket.SocketChannel; import io.netty.channel.socket.nio.NioSocketChannel; import io.netty.handler.codec.LengthFieldBasedFrameDecoder; import io.netty.handler.codec.LengthFieldPrepender; import io.netty.handler.codec.MessageToMessageCodec; import io.netty.util.CharsetUtil; import lime.pri.limeNio.netty.rpc04.core.assist.MethodStaics; import lime.pri.limeNio.netty.rpc04.core.assist.NullWritable; import lime.pri.limeNio.netty.rpc04.core.client.assist.HostAndPort; import lime.pri.limeNio.netty.rpc04.core.client.rpcClient.RPCClient; /** * 通过TCP/IP协议实现远程方法调用 * * @author lime * */ public class RemoteRPCClient implements RPCClient { private Bootstrap bootstrap; private EventLoopGroup worker; private HostAndPort hostAndPort; public RemoteRPCClient() { super(); } public RemoteRPCClient(HostAndPort hostAndPost) { this.hostAndPort = hostAndPost; // 初始化数据 this.bootstrap = new Bootstrap(); this.worker = new NioEventLoopGroup(); this.bootstrap.group(this.worker); this.bootstrap.channel(NioSocketChannel.class); } public Object invokeMethod(final MethodStaics methodStaics) { bootstrap.handler(new ChannelInitializer<SocketChannel>() { @Override protected void initChannel(SocketChannel ch) throws Exception { ChannelPipeline pipeline = ch.pipeline(); pipeline.addLast(new LengthFieldBasedFrameDecoder(65535, 0, 2, 0, 2)) .addLast(new LengthFieldPrepender(2)).addLast(new MessageToMessageCodec<ByteBuf, Object>() { @Override protected void encode(ChannelHandlerContext ctx, Object msg, List<Object> out) throws Exception { out.add(Unpooled.buffer() .writeBytes(JSON.toJSONString(msg, SerializerFeature.WriteClassName) .getBytes(CharsetUtil.UTF_8))); } @Override protected void decode(ChannelHandlerContext ctx, ByteBuf msg, List<Object> out) throws Exception { out.add(JSON.parse(msg.toString(CharsetUtil.UTF_8))); } }).addLast(new ChannelHandlerAdapter() { @Override public void channelActive(ChannelHandlerContext ctx) throws Exception { ChannelFuture channelFuture = ctx.writeAndFlush(methodStaics); channelFuture.addListener(ChannelFutureListener.FIRE_EXCEPTION_ON_FAILURE); channelFuture.addListener(ChannelFutureListener.CLOSE_ON_FAILURE); } @Override public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception { methodStaics.setResult(msg); } @Override public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception { System.err.println(cause); } }); } }); ChannelFuture channelFuture; try { channelFuture = bootstrap.connect(new InetSocketAddress(hostAndPort.getHost(), hostAndPort.getPort())) .sync(); channelFuture.channel().closeFuture().sync(); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } // 服务端返回值为null,处理方式 return methodStaics.getResult() instanceof NullWritable ? null : methodStaics.getResult(); } public void destroy() throws Exception { worker.shutdownGracefully(); } }
Class : RPCObjectProxy
package lime.pri.limeNio.netty.rpc04.core.client.proxy; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import org.springframework.beans.factory.FactoryBean; import lime.pri.limeNio.netty.rpc04.core.assist.MethodStaics; import lime.pri.limeNio.netty.rpc04.core.client.rpcClient.RPCClient; /** * 通过接口动态创建代理对象 * * @author lime * @param <T> * * 实现FactoryBean接口,与Spring整合 * */ public class RPCObjectProxy implements InvocationHandler, FactoryBean<Object> { private RPCClient rpcClient; private Class<?> targetInterface; public RPCObjectProxy() { super(); } public RPCObjectProxy(RPCClient rpcClient, Class<?> targetInterface) { super(); this.rpcClient = rpcClient; this.targetInterface = targetInterface; } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { return rpcClient .invokeMethod(new MethodStaics(targetInterface, method.getName(), args, method.getParameterTypes())); } // 产生代理对象 public Object getObject() throws Exception { return Proxy.newProxyInstance(RPCObjectProxy.class.getClassLoader(), new Class[] { targetInterface }, this); } public Class<?> getObjectType() { return targetInterface; } public boolean isSingleton() { return true; } }
Class : HostAndPort
package lime.pri.limeNio.netty.rpc04.core.client.assist; import java.io.Serializable; public class HostAndPort implements Serializable{ /** * */ private static final long serialVersionUID = 1L; private String host; private int port; public HostAndPort() { super(); // TODO Auto-generated constructor stub } public HostAndPort(String host, int port) { super(); this.host = host; this.port = port; } public String getHost() { return host; } public void setHost(String host) { this.host = host; } public int getPort() { return port; } public void setPort(int port) { this.port = port; } @Override public String toString() { return "HostAndPort [host=" + host + ", port=" + port + "]"; } }
辅助类 :
Class : MethodStaics
package lime.pri.limeNio.netty.rpc04.core.assist; import java.io.Serializable; import java.util.Arrays; /** * @author lime * */ public class MethodStaics implements Serializable{ /** * */ private static final long serialVersionUID = 1L; private Class<?> targetInterface; private String method; private Object[] args; private Class[] parameterTypes; private Object result; public MethodStaics() { super(); // TODO Auto-generated constructor stub } public MethodStaics(Class<?> targetInterface, String method, Object[] args, Class[] parameterTypes) { super(); this.targetInterface = targetInterface; this.method = method; this.args = args; this.parameterTypes = parameterTypes; } @Override public String toString() { return "MethodStaics [targetInterface=" + targetInterface + ", method=" + method + ", args=" + Arrays.toString(args) + ", parameterTypes=" + Arrays.toString(parameterTypes) + "]"; } public Class<?> getTargetInterface() { return targetInterface; } public void setTargetInterface(Class<?> targetInterface) { this.targetInterface = targetInterface; } public String getMethod() { return method; } public void setMethod(String method) { this.method = method; } public Object[] getArgs() { return args; } public void setArgs(Object[] args) { this.args = args; } public Class[] getParameterTypes() { return parameterTypes; } public void setParameterTypes(Class[] parameterTypes) { this.parameterTypes = parameterTypes; } public Object getResult() { return result; } public void setResult(Object result) { this.result = result; } }
Class : NullWritable
package lime.pri.limeNio.netty.rpc04.core.assist; import java.io.Serializable; public class NullWritable implements Serializable{ /** * */ private static final long serialVersionUID = 1L; }
Class : User
package lime.pri.limeNio.netty.rpc04.entity; import java.io.Serializable; import java.util.Date; public class User implements Serializable{ /** * */ private static final long serialVersionUID = 1L; private int id; private String name; private Date birth; public User() { super(); // TODO Auto-generated constructor stub } public User(int id, String name, Date birth) { super(); this.id = id; this.name = name; this.birth = birth; } @Override public String toString() { return "User [id=" + id + ", name=" + name + ", birth=" + birth + "]"; } public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public Date getBirth() { return birth; } public void setBirth(Date birth) { this.birth = birth; } }
测试类 :
Class : TtServer
package lime.pri.limeNio.netty.rpc04.tT; import org.springframework.context.support.ClassPathXmlApplicationContext; public class TtServer { public static void main(String[] args) throws Exception { new ClassPathXmlApplicationContext("classpath:spring/rpc04_server.xml"); } }
Class :
package lime.pri.limeNio.netty.rpc04.tT; import java.util.List; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; import lime.pri.limeNio.netty.rpc04.entity.User; import lime.pri.limeNio.netty.rpc04.service.IUserService; public class TtClient { public static void main(String[] args) throws Exception { ApplicationContext ctx = new ClassPathXmlApplicationContext("classpath:spring/rpc04_client.xml"); IUserService userService = (IUserService) ctx.getBean("userService"); User user = userService.queryById(23); System.out.println(user); System.out.println("-- --"); IUserService personService = (IUserService) ctx.getBean("personService"); List<User> list = personService.queryAll(10, 4); for(User u : list){ System.out.println(u); } } }
Console : Server
六月 25, 2017 3:35:47 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh 信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@17f052a3: startup date [Sun Jun 25 15:35:47 CST 2017]; root of context hierarchy 六月 25, 2017 3:35:47 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions 信息: Loading XML bean definitions from class path resource [spring/rpc04_server.xml] 服务启动@9999 ...
Console : Client
六月 25, 2017 3:52:39 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh 信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@17f052a3: startup date [Sun Jun 25 15:52:39 CST 2017]; root of context hierarchy 六月 25, 2017 3:52:39 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions 信息: Loading XML bean definitions from class path resource [spring/rpc04_client.xml] User [id=23, name=lime_23, birth=Sun Jun 25 15:52:10 CST 2017] -- -- User [id=31, name=lime_31, birth=Sun Jun 25 15:52:10 CST 2017] User [id=32, name=lime_32, birth=Sun Jun 25 15:52:10 CST 2017] User [id=33, name=lime_33, birth=Sun Jun 25 15:52:10 CST 2017] User [id=34, name=lime_34, birth=Sun Jun 25 15:52:10 CST 2017] User [id=35, name=lime_35, birth=Sun Jun 25 15:52:10 CST 2017] User [id=36, name=lime_36, birth=Sun Jun 25 15:52:10 CST 2017] User [id=37, name=lime_37, birth=Sun Jun 25 15:52:10 CST 2017] User [id=38, name=lime_38, birth=Sun Jun 25 15:52:10 CST 2017] User [id=39, name=lime_39, birth=Sun Jun 25 15:52:10 CST 2017] User [id=40, name=lime_40, birth=Sun Jun 25 15:52:10 CST 2017] -- -- null -- --
啦啦啦