akka-rpc

   最近一直在准备构建一个rpc服务,为避免重复造轮子,在择取的过程中发现了很多优秀的rpc架构。

1:common-rpc:http://my.oschina.net/liubingsmile/blog/387154

已经足够成熟。

2:基于akka的rpc

http://my.oschina.net/FengJ/blog/268387

现在仅仅对其初始化的时候 做小小的改动,方便使用。

原有代码对服务的注册:

// Server 加入发布的服务 
Map < Class <?>, Object > beans = new HashMap < Class <?>, Object >(); 
beans . put ( ExampleInterface . class , new ExampleInterfaceImpl ()); 
这样服务多了,很不方便。

   现在选择基于spring的其中一种方式,略微改动,主要是基于Java的@interface注解,加上spring的bean工厂。原理比较简单。

 首先创建一个spring的配置文件:

rpcServerContext.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"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context.xsd">
       <!--service-->
<context:component-scan base-package="com.topteam.example.service.impl"/>
<bean id="rpcServer" class="com.topteam.rpc.anotaion.RpcServer"></bean>
</beans>

RpcServer :扫描注解接口,进行akka-rpc服务映射的初始化

public class RpcServer implements ApplicationContextAware, InitializingBean 
{
    /**
     * 存放 服务名 与 服务对象 之间的映射关系
     */
    private Map<Class<?>, Object> handlerMap = new HashMap<Class<?>, Object>();
    
public void setApplicationContext(ApplicationContext ctx) throws BeansException
    {
        // 扫描带有 RpcService 注解的类并初始化 handlerMap 对象
        Map<String, Object> serviceBeanMap = ctx.getBeansWithAnnotation(RpcService.class);
        if (MapUtils.isNotEmpty(serviceBeanMap))
        {
            for (Object serviceBean : serviceBeanMap.values()) 
            {
                RpcService rpcService = serviceBean.getClass().getAnnotation(RpcService.class);
                handlerMap.put(rpcService.value(),serviceBean);
            }
            this.setHandlerMap(handlerMap);
        }
    }


public void afterPropertiesSet() throws Exception 
{
// TODO Auto-generated method stub
}

public Map<Class<?>, Object> getHandlerMap() 
    {
return handlerMap;
}


public void setHandlerMap(Map<Class<?>, Object> handlerMap)
{
this.handlerMap = handlerMap;
}
}

RpcBootstrap:用来扫描

public class RpcBootstrap 
{
private ClassPathXmlApplicationContext application;
public void init()
{
System.out.println("*************__init__************");
ClassPathXmlApplicationContext application = new ClassPathXmlApplicationContext("rpcServerContext.xml");
this.application = application;
}

public void close() 
{
System.out.println("**************__close__***********");
}


public ClassPathXmlApplicationContext getApplication() 
{
return application;
}


public void setApplication(ClassPathXmlApplicationContext application) 
{
this.application = application;
}
}


@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Component
public @interface RpcService 
{
    /**
     * 服务接口类
     */
    Class<?> value();
}


扫描注解的接口:

public class RpcServer implements ApplicationContextAware, InitializingBean 
{
    /**
     * 存放 服务名 与 服务对象 之间的映射关系
     */
    private Map<Class<?>, Object> handlerMap = new HashMap<Class<?>, Object>();
    
public void setApplicationContext(ApplicationContext ctx) throws BeansException
    {
        // 扫描带有 RpcService 注解的类并初始化 handlerMap 对象
        Map<String, Object> serviceBeanMap = ctx.getBeansWithAnnotation(RpcService.class);
        if (MapUtils.isNotEmpty(serviceBeanMap))
        {
            for (Object serviceBean : serviceBeanMap.values()) 
            {
                RpcService rpcService = serviceBean.getClass().getAnnotation(RpcService.class);
                handlerMap.put(rpcService.value(),serviceBean);
            }
            this.setHandlerMap(handlerMap);
        }
    }


public void afterPropertiesSet() throws Exception 
{
// TODO Auto-generated method stub
}

public Map<Class<?>, Object> getHandlerMap() 
    {
return handlerMap;
}


public void setHandlerMap(Map<Class<?>, Object> handlerMap)
{
this.handlerMap = handlerMap;
}
}

接口实现:

@RpcService(ExampleInterface.class)
public class ExampleInterfaceImpl implements ExampleInterface 
{
/**

*/
private static final long serialVersionUID = 1L;


public String sayHello(String name)
{
System.out.println("Be Called !");
ExampleOtherBean bean = new ExampleOtherBean(name);
return "Hello " + bean.toString();
}
}


改造原有的spring主上下文:

spring-context.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"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context.xsd">


    <context:component-scan base-package="com.topteam.rpc.anotaion"/>
    <bean id="bootstrap" class="com.topteam.rpc.bootstrap.RpcBootstrap" init-method="init" destroy-method="close"></bean>
    
    <bean id="akkaRpcServer" class="com.topteam.rpc.server.AkkaRpcServer" init-method="start" destroy-method="close"></bean>
<bean id="akkaRpcClient" class="com.topteam.rpc.client.AkkaRpcClientFactory" scope="singleton"></bean>
</beans>



测试:

public class StringSupportTest 
{
public static void main(String[] args)
{
ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-context.xml");

AkkaRpcServer akkaRpcServer = (AkkaRpcServer)applicationContext.getBean("akkaRpcServer");

final Config config = ConfigFactory
.parseString("akka.remote.netty.tcp.port=" + 2551)
.withFallback(ConfigFactory.parseString("akka.cluster.roles = [RpcServer]"))
.withFallback(ConfigFactory.load());
ActorSystem system = ActorSystem.create("EsbSystem", config);

// Server 加入发布的服务
//Map<Class<?>, Object> beans = new HashMap<Class<?>, Object>();
        //beans.put(ExampleInterface.class, new ExampleInterfaceImpl());

/**
* 服务创建actors
*/
System.out.println(akkaRpcServer.getServices());
system.actorOf(Props.create(RpcServerActor.class, akkaRpcServer.getServices()), "rpcServer");


// int count = 10000;
//
// AkkaRpcClient client = AkkaRpcClient.getInstance();
// long start = System.currentTimeMillis();
//
// ExampleInterface example = client.getBean(ExampleInterface.class);
//
// for(int i = 0; i < count; i++)
// {
// System.out.println(example.sayHello("rpc_" + i));
// }
//
// long time = System.currentTimeMillis() - start;
// System.out.println("time :" + time);


}

/*
* private ApplicationContext applicationContext =new ClassPathXmlApplicationContext("classpath:spring-context.xml");
* @Test
public void akkaRpcServerSpringInjectTest()
{
AkkaRpcServer akkaRpcServer = applicationContext.getBean(AkkaRpcServer.class);
Assert.assertNotNull(akkaRpcServer);
}


@Test
public void akkaRpcClientFactoryInjectTest()
{
AkkaRpcClientFactory akkaRpcServer = applicationContext.getBean(AkkaRpcClientFactory.class);
Assert.assertNotNull(akkaRpcServer.getClient());
}*/
}




你可能感兴趣的:(akka-rpc)