HttpClient方式实现RPC远程调用

问题:

RPC,所谓远程调用,就是由服务消费者发起请求调用服务提供者中的方法完成功能处理,然后服务提供者再将执行结果响应给服务消费者的一个过程。那么浏览器和服务器的交互方式,好像就是远程调用,那么我们能不能参照
浏览器和服务器交互的模式来实现项目之间的远程调用呢?

方案:

  • 服务器消费者--->浏览器--->在服务消费者中声明类似浏览器发起请求和接收响应的代码。
  • 服务器提供者--->MVC的web项目

实现:

HttpClient

使用:

1、服务提供者的代码就是正常的Web项目
2、服务器消费中导入HttpClient的jar包,然后编写相关代码完成rpc远程调用,参照源码

一.服务提供方代码

package com.bjsxt.controller;

import java.util.List;

import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

import com.alibaba.fastjson.JSON;
import com.bjsxt.pojo.Pet;
import com.bjsxt.service.PetService;

@Controller
public class PetCon {
    @Resource
    private PetService petServiceImpl;
    
    /**
     * 获取所有的宠物信息
     */
    @RequestMapping("pet")
    public String getPet(HttpServletRequest req){
        List<Pet> pl = petServiceImpl.getPetInfoService();
        req.setAttribute("pl", pl);
        return "pet.jsp";   
    }
    /**
     * 获取所有的宠物信息
     * @param pname
     * @param color
     * @return
     */
    @ResponseBody
    @RequestMapping("pet2")
    public List<Pet> getPet(String pname,String color){
        System.out.println("Pet2请求数据===="+pname+":"+color);
        List<Pet> pl = petServiceImpl.getPetInfoService();
        return pl;
        
    }
    
    /**
     * 处理Ajax请求,返回所有的宠物信息,Jsop方式
     * @param pname
     * @param color
     * @return 
     * callback({name:zhansgan,age:18})
     */
    @ResponseBody
    @RequestMapping(value="petAjax",produces="text/script;charset=UTF-8")
    public String getPetAjax(String callback,String pname,String color){
        System.out.println("getPetAjax请求数据==跨域=="+pname+":"+color);
        List<Pet> pl = petServiceImpl.getPetInfoService();
        //将数据转换为json字符串
        String jsonString = JSON.toJSONString(pl);
        return callback+"("+jsonString+")";
    }
    /**
     * 获取所有的宠物信息,coresFilter方式
     */
    @ResponseBody
    @RequestMapping("petFilter")
    public List<Pet> getPetFilter(String pname,String color){
        System.out.println("Pet2请求数据===="+pname+":"+color);
        List<Pet> pl = petServiceImpl.getPetInfoService();
        return pl;
        
    }
        
}

service接口和service实现类略....

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/xsd/maven-4.0.0.xsd">
  <modelVersion>4.0.0</modelVersion>
  <groupId>com.bjsxt.rpc</groupId>
  <artifactId>04-HttpServer</artifactId>
  <version>0.0.1-SNAPSHOT</version>
  <packaging>war</packaging>
   <properties>
    <commons.version>1.2</commons.version>
    <jsp.version>2.0</jsp.version>
    <jstl.version>1.2</jstl.version>
    <servlet.version>2.5</servlet.version>
    <spring.version>4.3.18.RELEASE</spring.version>
  </properties>
  <dependencies>
    <dependency>
      <groupId>commons-logging</groupId>
      <artifactId>commons-logging</artifactId>
      <version>${commons.version}</version>
    </dependency>
    
    <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>jsp-api</artifactId>
        <version>${jsp.version}</version>
        <scope>provided</scope>
    </dependency>
    <!-- servlet -->
    <dependency>
        <groupId>javax.servlet</groupId>
          <artifactId>servlet-api</artifactId>
          <version>${servlet.version}</version>
          <scope>provided</scope>
    </dependency>
    <!-- jstl -->
    <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>jstl</artifactId>
        <version>${jstl.version}</version>
    </dependency>
    
    <dependency>
         <groupId>org.springframework</groupId>
         <artifactId>spring-webmvc</artifactId>
         <version>${spring.version}</version>
    </dependency>
    <dependency>
        <groupId>com.fasterxml.jackson.core</groupId>
        <artifactId>jackson-databind</artifactId>
        <version>2.9.0</version>
    </dependency>
    <dependency>
        <groupId>com.alibaba</groupId>
        <artifactId>fastjson</artifactId>
        <version>1.2.47</version>
    </dependency>
    <!-- 添加CoresFilter过滤器依赖 -->
    <dependency>
    
        <groupId>com.thetransactioncompany</groupId>
        <artifactId>cors-filter</artifactId>
        <version>2.5</version>
    </dependency>
  </dependencies>
  
  
  <build>
     <plugins>
        <!-- 
            tomcat插件,maven内嵌的tomcat 
            作用:用来测试maven的web项目
        -->
        <plugin>
             <groupId>org.apache.tomcat.maven</groupId>
             <artifactId>tomcat7-maven-plugin</artifactId>
             <version>2.2</version>
             <configuration>
                <port>7070</port><!-- tomcat的访问端口 -->
                <path>/server</path><!-- web项目的上下文 -->
                <!-- http://localhost:8080/xxx -->
             </configuration>
        </plugin>
    </plugins>
    
  </build>
  
</project>

三.服务消费方代码(重点)

package com.bjsxt.main;

import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.List;

import org.apache.http.Header;
import org.apache.http.HttpResponse;
import org.apache.http.client.HttpClient;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.message.BasicNameValuePair;

import com.alibaba.fastjson.JSON;
import com.bjsxt.pojo.Pet;

public class TestHttpClient {
    public static void main(String[] args) {
        
        try {
            //获取HttpClient链接对象
            HttpClient hc=HttpClients.createDefault();
            //创建URL
            String url="http://localhost:7070/server/pet2";
            //创建请求对象
            HttpPost hp=new HttpPost(url);
            //创建请求实体
                //创建BasicNameValuePair对象,存储键值对请求数据
                BasicNameValuePair bp1=new BasicNameValuePair("pname", "大黄");
                BasicNameValuePair bp2=new BasicNameValuePair("color", "黄色");
                //创建List集合对象,存储请求实体数据
                List<BasicNameValuePair> lb=new ArrayList<>();
                lb.add(bp1);
                lb.add(bp2);
                //创建封装请求实体数据的对象
                UrlEncodedFormEntity     ufo=new  UrlEncodedFormEntity(lb,"UTF-8");
                //设置实体数据到HttpPost请求对象中
                hp.setEntity(ufo);
            //发起请求
                HttpResponse resp = hc.execute(hp);
            //获取响应头
            Header[] allHeaders = resp.getAllHeaders();
            for(Header h:allHeaders){
                System.out.println(h.getName()+":"+h.getValue());
            }
            //获取响应实体
            InputStream is = resp.getEntity().getContent();
            int len=0;
            //将is字节流,转化为字符流
            InputStreamReader reader = new InputStreamReader(is);
            
            char[] buf=new char[1024];          //创建StringBuffer对象
            StringBuffer result = new StringBuffer();
            while((len=reader.read(buf))!=-1){
                result.append(String.valueOf(buf,0,len));
            }
            System.out.println(result);
            List<Pet> parseArray = JSON.parseArray(result.toString(), Pet.class);
            System.out.println(parseArray);
        } catch (Exception e) {
            // TODO: handle exception
        }
    }
}

pom.xml文件

  <dependencies>
    <!-- httpclient客户端工具包 -->
    <dependency>
        <groupId>org.apache.httpcomponents</groupId>
        <version>4.3.5</version>
        <artifactId>httpclient</artifactId>
    </dependency>
    <!-- javabean对象和json串之间转化为工具包 -->
    <dependency>
        <groupId>com.alibaba</groupId>
        <artifactId>fastjson</artifactId>
        <version>1.2.47</version>
    </dependency>
  </dependencies>
  
  
</project>

四.工具类

下面是自己常用的一个对post和get请求封装的HttpClient工具类

package com.ego.commons.utils;

import java.io.IOException;
import java.net.URI;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;

import org.apache.http.NameValuePair;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.utils.URIBuilder;
import org.apache.http.entity.ContentType;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.util.EntityUtils;

public class HttpClientUtil {

    public static String doGet(String url, Map<String, String> param) {

        // 创建Httpclient对象
        CloseableHttpClient httpclient = HttpClients.createDefault();

        String resultString = "";
        CloseableHttpResponse response = null;
        try {
            // 创建uri
            URIBuilder builder = new URIBuilder(url);
            if (param != null) {
                for (String key : param.keySet()) {
                    builder.addParameter(key, param.get(key));
                }
            }
            URI uri = builder.build();

            // 创建http GET请求
            HttpGet httpGet = new HttpGet(uri);

            // 执行请求
            response = httpclient.execute(httpGet);
            // 判断返回状态是否为200
            if (response.getStatusLine().getStatusCode() == 200) {
                resultString = EntityUtils.toString(response.getEntity(), "UTF-8");
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                if (response != null) {
                    response.close();
                }
                httpclient.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        return resultString;
    }

    public static String doGet(String url) {
        return doGet(url, null);
    }

    public static String doPost(String url, Map<String, String> param) {
        // 创建Httpclient对象
        CloseableHttpClient httpClient = HttpClients.createDefault();
        CloseableHttpResponse response = null;
        String resultString = "";
        try {
            // 创建Http Post请求
            HttpPost httpPost = new HttpPost(url);
            // 创建参数列表
            if (param != null) {
                List<NameValuePair> paramList = new ArrayList<>();
                for (String key : param.keySet()) {
                    paramList.add(new BasicNameValuePair(key, param.get(key)));
                }
                // 模拟表单
                UrlEncodedFormEntity entity = new UrlEncodedFormEntity(paramList,"utf-8");
                httpPost.setEntity(entity);
            }
            // 执行http请求
            response = httpClient.execute(httpPost);
            resultString = EntityUtils.toString(response.getEntity(), "utf-8");
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                response.close();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }

        return resultString;
    }

    public static String doPost(String url) {
        return doPost(url, null);
    }
    
    public static String doPostJson(String url, String json) {
        // 创建Httpclient对象
        CloseableHttpClient httpClient = HttpClients.createDefault();
        CloseableHttpResponse response = null;
        String resultString = "";
        try {
            // 创建Http Post请求
            HttpPost httpPost = new HttpPost(url);
            // 创建请求内容
            StringEntity entity = new StringEntity(json, ContentType.APPLICATION_JSON);
            httpPost.setEntity(entity);
            // 执行http请求
            response = httpClient.execute(httpPost);
            resultString = EntityUtils.toString(response.getEntity(), "utf-8");
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            try {
                response.close();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }

        return resultString;
    }
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,185评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,652评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,524评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,339评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,387评论 6 391
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,287评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,130评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,985评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,420评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,617评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,779评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,477评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,088评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,716评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,857评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,876评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,700评论 2 354