服务端开发,在很多情况下,需要使用到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
相关推荐
标题"jsonrpc-frontend:前端应用程序发送 json-rpc 请求进行测试"表明我们要讨论的是一个专门针对前端应用的库或工具,它帮助开发者方便地发送JSON-RPC请求进行功能测试。这个工具可能包括对JSON-RPC规范的实现、...
JSON-RPC(JavaScript Object Notation Remote Procedure Call)是一种轻量级的远程过程调用协议,常用于客户端-服务器通信。在Android开发中,有时我们需要与远程服务器进行数据交互,这时JSON-RPC客户端库就显得尤...
- **基本结构**:请求对象必须包含 "jsonrpc" 字段,其值为 "2.0",指示使用的是 JSON-RPC 2.0 协议。 - **方法调用**:"method" 字段用于指定要调用的方法名。方法名采用字符串形式,其中以 "rpc." 开头的方法名...
JSON-RPC-Java是一个用Java来实现动态JSON-RPC的框架. 利用它内置的一个轻级量JSON-RPC JavaScripIt客户端,可以让你透明地在JavaScript中调用Java代码。JSON-RPC-Java可运行在Servlet容器中如Tomcat也可以运行在...
本压缩包“json-rpc-java-1.0.zip”包含了JSON-RPC在Java平台上的实现及相关使用示例。 首先,JSON-RPC的基本原理是,客户端通过HTTP请求将一个JSON格式的请求体发送到服务器,请求体包含了要调用的方法名和参数。...
7. **框架支持**: 在Java中,有多个库可以帮助开发者实现JSON-RPC,例如JSON-RPC 2.0规范下的net.sf.jsonrpc4j和jsonrpc-server等。这些库提供了一套完整的解决方案,包括服务器端的处理、客户端的调用以及错误处理...
在提供的压缩包“json-rpc-1.0(with source).rar”中,包含了JSON-RPC 1.0的实现以及相应的源代码。 **JSON-RPC 1.0协议详解** JSON-RPC 1.0定义了客户端和服务器之间通信的基本结构,主要包含以下几个部分: 1. *...
### JSON-RPC-Java在构建下一代Web应用中的应用 #### 概述 在2005年的演讲中,Michael Clark探讨了如何利用JSON-RPC-Java来构建高度动态的Web应用程序。这一技术允许开发人员通过JavaScript透明地调用远程Java方法...
JSON-RPC(JavaScript Object Notation Remote Procedure Call)是一种轻量级的远程过程调用协议,常用于客户端-服务器通信,特别是在分布式系统中。它利用JSON(JavaScript Object Notation)作为数据交换格式,...
JSON-RPC-Java是一个用Java来实现动态JSON-RPC的框架. 利用它内置的一个轻级量JSON-RPC JavaScripIt客户端,可以让你透明地在JavaScript中调用Java代码。JSON-RPC-Java可运行在Servlet容器中如Tomcat也可以运行在...
这个"json-rpc-1.0.zip"压缩包包含了一个Java实现的JSON-RPC 1.0版本的库,主要用于在Java应用程序中处理JSON数据并实现远程调用功能。 在Java中,JSON-RPC库允许开发者通过HTTP或WebSocket等协议,将JSON格式的...
4. **`jsonrpc`**:版本号,对于JSON-RPC 2.0,值应为`"2.0"`。 ### JSON-RPC工作流程 1. **客户端发起请求**:客户端构造一个包含`method`、`params`和`id`的JSON对象,通过HTTP请求发送到服务器。 2. **服务器...
本项目"json-rpc-java-1.0"提供了JSON-RPC的Java实现,包含`jsonArray`和`jsonObject`等关键组件。 1. JSON基础知识: JSON是一种轻量级的数据交换格式,易于人阅读和编写,同时也易于机器解析和生成。它的基本...
继《JavaScript高级应用与实践》之后推出的json-rpc-for-java开源代码,是仅仅100行的javascript代码和不到10个java文件实现的超级轻量级的通过 javaScript快速调用java对象并返回任意对象的轻量级框架,并且支持...
在这个项目中,我们可能使用的JSON-RPC框架有如JsonRPC4j、gson-rpc等。这些框架提供了方便的API来简化服务端接口定义和服务注册,以及客户端的调用过程。例如,服务端通常需要创建一个实现了特定接口的类,然后通过...
json-rpc-for-java,是仅仅不到100行的javascript代码和不到10个java文件实现的超级轻量级的通过 javaScript快速调用java对象并返回任意对象的轻量级框架,并且支持级联调用,也就是说不需要额外 的JavaScript编程,...
在Python中,json-rpc-python库提供了一种方便的方式来实现JSON-RPC服务器和客户端,允许程序通过网络进行通信,执行远程方法就像它们是在本地一样。 在Python中,json-rpc-python库主要包含以下几个核心概念: 1....
基于JSON-RPC 1.0的C语言实现,如"jsonrpc-c-master",提供了一种在C语言环境中构建服务端应用的方法,尤其适用于嵌入式系统或对性能有严格要求的环境。 **JSON-RPC 1.0协议基础** JSON-RPC 1.0是一个简单的请求/...
2. **RPC库**:Python有一些现成的库,如`SimpleJSON-RPC`或`jsonrpc-server`,它们提供了一套简单的API来定义服务端的RPC方法和客户端的调用。 3. **Web框架集成**:对于更复杂的应用,可以将JSON-RPC与Web框架如...