JSON-RPC(jsonrpc4j)使用demo


服務端開發,在很多情況下,需要使用到RPC框架,今天發現一款很輕量的RPC框架——JSON-RPC。json rpc 是一種以json為消息格式的遠程調用服務,它是一套允許運行在不同操作系統、不同環境的程序實現基於Internet過程調用的規范和一系列的實現。這種遠程過程調用可以使用http作為傳輸協議,也可以使用其它傳輸協議,傳輸的內容是json消息體。
json rpc 和xmlrpc相比具有很多優點。首先xmlrpc是以xml作為消息格式,xml具有體積大,格式復雜,傳輸占用帶寬。程序對xml的解析也比較復雜,並且耗費較多服務器資源。json相比xml體積小巧,並且解析相對容易很多。
json-rpc是一種非常輕量級的跨語言遠程調用協議,實現及使用簡單。僅需幾十行代碼,即可實現一個遠程調用的客戶端,方便語言擴展客戶端的實現。服務器端有php、java、python、ruby、.net等語言實現,是非常不錯的及輕量級的遠程調用協議。
其官網地址是:http://www.jsonrpc.org/
其Wiki地址是:http://json-rpc.org/wiki/implementations
可以看到,JSON-RPC有C,C++,C#,Javascipt,Erlang,Objective-C,Java等多種語言的實現,這里我簡單介紹下Java使用JSON-RPC的示例,今天折騰了半天弄得。
首先就是要下載jsonrpc4j的jar包以及其依賴的jar包,這里我直接搭建Maven項目,使用Maven來管理jar包,其pom文件如下:

pom.xml

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<groupId>com.hjc.demo</groupId>
	<artifactId>jsonrpc_server</artifactId>
	<packaging>war</packaging>
	<version>0.0.1-SNAPSHOT</version>
	<name>jsonrpc_server Maven Webapp</name>
	<url>http://maven.apache.org</url>
	<dependencies>
		<!-- jsonrpc4j -->
		<dependency>
			<groupId>com.github.briandilley.jsonrpc4j</groupId>
			<artifactId>jsonrpc4j</artifactId>
			<version>1.0</version>
		</dependency>
		<!-- jsonrpc4j依賴的包 -->
		<dependency>
			<groupId>junit</groupId>
			<artifactId>junit</artifactId>
			<version>4.10</version>
		</dependency>
		<dependency>
			<groupId>com.fasterxml.jackson.core</groupId>
			<artifactId>jackson-core</artifactId>
			<version>2.0.2</version>
		</dependency>
		<dependency>
			<groupId>com.fasterxml.jackson.core</groupId>
			<artifactId>jackson-databind</artifactId>
			<version>2.0.2</version>
		</dependency>
		<dependency>
			<groupId>com.fasterxml.jackson.core</groupId>
			<artifactId>jackson-annotations</artifactId>
			<version>2.0.2</version>
		</dependency>
		<dependency>
			<groupId>javax.portlet</groupId>
			<artifactId>portlet-api</artifactId>
			<version>2.0</version>
		</dependency>
		<dependency>
			<groupId>org.springframework</groupId>
			<artifactId>spring-core</artifactId>
			<version>3.1.2.RELEASE</version>
		</dependency>
		<dependency>
			<groupId>org.springframework</groupId>
			<artifactId>spring-context</artifactId>
			<version>3.1.2.RELEASE</version>
		</dependency>
		<dependency>
			<groupId>org.springframework</groupId>
			<artifactId>spring-web</artifactId>
			<version>3.1.2.RELEASE</version>
		</dependency>
		<dependency>
			<groupId>org.springframework</groupId>
			<artifactId>spring-test</artifactId>
			<version>3.1.2.RELEASE</version>
		</dependency>
		<dependency>
			<groupId>commons-codec</groupId>
			<artifactId>commons-codec</artifactId>
			<version>1.4</version>
		</dependency>
		<dependency>
			<groupId>org.apache.httpcomponents</groupId>
			<artifactId>httpcore-nio</artifactId>
			<version>4.2.1</version>
		</dependency>
		<dependency>
			<groupId>org.jmock</groupId>
			<artifactId>jmock-junit4</artifactId>
			<version>2.5.1</version>
		</dependency>
		<dependency>
			<groupId>org.jmock</groupId>
			<artifactId>jmock</artifactId>
			<version>2.5.1</version>
		</dependency>
		<dependency>
			<groupId>org.eclipse.jetty</groupId>
			<artifactId>jetty-server</artifactId>
			<version>9.0.0.RC0</version>
		</dependency>
		<dependency>
			<groupId>org.eclipse.jetty</groupId>
			<artifactId>jetty-servlet</artifactId>
			<version>9.0.0.RC0</version>
		</dependency>
		<dependency>
			<groupId>org.ow2.chameleon.fuchsia.base.json-rpc</groupId>
			<artifactId>org.ow2.chameleon.fuchsia.base.json-rpc.json-rpc-bundle</artifactId>
			<version>0.0.2</version>
		</dependency>

	</dependencies>
	<build>
		<finalName>jsonrpc_server</finalName>
	</build>
</project>

待以上jar包等環境部署好了之后,就可以進行開發了,我們先要部署一個Servlet,作為RPC調用的接口。(我的環境的Tomcat6,Tomcat7的朋友可以直接使用@WebServlet來部署Servlet,具體代碼因環境而異)。

web.xml

<!DOCTYPE web-app PUBLIC
 "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
 "http://java.sun.com/dtd/web-app_2_3.dtd" >

<web-app>
  <display-name>Archetype Created Web Application</display-name>
  <servlet>
  	<servlet-name>RpcServer</servlet-name>
  	<display-name>RpcServer</display-name>
  	<description></description>
  	<servlet-class>com.hjc.demo.RpcServer</servlet-class>
  </servlet>
  <servlet-mapping>
  	<servlet-name>RpcServer</servlet-name>
  	<url-pattern>/rpc</url-pattern>
  </servlet-mapping>
</web-app>

RpcServer.java

package com.hjc.demo;

import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import com.googlecode.jsonrpc4j.JsonRpcServer;

public class RpcServer extends HttpServlet {
	private static final long serialVersionUID = 1L;
	private JsonRpcServer rpcServer = null;

	public RpcServer() {
		super();
		rpcServer = new JsonRpcServer(new DemoServiceImply(), DemoService.class);
	}

	@Override
	protected void service(HttpServletRequest request,
			HttpServletResponse response) throws ServletException, IOException {
		rpcServer.handle(request, response);
	}

}

其中,DemoService是RPC調用的接口,DemoServiceImply是DemoService接口的實現,他們的代碼分別如下:

DemoService.java

package com.hjc.demo;

public interface DemoService {

	public DemoBean getDemo(String code, String msg);

	public Integer getInt(Integer code);

	public String getString(String msg);

	public void doSomething();
}

DemoServiceImply.java

package com.hjc.demo;

public class DemoServiceImply implements DemoService {

	public DemoBean getDemo(String code, String msg) {
		DemoBean bean1 = new DemoBean();
		bean1.setCode(Integer.parseInt(code));
		bean1.setMsg(msg);
		return bean1;
	}

	public Integer getInt(Integer code) {
		return code;
	}

	public String getString(String msg) {
		return msg;
	}

	public void doSomething() {
		System.out.println("do something");
	}

}

DemoBean是一個普通的JavaBean,其代碼如下:

DemoBean.java

package com.hjc.demo;

import java.io.Serializable;

public class DemoBean implements Serializable{
	private static final long serialVersionUID = -5141784402935371524L;
	private int code;
	private String msg;

	public int getCode() {
		return code;
	}

	public void setCode(int code) {
		this.code = code;
	}

	public String getMsg() {
		return msg;
	}

	public void setMsg(String msg) {
		this.msg = msg;
	}
}

以上代碼即完成了服務端的代碼,接下來寫一個客戶端的測試類:

JsonRpcTest.java

package com.hjc.test;

import java.net.MalformedURLException;
import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import com.fasterxml.jackson.databind.node.ObjectNode;
import com.googlecode.jsonrpc4j.JsonRpcHttpClient;
import com.hjc.demo.DemoBean;

public class JsonRpcTest {
	static JsonRpcHttpClient client;

	public JsonRpcTest() {

	}

	public static void main(String[] args) throws Throwable {
		// 實例化請求地址,注意服務端web.xml中地址的配置
		try {
			client = new JsonRpcHttpClient(new URL(
					"http://127.0.0.1:8080/jsonrpc_server/rpc"));
			// 請求頭中添加的信息
			Map<String, String> headers = new HashMap<String, String>();
			headers.put("UserKey", "hjckey");
			// 添加到請求頭中去
			client.setHeaders(headers);
			JsonRpcTest test = new JsonRpcTest();
			test.doSomething();
			DemoBean demo = test.getDemo(1, "哈");
			int code = test.getInt(2);
			String msg = test.getString("哈哈哈");
			// print
			System.out.println("===========================javabean");
			System.out.println(demo.getCode());
			System.out.println(demo.getMsg());
			System.out.println("===========================Integer");
			System.out.println(code);
			System.out.println("===========================String");
			System.out.println(msg);
			System.out.println("===========================end");
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

	public void doSomething() throws Throwable {
		client.invoke("doSomething", null);
	}

	public DemoBean getDemo(int code, String msg) throws Throwable {
		String[] params = new String[] { String.valueOf(code), msg };
		DemoBean demo = null;
		demo = client.invoke("getDemo", params, DemoBean.class);
		return demo;
	}

	public int getInt(int code) throws Throwable {
		Integer[] codes = new Integer[] { code };
		return client.invoke("getInt", codes, Integer.class);
	}

	public String getString(String msg) throws Throwable {
		String[] msgs = new String[] { msg };
		return client.invoke("getString", msgs, String.class);
	}

}

最后的打印結果如下:

服務端:

do something

客戶端:

===========================javabean
1
哈
===========================Integer
2
===========================String
哈哈哈
===========================end

以上就完成了JSON-RPC全部的代碼,值得說的是,這其中我也爬了很多坑,這里也提一下,JsonRpcHttpClient類的invoke方法,如果你要傳遞參數,必須是一個數組,這里我傳的String[]和Integer[],如果傳遞了別的類型會發生什么呢?你可以試一下這樣調用:

client.invoke("getInt", 3, Integer.class);

然后你就會發現服務端報錯:

java.lang.IllegalArgumentException: Unknown params node type: 2
	at com.googlecode.jsonrpc4j.JsonRpcServer.findBestMethodByParamsNode(JsonRpcServer.java:612)
	at com.googlecode.jsonrpc4j.JsonRpcServer.handleObject(JsonRpcServer.java:373)
	at com.googlecode.jsonrpc4j.JsonRpcServer.handleNode(JsonRpcServer.java:293)
	at com.googlecode.jsonrpc4j.JsonRpcServer.handle(JsonRpcServer.java:230)
	at com.googlecode.jsonrpc4j.JsonRpcServer.handle(JsonRpcServer.java:207)
	at com.hjc.demo.RpcServer.service(RpcServer.java:24)
        ...

點進源碼,你就全明白了,進入JsonRpcServer類查看findBestMethodByParamsNode方法,其代碼如下:

/**
	 * Finds the {@link Method} from the supplied {@link Set} that
	 * best matches the rest of the arguments supplied and returns
	 * it as a {@link MethodAndArgs} class.
	 *
	 * @param methods the {@link Method}s
	 * @param paramsNode the {@link JsonNode} passed as the parameters
	 * @return the {@link MethodAndArgs}
	 */
	private MethodAndArgs findBestMethodByParamsNode(Set<Method> methods, JsonNode paramsNode) {

		// no parameters
		if (paramsNode==null || paramsNode.isNull()) {
			return findBestMethodUsingParamIndexes(methods, 0, null);

		// array parameters
		} else if (paramsNode.isArray()) {
			return findBestMethodUsingParamIndexes(methods, paramsNode.size(), ArrayNode.class.cast(paramsNode));

		// named parameters
		} else if (paramsNode.isObject()) {
			Set<String> fieldNames = new HashSet<String>();
			Iterator<String> itr=paramsNode.fieldNames();
			while (itr.hasNext()) {
				fieldNames.add(itr.next());
			}
			return findBestMethodUsingParamNames(methods, fieldNames, ObjectNode.class.cast(paramsNode));

		}

		// unknown params node type
		throw new IllegalArgumentException("Unknown params node type: "+paramsNode.toString());
	}

所以我就明白了,除了null(不傳參數),Array,Object,其他的類都是“Unknown params node type”,當時我也很疑惑,Integer不也是Object嘛,斷點一下,原來它認為這是IntNode類,所以不能識別,其他的參數我也試過一些,目前還是數組是最靠譜的,網上資料還比較少,希望有懂得給指點指點。
這也是我今天花了半天時間弄得JSON-RPC,喜歡的朋友,源碼地址在下面:
https://github.com/hjcenry/json-rpc-demo


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM