码迷,mamicode.com
首页 > Web开发 > 详细

JSON-RPC轻量级远程调用协议介绍及使用

时间:2014-11-10 11:28:12      阅读:330      评论:0      收藏:0      [点我收藏+]

标签:style   blog   http   io   color   ar   os   使用   java   

这个项目能够帮助开发人员利用Java编程语言轻松实现JSON-RPC远程调用。jsonrpc4j使用Jackson类库实现Java对象与JSON对象之间的相互转换。jsonrpc4j包含一个JSON-RPC服务器,支持Stream与HTTP(GET与POST),同时还提供一个支持Stream的JSON-RPC客户端。此外还提供一个HTTP客户端、Spring Service Provider和Spring Service Consumer。

 

https://github.com/briandilley/jsonrpc4j (包下载)

-----------------------------------------------

JSON-RPC轻量级远程调用协议介绍及使用

目录

技术简介    1

一、JSON-RPC协议描述    1

二、JSON-RPC调用简单示例    1

2.1、服务器端Java调用示例    1

2.2、Java客户端调用示例    2

2.3、PHP客户端调用示例    2

2.3、JavaScript客户端调用示例    2

2.4、直接GET请求进行调用    2

三、JSON-RPC总结    3

参考文档    3

 

 

技术简介

json-rpc是基于json的跨语言远程调用协议,比xml-rpc、webservice等基于文本的协议传输数据格小;相对hessian、java-rpc等二进制协议便于调试、实现、扩展,是非常优秀的一种远程调用协议。目前主流语言都已有json-rpc的实现框架,java语言中较好的json-rpc实现框架有jsonrpc4j、jpoxy、json-rpc。三者之中jsonrpc4j既可独立使用,又可与spring无缝集合,比较适合于基于spring的项目开发。

 

一、JSON-RPC协议描述

json-rpc协议非常简单,发起远程调用时向服务端传输数据格式如下:

   { "method": "sayHello", "params": ["Hello JSON-RPC"], "id": 1}

参数说明:

method: 调用的方法名

params: 方法传入的参数,若无参数则传入 []

id : 调用标识符,用于标示一次远程调用过程

服务器其收到调用请求,处理方法调用,将方法效用结果效应给调用方;返回数据格式:

 

 {   
    "result":          "Hello JSON-RPC",         
    "error":                null,       
      "id":                      1
 }                        

参数说明:

result: 方法返回值,若无返回值,则返回null。若调用错误,返回null。

error :调用时错误,无错误返回null。

id : 调用标识符,与调用方传入的标识符一致。

以上就是json-rpc协议规范,非常简单,小巧,便于各种语言实现。

 

二、JSON-RPC简单示例

 

2.1、服务器端Java调用示例

jsonrpc4j服务器端java示例:

public class HelloWorldServlet extends HttpServlet {

    private static final long serialVersionUID = 3638336826344504848L;

    private JsonRpcServer rpcService = null;

    @Override

    public void init(ServletConfig config) throws ServletException {

        super.init(config);

        rpcService = new JsonRpcServer(new HelloWorldService(), HelloWorldService.class);

    }

    @Override

    protected void service(HttpServletRequest req, HttpServletResponse resp)

            throws ServletException, IOException {

        rpcService.handle(req, resp);    

    }

}

 

2.2、Java客户端调用示例

jsonrpc4j的Java客户端调用示例:

        JsonRpcHttpClient client = new JsonRpcHttpClient(

             new URL("http://127.0.0.1:8080/index.json"));

        Map<String,String> headers = new HashMap<String,String>();

        headers.put("name", "剑白");

     client.setHeaders(headers);

        String properties = client.invoke("getSystemProperties", null, String.class);

        System.out.println(properties);

 

2.3、PHP客户端调用示例

基于json-rpc-php的PHP客户端调用示例:

<?php include(dirname(__FILE__)."/lib/client/JsonRpcClient.php");

$client = new JsonRpcClient("http://10.13.49.234:8080/index.json");

$response = $client->getSystemProperties();

echo $response->result;

?>

 

2.3、JavaScript客户端调用示例

基于jsonrpcjs的JavaScript客户端调用示例:

var rpc = new jsonrpc.JsonRpc(‘http://127.0.0.1:8080/index.json‘);

rpc.call(‘getSystemProperties‘, function(result){

alert(result);

});

 

 

2.4、直接GET请求进行调用

无需任何客户端,只需手工拼接参数进行远程调用,请求URL如下:

http://127.0.0.1:8080/index.json?method=getSystemProperties&id=3325235235235&params=JTViJTVk

参数说明:

method : 方法名

params :调用参数,json的数组格式[], 将参数需先进行url编码,再进行base64编码

id : 调用标识符,任意值。

 

三、JSON-RPC总结

json-rpc是一种非常轻量级的跨语言远程调用协议,实现及使用简单。仅需几十行代码,即可实现一个远程调用的客户端,方便语言扩展客户端的实现。服务器端有php、java、python、ruby、.net等语言实现,是非常不错的及轻量级的远程调用协议。

 

参考文档

http://code.google.com/p/jsonrpc4j/

http://json-rpc.org/wiki/implementations

http://en.wikipedia.org/wiki/JSON-RPC

https://github.com/gimmi/jsonrpcjs

http://bitbucket.org/jbg/php-json-rpc

https://github.com/Pozo/json-rpc-php

https://github.com/subutux/json-rpc2php

 

 

-------------------------------------------------------------

一、JSON-RPC-Java简介
        JSON-RPC-Java是一个用Java来实现动态JSON-RPC的框架. 利用它内置的一个轻级量JSON-RPC JavaScripIt客户端,可以让你透明地在JavaScript中调用Java代码。JSON-RPC-Java可运行在Servlet容器中如Tomcat也可以运行在JBoss与其它J2EE应用服务器中因此可以在一个基于JavaScript与DHTML的Web应用程序中利用它来直接调用普通Java方法与EJB方法。我们可以很方便的使用JSON-RPC-Java来开发我的们Ajax应用。
二、JSON-RPC-Java实践
1、从http://oss.metaparadigm.com/jsonrpc/download.html下载最新稳定版的JSON-RPC-Java框架并解压。新建一个JavaWeb工程,将jsonrpc-1.0.jar放到/webroot/WEB-INF/lib目录下,将jsonrpc.js放在/webroot下的任意目录下,保证在jsp页面可以引用即可。
 2、修改/webroot/WEB-INF/web.xml文件,添加以下代码:

 1bubuko.com,布布扣<servlet>
 2bubuko.com,布布扣        <servlet-name>JSONRPCServlet</servlet-name>
 3bubuko.com,布布扣        <servlet-class>
 4bubuko.com,布布扣            com.metaparadigm.jsonrpc.JSONRPCServlet
 5bubuko.com,布布扣        </servlet-class>
 6bubuko.com,布布扣    </servlet>
 7bubuko.com,布布扣    <servlet-mapping>
 8bubuko.com,布布扣        <servlet-name>JSONRPCServlet</servlet-name>
 9bubuko.com,布布扣        <url-pattern>/JSON-RPC</url-pattern>
10bubuko.com,布布扣    </servlet-mapping>

当然这里代码不一定要和上面一模一样,有一点基础即可看出这里只是添加了一个Servlet配置而,只要符合Servlet的配置规则即可。 
3、编Java文件。
        Example.java 此文件提供各种业务操作,此方件编写没有什么特别的要求。但为了在JSP页面用标签,我们最好提供一个无参的构造方法。

bubuko.com,布布扣
 1bubuko.com,布布扣package net.vicp.jiasoft;
 2bubuko.com,布布扣
 3bubuko.com,布布扣import java.io.Serializable;
 4bubuko.com,布布扣import java.util.List;
 5bubuko.com,布布扣import java.util.Map;
 6bubuko.com,布布扣import java.util.Set;
 7bubuko.com,布布扣
 8bubuko.com,布布扣public class Example implements Serializable {
 9bubuko.com,布布扣    private final static long serialVersionUID = 1L;
10bubuko.com,布布扣    
11bubuko.com,布布扣    public String sayString(String name) {
12bubuko.com,布布扣        return "Hello " + name + " !";
13bubuko.com,布布扣    }
14bubuko.com,布布扣
15bubuko.com,布布扣    public List sayList(List list) {
16bubuko.com,布布扣        list.add(new Integer(6));
17bubuko.com,布布扣        return list;
18bubuko.com,布布扣    }
19bubuko.com,布布扣    
20bubuko.com,布布扣    public Map sayMap (Map map) {
21bubuko.com,布布扣        map.put("age","23");
22bubuko.com,布布扣        return map;
23bubuko.com,布布扣    }
24bubuko.com,布布扣    
25bubuko.com,布布扣    public Set saySet (Set set) {
26bubuko.com,布布扣        set.add("sex");
27bubuko.com,布布扣        return set;
28bubuko.com,布布扣    }
29bubuko.com,布布扣    
30bubuko.com,布布扣    public User sayUser (User user) {
31bubuko.com,布布扣        user.setAge(25);
32bubuko.com,布布扣        return user;
33bubuko.com,布布扣    }
34bubuko.com,布布扣}


        User.java 此文件用于自定义数据类型传输数。要求是一个javabean,即要为每一个属性提供set和get方法,还有一个无参的构方法即可。

bubuko.com,布布扣
 1bubuko.com,布布扣package net.vicp.jiasoft;
 2bubuko.com,布布扣
 3bubuko.com,布布扣import java.io.Serializable;
 4bubuko.com,布布扣
 5bubuko.com,布布扣public class User implements Serializable {
 6bubuko.com,布布扣    private String name;
 7bubuko.com,布布扣    private int age;
 8bubuko.com,布布扣    public int getAge() {
 9bubuko.com,布布扣        return age;
10bubuko.com,布布扣    }
11bubuko.com,布布扣    public void setAge(int age) {
12bubuko.com,布布扣        this.age = age;
13bubuko.com,布布扣    }
14bubuko.com,布布扣    public String getName() {
15bubuko.com,布布扣        return name;
16bubuko.com,布布扣    }
17bubuko.com,布布扣    public void setName(String name) {
18bubuko.com,布布扣        this.name = name;
19bubuko.com,布布扣    }
20bubuko.com,布布扣}


4、编写javascript文件。
        Example.js 用于此处理客户端操作。

bubuko.com,布布扣
 1bubuko.com,布布扣//The javascript file of the JSON-RPC-Java example application.
 2bubuko.com,布布扣var jsonrpc = null;
 3bubuko.com,布布扣//初始化JSONRpcClient对象
 4bubuko.com,布布扣function onLoad() {
 5bubuko.com,布布扣    jsonrpc = new JSONRpcClient("JSON-RPC");
 6bubuko.com,布布扣}
 7bubuko.com,布布扣//提交普通的字符串对象
 8bubuko.com,布布扣function sayString() {
 9bubuko.com,布布扣    var who = document.getElementById("who");
10bubuko.com,布布扣    var result = jsonrpc.hello.sayString(cbString, who.value);
11bubuko.com,布布扣}
12bubuko.com,布布扣function cbString(result, exception) {
13bubuko.com,布布扣    if (exception == null) {
14bubuko.com,布布扣        document.getElementById("say").innerHTML = "<em>操作提示:提交一个字符串并返回.</em><h2>" + result + "</h2><hr/>";
15bubuko.com,布布扣    } else {
16bubuko.com,布布扣        alert(exception.message);
17bubuko.com,布布扣    }
18bubuko.com,布布扣}
19bubuko.com,布布扣
20bubuko.com,布布扣//提交一个list对象并且将其返回
21bubuko.com,布布扣function sayList() {
22bubuko.com,布布扣    var list = {"javaClass":"java.util.ArrayList", "list":[1, 2, 3]};
23bubuko.com,布布扣    jsonrpc.hello.sayList(cbList, list);
24bubuko.com,布布扣}
25bubuko.com,布布扣function cbList(result, exception) {
26bubuko.com,布布扣    if (exception == null) {
27bubuko.com,布布扣        var list = result.list;
28bubuko.com,布布扣        var str = "<em>操作提示:提交一个List并返回.</em>";
29bubuko.com,布布扣        for (var value in list) {
30bubuko.com,布布扣            str += "<h2>list[" + value + "]=" + list[value] + "</h2><hr/>";
31bubuko.com,布布扣        }
32bubuko.com,布布扣        document.getElementById("say").innerHTML = str;
33bubuko.com,布布扣    //"list"可以直接用下面的语句输出
34bubuko.com,布布扣    //document.getElementById("say").innerHTML = "<h2>" + list + "</h2>";
35bubuko.com,布布扣    } else {
36bubuko.com,布布扣        alert(exception.message);
37bubuko.com,布布扣    }
38bubuko.com,布布扣}
39bubuko.com,布布扣
40bubuko.com,布布扣//提交一个map对象并且将其返回
41bubuko.com,布布扣function sayMap() {
42bubuko.com,布布扣    var map = {"javaClass":"java.util.HashMap", "map":{"name":"Huaxu", "sex":"\u7537"}};
43bubuko.com,布布扣    jsonrpc.hello.sayMap(cbMap, map);
44bubuko.com,布布扣}
45bubuko.com,布布扣function cbMap(result, exception) {
46bubuko.com,布布扣    if (exception == null) {
47bubuko.com,布布扣        var map = result.map;
48bubuko.com,布布扣        var str = "<em>操作提示:提交一个Map并返回.</em>";
49bubuko.com,布布扣        for (var key in map) {
50bubuko.com,布布扣            str += "<h2>" + key + ":" + map[key] + "</h2><hr/>";
51bubuko.com,布布扣        }
52bubuko.com,布布扣        document.getElementById("say").innerHTML = str;
53bubuko.com,布布扣    } else {
54bubuko.com,布布扣        alert(exception.message);
55bubuko.com,布布扣    }
56bubuko.com,布布扣}
57bubuko.com,布布扣
58bubuko.com,布布扣//提交一个set对象并且将其返回
59bubuko.com,布布扣function saySet() {
60bubuko.com,布布扣    //set属性是一个数组对象,每个数组元素就是set里的一个元素.
61bubuko.com,布布扣    var set = {"javaClass":"java.util.HashSet", "set":{"name":"name"}};
62bubuko.com,布布扣    jsonrpc.hello.saySet(cbSet, set);
63bubuko.com,布布扣}
64bubuko.com,布布扣function cbSet(result, exception) {
65bubuko.com,布布扣    if (exception == null) {
66bubuko.com,布布扣        var set = result.set;
67bubuko.com,布布扣        var str = "<em>操作提示:提交一个Set并返回.</em>";
68bubuko.com,布布扣        for (var value in set) {
69bubuko.com,布布扣            str += "<h2>" + value + "</h2><hr/>";
70bubuko.com,布布扣        }
71bubuko.com,布布扣        document.getElementById("say").innerHTML = str ;
72bubuko.com,布布扣    } else {
73bubuko.com,布布扣        alert(exception.message);
74bubuko.com,布布扣    }
75bubuko.com,布布扣}
76bubuko.com,布布扣
77bubuko.com,布布扣//提交一个自定义的User对象并且将其返回
78bubuko.com,布布扣function sayUser() {
79bubuko.com,布布扣    //每个JSON对象属性对应一个User对象属性.
80bubuko.com,布布扣    var user = {"javaClass":"net.vicp.jiasoft.User", "name":"Huaxu", "age":23};
81bubuko.com,布布扣    jsonrpc.hello.sayUser(cbUser, user);
82bubuko.com,布布扣}
83bubuko.com,布布扣function cbUser(user, exception) {
84bubuko.com,布布扣    if (exception == null) {
85bubuko.com,布布扣        var str = "<em>操作提示:提交一个自定义的User对象并返回.</em><h2>name:" + user.name + "</h2><hr/>" + "<h2>age:" + user.age + "</h2><hr/>";
86bubuko.com,布布扣        document.getElementById("say").innerHTML = str ;
87bubuko.com,布布扣    } else {
88bubuko.com,布布扣        alert(exception.message);
89bubuko.com,布布扣    }
90bubuko.com,布布扣}
91bubuko.com,布布扣


此文件中的jsonrpc.hello.say*()方法即是在Example.java中定义的。
4、编写jsp页面。
        json.jsp 这个文件做什么不用说了吧,当然给用户一个操作的介面了。

bubuko.com,布布扣
 1bubuko.com,布布扣<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
 2bubuko.com,布布扣 "http://www.w3.org/TR/html4/loose.dtd">
 3bubuko.com,布布扣<%@ page contentType="text/html; charset=UTF-8"%>
 4bubuko.com,布布扣<!-- 此处在session作用域中放入一个名为"JSONRPCBridge"的"com.metaparadigm.jsonrpc.JSONRPCBridge"对象,要注意的是名字一定要为"JSONRPCBridge",否可能出错! -->
 5bubuko.com,布布扣<jsp:useBean id="JSONRPCBridge" scope="session"
 6bubuko.com,布布扣    class="com.metaparadigm.jsonrpc.JSONRPCBridge" />
 7bubuko.com,布布扣<jsp:useBean id="hello" scope="session" class="net.vicp.jiasoft.Example" />
 8bubuko.com,布布扣<%
 9bubuko.com,布布扣//此处将上面实例化的"net.vicp.jiasoft.Example"对象注册到"JSONRPCBridge"属性中,这样做即把服器端的方法暴露给客户端了,所以在Example.js中可以用jsonrpc.hello.say*()的方式调用"Example.java"类的方法。
10bubuko.com,布布扣JSONRPCBridge.registerObject("hello", hello);
11bubuko.com,布布扣%>
12bubuko.com,布布扣<html>
13bubuko.com,布布扣    <head>
14bubuko.com,布布扣        <title>JSON-RPC-Java Example by Huaxu</title>
15bubuko.com,布布扣        <script type="text/javascript" src="example.js"></script>
16bubuko.com,布布扣        <script type="text/javascript" src="jsonrpc.js"></script>
17bubuko.com,布布扣    </head>
18bubuko.com,布布扣    <style>
19bubuko.com,布布扣        em{
20bubuko.com,布布扣            color: red;
21bubuko.com,布布扣        }
22bubuko.com,布布扣    </style>
23bubuko.com,布布扣    <body bgcolor="#ffffff" onLoad="onLoad()">
24bubuko.com,布布扣        <center>
25bubuko.com,布布扣            <h2>
26bubuko.com,布布扣                JSON-RPC-Java Example by Huaxu
27bubuko.com,布布扣            </h2>
28bubuko.com,布布扣            <p>
29bubuko.com,布布扣                The JSON-RPC-Java
30bubuko.com,布布扣                <em>Example</em> application .
31bubuko.com,布布扣            </p>
32bubuko.com,布布扣            <p>
33bubuko.com,布布扣                <strong>Who:</strong>
34bubuko.com,布布扣                <input type="text" id="who" size="30" value="Huaxu" />
35bubuko.com,布布扣                <br />
36bubuko.com,布布扣                <br />
37bubuko.com,布布扣                <input type="button" value="Say String" onclick="sayString()" />
38bubuko.com,布布扣                <input type="button" value="Say List" onclick="sayList()" />
39bubuko.com,布布扣                <input type="button" value="Say Map" onclick="sayMap()" />
40bubuko.com,布布扣                <input type="button" value="Say Set" onclick="saySet()" />
41bubuko.com,布布扣                <input type="button" value="Say User" onclick="sayUser()" />
42bubuko.com,布布扣            <hr />
43bubuko.com,布布扣            </p>
44bubuko.com,布布扣            <div id="say" style="width: 400px">
45bubuko.com,布布扣            </div>
46bubuko.com,布布扣        </center>
47bubuko.com,布布扣    </body>
48bubuko.com,布布扣</html>
49bubuko.com,布布扣


源码下载:JsonRpcExample 

JSON-RPC轻量级远程调用协议介绍及使用

标签:style   blog   http   io   color   ar   os   使用   java   

原文地址:http://www.cnblogs.com/steven9801/p/4086508.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!