网络编程 -- RPC实现原理 -- RPC -- 迭代版本V2 -- 本地方法调用 整合 Spring

网络编程 -- RPC实现原理 -- 目录

  啦啦啦

V2——RPC -- 本地方法调用 + Spring

  1. 配置applicationContext.xml文件 注入 bean 及 管理 bean 之间的依赖关系

  2. RPCObjectProxy 类 实现 FactoryBean<Object> 接口,通过 public Object getObject() throws Exception 返回代理类

  3. List<User> users = userService.queryAll(10, 4); : 调用目标对象的  Object invokeMethod(MethodStaics methodStaics);  方法,通过反射返回指定接口实现方法的返回值。

  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="userServiceProxy" class="lime.pri.limeNio.netty.rpc02.core.RPCObjectProxy">
        <constructor-arg index="0" ref="rpcClient" />
        <constructor-arg index="1" value="lime.pri.limeNio.netty.rpc02.service.IUserService" />
    </bean>

    <bean id="rpcClient" class="lime.pri.limeNio.netty.rpc02.core.impl.LocalRPCClient">
        <property name="beanFactory">
            <map>
                <entry key="IUserService" value-ref="userService"></entry>
            </map>
        </property>
    </bean>

    <bean id="userService" class="lime.pri.limeNio.netty.rpc02.service.impl.UserService"></bean>

</beans>

  Class :RPCObjectProxy

package lime.pri.limeNio.netty.rpc02.core;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

import org.springframework.beans.factory.FactoryBean;

/**
 * 通过接口动态创建代理对象
 *
 * @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 : RPCClient

package lime.pri.limeNio.netty.rpc02.core;

/**
 * 通过RPCClient实现对远程方法的调用
 *
 * @author lime
 *
 */
public interface RPCClient {

    Object invokeMethod(MethodStaics methodStaics);
}

  Class : LocalRPCClient

package lime.pri.limeNio.netty.rpc02.core.impl;

import java.lang.reflect.Method;
import java.util.Map;

import lime.pri.limeNio.netty.rpc02.core.MethodStaics;
import lime.pri.limeNio.netty.rpc02.core.RPCClient;
import lime.pri.limeNio.netty.rpc02.service.IUserService;

public class LocalRPCClient implements RPCClient {

    private Map<String,Object> beanFactory;
    public Object invokeMethod(MethodStaics methodStaics) {
        try {
            IUserService object = (IUserService) beanFactory.get(methodStaics.getTargetInterface().getSimpleName());
            Method method = object.getClass().getDeclaredMethod(methodStaics.getMethod(),
                    methodStaics.getParameterTypes());
            return method.invoke(object, methodStaics.getArgs());
        } catch (Exception e) {
            System.out.println(e);
        }
        return null;
    }
    public Map<String,Object> getBeanFactory() {
        return beanFactory;
    }
    public void setBeanFactory(Map<String,Object> beanFactory) {
        this.beanFactory = beanFactory;
    }

}

  Class : IUserService

package lime.pri.limeNio.netty.rpc02.service;

import java.util.List;

import lime.pri.limeNio.netty.rpc02.entity.User;

public interface IUserService {

    User queryById(Integer id);

    List<User> queryAll(Integer pageSize, Integer pageNum);
}

  Class : UserService

package lime.pri.limeNio.netty.rpc02.service.impl;

import java.util.ArrayList;
import java.util.Calendar;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import lime.pri.limeNio.netty.rpc02.entity.User;
import lime.pri.limeNio.netty.rpc02.service.IUserService;

public class UserService implements IUserService {

    private static Map<Integer, User> userMap = new HashMap<Integer, User>();

    static {
        Calendar calendar = Calendar.getInstance();
        for (int i = 1; i <= 100; i++) {
            calendar.set(Calendar.YEAR, i + 1900);
            userMap.put(i, new User(i, "lime_" + i, calendar.getTime()));
        }
    }

    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));
    }

}

  Class : User

package lime.pri.limeNio.netty.rpc02.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 : MethodStaics

package lime.pri.limeNio.netty.rpc02.core;

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;

    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;
    }
}

  Class : Tt

package lime.pri.limeNio.netty.rpc02.tT;

import java.util.List;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

import lime.pri.limeNio.netty.rpc02.entity.User;
import lime.pri.limeNio.netty.rpc02.service.IUserService;

public class Tt {

    public static void main(String[] args) throws Exception {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("classpath:spring/rpc02.xml");
        IUserService userService = (IUserService) ctx.getBean("userServiceProxy");
        List<User> users = userService.queryAll(10, 3);
        for(User user : users){
            System.out.println(user);
        }
        System.out.println("-- -- ");
        User user = userService.queryById(23);
        System.out.println(user);
    }
}

  Console :

六月 25, 2017 12:07:23 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh
信息: Refreshing org[email protected]17f052a3: startup date [Sun Jun 25 12:07:23 CST 2017]; root of context hierarchy
六月 25, 2017 12:07:23 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
信息: Loading XML bean definitions from class path resource [spring/rpc02.xml]
User [id=21, name=lime_21, birth=Sat Jun 25 12:07:23 CST 1921]
User [id=22, name=lime_22, birth=Sun Jun 25 12:07:23 CST 1922]
User [id=23, name=lime_23, birth=Mon Jun 25 12:07:23 CST 1923]
User [id=24, name=lime_24, birth=Wed Jun 25 12:07:23 CST 1924]
User [id=25, name=lime_25, birth=Thu Jun 25 12:07:23 CST 1925]
User [id=26, name=lime_26, birth=Fri Jun 25 12:07:23 CST 1926]
User [id=27, name=lime_27, birth=Sat Jun 25 12:07:23 CST 1927]
User [id=28, name=lime_28, birth=Mon Jun 25 12:07:23 CST 1928]
User [id=29, name=lime_29, birth=Tue Jun 25 12:07:23 CST 1929]
User [id=30, name=lime_30, birth=Wed Jun 25 12:07:23 CST 1930]
-- --
User [id=23, name=lime_23, birth=Mon Jun 25 12:07:23 CST 1923]

  啦啦啦

时间: 2024-10-17 15:34:19

网络编程 -- RPC实现原理 -- RPC -- 迭代版本V2 -- 本地方法调用 整合 Spring的相关文章

网络编程 -- RPC实现原理 -- RPC -- 迭代版本V1 -- 本地方法调用

网络编程 -- RPC实现原理 -- 目录 啦啦啦 V2--RPC -- 本地方法调用:不通过网络 入门 1. RPCObjectProxy rpcObjectProxy = new RPCObjectProxy(new LocalRPCClient()); : 绑定目标对象 2. IUserService userService = (IUserService) rpcObjectProxy.create(IUserService.class); :返回代理类 3. List<User> u

网络编程 -- RPC实现原理 -- Netty -- 迭代版本V2 -- 对象传输

网络编程 -- RPC实现原理 -- 目录 啦啦啦 V2--Netty -- 使用序列化和反序列化在网络上传输对象 只能传输( ByteBuf, FileRegion )两种类型,因此必须将对象在发送之前进行序列化,放进ByteBuf中,客户端接收到ByteBuf时,将字节码取出,反序列化成对象. Class : Server package lime.pri.limeNio.netty.netty02.exercise; import java.net.InetSocketAddress; i

网络编程 -- RPC实现原理 -- Netty -- 迭代版本V4 -- 粘包拆包

网络编程 -- RPC实现原理 -- 目录 啦啦啦 V2--Netty -- new LengthFieldPrepender(2) : 设置数据包 2 字节的特征码 new LengthFieldBasedFrameDecoder(65535, 0, 2, 0, 2) :  65535 :数据包长度.0:分隔符偏移值.2:分隔符长度.0:数据包偏移值.2:数据包长度. Class : Server package lime.pri.limeNio.netty.netty04; import j

网络编程 -- RPC实现原理 -- Netty -- 迭代版本V3 -- 编码解码

网络编程 -- RPC实现原理 -- 目录 啦啦啦 V2--Netty -- pipeline.addLast(io.netty.handler.codec.MessageToMessageCodec<INBOUND_IN, OUTBOUND_IN>) 覆写编码解码方法. pipeline相当于拦截器.在pipeline中添加MessageToMessageCodec接口的实现类,该接口的实现类中的encode()方法自动将发送的Object对象转换为ByteBuf,decode()方法自动将

Python网络编程04/recv原理/高大上版解决粘包方式

目录 Python网络编程04/recv原理/高大上版解决粘包方式 1.昨日内容回顾 2.recv工作原理 3.高大上版解决粘包方式(自定制报头) 3.1 解决思路: 3.2 服务端 3.3客户端 4.基于UDP协议的socket通信 4.1服务端 4.2客户端 Python网络编程04/recv原理/高大上版解决粘包方式 1.昨日内容回顾 1. 通信循环 2. 链接通信循环 3. 远程执行命令: subprocess.Popen() # bytes: 网络传输, 文件存储时. 4. 粘包现象

网络编程 -- RPC实现原理 -- NIO多线程 -- 迭代版本V2

网络编程 -- RPC实现原理 -- 目录 啦啦啦 V2--增加WriteQueue队列,存放selectionKey.addWriteEventToQueue()添加selectionKey并唤醒阻塞的selector.等selector唤醒之后再注册OP_WRITE事件. ( selectionKey.cancel();清除key对应事件之后,由于多线程 main线程和对应的IO线程会抢夺selector资源. 在selector.select()和sc.register(selection

网络编程 -- RPC实现原理 -- RPC -- 迭代版本V3 -- 远程方法调用 整合 Spring

网络编程 -- RPC实现原理 -- 目录 啦啦啦 V3--RPC -- 远程方法调用 及 null的传输 + Spring 服务提供商: 1. 配置 rpc03_server.xml 注入 服务提供商 rpcServiceProvider并指定初始化方法 及 服务实例 IUserService 2. 读取 服务消费者 请求的 MethodStaics ,通过反射获取服务端实例方法的返回值.返回值为null值,则映射为NullWritable实例返回.不为null,则不加以约束. 服务代理商:

RPC通信原理

什么是 RPCRPC(Remote Procedure Call Protocol)远程过程调用协议.通俗的描述是:客户端在不知道调用细节的情况下,调用存在于远程计算上的某个过程或函数,就像调用本地应用程序中的一样. 正式的描述是:一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的协议. RPC 主要特质: RPC 是协议:协议意味着规范.目前典型的 RPC 实现包括Hetty 等.但这些实现往往都会附加其他重要功能,例如 Du理.访问权限管理等功能. 网络协议和网络 IO 模型

[转帖]脑残式网络编程入门(二):我们在读写Socket时,究竟在读写什么?

脑残式网络编程入门(二):我们在读写Socket时,究竟在读写什么? http://www.52im.net/thread-1732-1-1.html 1.引言 本文接上篇<脑残式网络编程入门(一):跟着动画来学TCP三次握手和四次挥手>,继续脑残式的网络编程知识学习 ^_^. 套接字socket是大多数程序员都非常熟悉的概念,它是计算机网络编程的基础,TCP/UDP收发消息都靠它.我们熟悉的web服务器底层依赖它,我们用到的MySQL关系数据库.Redis内存数据库底层依赖它.我们用微信和别