Avro RPC实例

RPC(Remote Procedure Call):远程过程调用,它是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的思想。

一个 RPC 的核心功能主要有 5 个部分组成,分别是:客户端、客户端 Stub、网络传输模块、服务端 Stub、服务端等。

Avro除了数据序列化功能外也提供了RPC功能。具体使用如下:

1. 在pom文件中添加相应依赖

        <dependency>
            <groupId>org.apache.avro</groupId>
            <artifactId>avro-ipc</artifactId>
            <version>1.9.1</version>
        </dependency>
        <dependency>
            <groupId>org.apache.avro</groupId>
            <artifactId>avro-ipc-netty</artifactId>
            <version>1.9.1</version>
        </dependency>

 修改插件配置

<plugin>
                <groupId>org.apache.avro</groupId>
                <artifactId>avro-maven-plugin</artifactId>
                <version>1.9.1</version>
                <executions>
                    <execution>
                        <phase>generate-sources</phase>
                        <goals>
                            <goal>schema</goal>
                            <goal>idl-protocol</goal>  //添加这一行
                        </goals>
                        <configuration>
                            <sourceDirectory>${project.basedir}/src/main/resources/</sourceDirectory>
                            <outputDirectory>${project.basedir}/src/main/java/</outputDirectory>
                        </configuration>
                    </execution>
                </executions>
            </plugin>

 

2. 编写服务API契约(avdl文件)

一个服务API契约在Avro中称之为 Avro Protocol ,Avro 提供了一种IDL语言来简化这个Avro Protocol的编写,如下:

@namespace("com.zpark.demo.avro.rpc")
protocol PersonService{
import schema "person.avsc";  //此处导入schema
  boolean addPerson(com.zpark.Person person); //定义一个远程服务
}

在实际运行时,上述的avdl文件会转成avpr的文件形式,我们可以通过 avroj-tools.jar来进行查看并输出avpr文件,命令如下:

java -jar avro-tools-1.9.1.jar idl person.avdl person.avpr

生成的Avro protocol模式文件内容如下:
{
  "protocol" : "PersonService",
  "namespace" : "com.zpark.demo.avro.rpc",
  "types" : [ {
    "type" : "record",
    "name" : "Person",
    "namespace" : "com.zpark",
    "fields" : [ {
      "name" : "id",
      "type" : "string"
    }, {
      "name" : "name",
      "type" : "string"
    }, {
      "name" : "age",
      "type" : [ "int", "null" ]
    } ]
  } ],
  "messages" : {
    "addPerson" : {
      "request" : [ {
        "name" : "person",
        "type" : "com.zpark.Person"
      } ],
      "response" : "boolean"
    }
  }
}


3.生成服务接口类

  使用 mvn clean install -DskipTests=true 对项目进行打包,就会触发maven插件根据编写的avdl文件生成服务契约接口。

package com.zpark.demo.avro.rpc;

@org.apache.avro.specific.AvroGenerated
public interface PersonService {
  public static final org.apache.avro.Protocol PROTOCOL = org.apache.avro.Protocol.parse("{\"protocol\":\"PersonService\",\"namespace\":\"com.zpark.demo.avro.rpc\",\"types\":[{\"type\":\"record\",\"name\":\"Person\",\"namespace\":\"com.zpark\",\"fields\":[{\"name\":\"id\",\"type\":\"string\"},{\"name\":\"name\",\"type\":\"string\"},{\"name\":\"age\",\"type\":[\"int\",\"null\"]}]}],\"messages\":{\"addPerson\":{\"request\":[{\"name\":\"person\",\"type\":\"com.zpark.Person\"}],\"response\":\"boolean\"}}}");
  /**
   */
  boolean addPerson(com.zpark.Person person);

  @SuppressWarnings("all")
  public interface Callback extends PersonService {
    public static final org.apache.avro.Protocol PROTOCOL = com.zpark.demo.avro.rpc.PersonService.PROTOCOL;
    /**
     * @throws java.io.IOException The async call could not be completed.
     */
    void addPerson(com.zpark.Person person, org.apache.avro.ipc.Callback<java.lang.Boolean> callback) throws java.io.IOException;
  }
}

 

4.实现服务接口

public class PersonServiceImpl implements PersonService {
    @Override
    public boolean addPerson(Person person) {
        System.out.println(person);
        return true ;
    }
}

 

5.编写Avro RPC Server端相关代码

public class AvroRpcServer {
    public static void main(String[] args) {
        NettyServer ns = new NettyServer(
                new SpecificResponder(PersonService.class,new PersonServiceImpl()),
                new InetSocketAddress(9998)
        );
        ns.start();
        System.out.println("服务端已启动");
    }
}

 

6.编写Avro RPC Client端相关代码

public class AvroClient {

    public static void main(String[] args) throws Exception{
        NettyTransceiver client = new NettyTransceiver(new InetSocketAddress("127.0.0.1",9998));
        PersonService proxy = SpecificRequestor.getClient(PersonService.class, client);
        Person person = new Person("001","zhangsan",23);
        boolean result = proxy.addPerson(person);
        if(result){
            System.out.println("服务调用成功");
        }else {
            System.out.println("服务调用失败");
        }

    }


}

 

7.测试

posted @ 2020-02-06 23:29  杭州胡欣  阅读(673)  评论(0编辑  收藏  举报