Android使用OkHttp进行重定向拦截处理的方法

网上有很多的OkHttp的教程,但是并没有一个是关于如何OkHttp处理重定向的。这里的处理重定向的意思是:把重定向请求拦截下来,然后我们自己去请求重定向后的网页,然后通过Jsoup解析自己需要的网页数据。比如说我们模拟用户登录,然后自己去请求解析登陆后跳转的网页的内容。为什么要做这样的一个东西呢?比如说课程表的查成绩功能,就可以使用这种方法来获取成绩。

大概的原理是怎样的呢?

我们先来说一下浏览器是怎么样做用户登录的:浏览器会将你输入的帐号和密码通过POST请求携带过去,当然可能还会有其它字段,因为这个POST请求是我们网页和服务器规定好的;登录成功后,服务器会返回一个Set-Cookie请求头字段,有了Cookie浏览器就可以通过GET请求访问登录后的网页,注意没有这个Cookie是无法请求登陆后的网页的,GET请求必须设置Cookie请求头字段,将服务器返回的Cookie携带过去。

明白了浏览器的行为之后,我们知道需要做的步骤就是4步:

  1. 自己做POST请求,并且不让它自动重定向
  2. 我们拿到POST请求返回来的响应,获取对应的Set-Cookie字段的内容
  3. 将该对应的内容添加到GET请求的Cookie请求头字段中,然后做GET请求
  4. 获取到的GET请求的响应体就是我们登陆后的网页内容了,如果是静态网页可以通过Jsoup解析自己想要的信息了

我们需要通过抓包或者Chorme浏览器自带的请求查看功能来查看POST请求提交的表单是怎样的。Chorme按F12即可

输入帐号密码点击登录即可

点击登陆后我们可以看到,控制面板出现了一堆访问的记录,第一个就是我们要找的,第一个的Status标志是302,302是重定向的意思。我们点击(pass.asp)这个请求,然后查看它的POST的请求头(点击Headers)。

我们看到了一堆的Set-Cookie字段,字段对应的内容就是我们要携带做GET请求的

同时我们可以看到POST提交的表单内容,有些网站的提交参数是经过加密的,如果要做通用的,我们需要找到它加密的方法,做同样的加密处理。

接下来使用OkHttp进行操作

由于OkHttp提供了自动携带Cookie进行请求的功能,于是我们可以很方便地进行处理了。

final OkHttpClient client = new OkHttpClient().newBuilder()
        .followRedirects(false) //禁制OkHttp的重定向操作,我们自己处理重定向
        .followSslRedirects(false)
        .cookieJar(new LocalCookieJar())  //为OkHttp设置自动携带Cookie的功能
        .build();

//CookieJar是用于保存Cookie的
class LocalCookieJar implements CookieJar{
  List<Cookie> cookies;
  @Override
  public List<Cookie> loadForRequest(HttpUrl arg0) {
     if (cookies != null)
        return cookies;
      return new ArrayList<Cookie>();
  }

  @Override
  public void saveFromResponse(HttpUrl arg0, List<Cookie> cookies) {
    this.cookies = cookies;
  }

}

为什么设置CookieJar就能自动携带Cookie了呢?给你看一段OkHttp的源码就知道了。

/**
  * Populates request with defaults and cookies.
  *
  * <p>This client doesn't specify a default {@code Accept} header because it doesn't know what
  * content types the application is interested in.
  */
 private Request networkRequest(Request request) throws IOException {
  Request.Builder result = request.newBuilder();

  //如果CookieJar的Cookie不为空,则设置Cookie字段
  List<Cookie> cookies = client.cookieJar().loadForRequest(request.url());
  if (!cookies.isEmpty()) {
   result.header("Cookie", cookieHeader(cookies));
  }

  return result.build();
 }

于是接下来我们就是做POST请求了

    final OkHttpClient client = new OkHttpClient().newBuilder()
        .followRedirects(false)
        .followSslRedirects(false)
        .cookieJar(new LocalCookieJar())
        .build();
    //构造一个POST请求
    RequestBody body = new FormBody.Builder().add("UserStyle", "student")
        .add("user", "xxx").add("password", "xxx").build();

    Request request = new Request.Builder().url("http://222.195.8.201/pass.asp").post(body).build();

    client.newCall(request).enqueue(new Callback() {

      @Override
      public void onResponse(Call call, Response response) throws IOException {
        /**
         * 如果不用CookieJar,那么就要自己去解析返回的Set-Cookie字段,解析之后通过addHeader("Cookie", cookie)
         * 添加Cookie请求头
         */
//       List<String> cookies = response.headers("Set-Cookie");
//       String cookie = "";
//       for(int i=cookies.size()-1; i>=0; i--){
//         cookie = cookie+ cookies.get(i).replace("path=/", "") + " ";
//       }

        //做GET请求
        Request redirectRequest = new Request.Builder().url("http://222.195.8.201/student/asp/Select_Success.asp")
//          .addHeader("Cookie", cookie)
           .build();
        //拿到登陆后操作的某个网页的内容
        Response response2 = client.newCall(redirectRequest).execute();
        String result = response2.body().string();
        System.out.println(result);
      }

      @Override
      public void onFailure(Call arg0, IOException arg1) {

      }
    });

拿到内容后就可以自己进行内容的解析和展示了。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android 中okhttp自定义Interceptor(缓存拦截器)

    Android 中okhttp自定义Interceptor(缓存拦截器) 前言: 新公司项目是没有缓存的,我的天,坑用户流量不是么.不知道有人就喜欢一个界面没事点来点去的么.怎么办?一个字"加". 由于项目的网络请求被我换成了retrofit.而retrofit的网络请求默认基于okhttp okhttp的缓存由返回的header 来决定.如果服务器支持缓存的话返回的headers里面会有这一句 "Cache-Control","max-age=time&

  • OKHttp3(支持Retrofit)的网络数据缓存Interceptor拦截器的实现

    前言:前段时间在开发APP的时候,经常出现由于用户设备环境的原因,拿不到从网络端获取的数据,所以在APP端展现的结果总是一个空白的框,这种情况对于用户体验来讲是极其糟糕的,所以,苦思冥想决定对OKHTTP下手(因为我在项目中使用的网络请求框架就是OKHTTP),则 写了这么一个网络数据缓存拦截器. OK,那么我们决定开始写了,我先说一下思路: 思路篇 既然要写的是网络数据缓存拦截器,主要是利用了OKHTTP强大的拦截器功能,那么我们应该对哪些数据进行缓存呢,或者在哪些情况下启用数据进行缓存机制呢

  • Android的OkHttp包中的HTTP拦截器Interceptor用法示例

    OkHttp(GitHub:https://github.com/square/okhttp) 的 Interceptor 就如同名称「拦截器」一样,拦截你的 Request 做一些你想做的事情再送出去.例如: 1.自动加上使用者目前使用的语言送出去取得对应语言的回传内容. 2.将 Request 计算出这个 Request 的 sigunature 再附加上送出去. 在 okHttp 中分成 Application Interceptor 和 Network Interceptor 两种. A

  • Android OKHttp3拦截器的使用方法

    本文介绍了Android OKHttp3拦截器的使用方法,分享给大家,具体如下: 添加Interceptor 在上一篇中我们已经知道了okhttp的基本使用,其中在介绍OkHttpClient初始化的时候,介绍了两种方式,第二种方式就可以对这个OkHttpClient对象设置拦截器,如下所示: // 配置一些信息进入OkHttpClient mOkHttpClient = new OkHttpClient().newBuilder() .connectTimeout(REQUEST_TIME,

  • Android 封装Okhttp+Retrofit+RxJava,外加拦截器实例

    1.创建一个接口,用来定义接口使用的 public interface Api { @POST("product/getProductDetail") Observable<Goods_Bean> getGoods(@QueryMap Map<String,String> map); @POST("product/addCart") Observable<Add_Bean> getAdd(@QueryMap Map<Stri

  • Android使用OkHttp进行重定向拦截处理的方法

    网上有很多的OkHttp的教程,但是并没有一个是关于如何OkHttp处理重定向的.这里的处理重定向的意思是:把重定向请求拦截下来,然后我们自己去请求重定向后的网页,然后通过Jsoup解析自己需要的网页数据.比如说我们模拟用户登录,然后自己去请求解析登陆后跳转的网页的内容.为什么要做这样的一个东西呢?比如说课程表的查成绩功能,就可以使用这种方法来获取成绩. 大概的原理是怎样的呢? 我们先来说一下浏览器是怎么样做用户登录的:浏览器会将你输入的帐号和密码通过POST请求携带过去,当然可能还会有其它字段

  • Android开发OkHttp执行流程源码分析

    目录 前言 介绍 执行流程 OkHttpClient client.newCall(request): RealCall.enqueue() Dispatcher.enqueue() Interceptor RetryAndFollowUpInterceptor BridgeInterceptor CacheInterceptor 前言 OkHttp 是一套处理 HTTP 网络请求的依赖库,由 Square 公司设计研发并开源,目前可以在 Java 和 Kotlin 中使用. 对于 Androi

  • Android使用OKHttp库实现视频文件的上传到服务器功能

    1 服务器接口简介 此处我使用的服务器接口是使用Flask编写,具体实现代码: # -*- coding: utf-8 -*- from flask import Flask, render_template, jsonify, request import time import os import base64 app = Flask(__name__) UPLOAD_FOLDER = 'E:\myupload\picture' app.config['UPLOAD_FOLDER'] = U

  • android 使用OkHttp上传多张图片的实现代码

    简述 还是先来说说为啥用OkHttp作为多图片上传的框架,原因有两点: 1.OkHttp可以作为Volley底层传输协议,速度更快 2.使用Xutils和KJFramework上传图片存在一个小问题,首先,可以上传,并且可以上传多张图片,也可以上传其他的参数,那问题在哪里呢?在后台接受参数时很不灵活,Xutlis及KJFramework使用HashMap来上传每个参数,每一张图片也必须有一个唯一的key,上传一张图片就要定义一个参数来接收,上传两张图片就要定义两个参数来接收,当上传的图片数量不确

  • Android WebView 内处理302重定向不跳转的解决

    最近项目中Webview加载第三方的具有302的重定向,但是却发现却没有重新跳转.最后发现问题如下: public boolean shouldOverrideUrlLoading(WebView view, String url) { super.shouldOverrideUrlLoading(view, url); ........ ......... return true: } 发现最后返回的true,所以导致不会重定向跳转,只要返回fasle就可以重定向跳转了,如果你某些情况下需要,

  • Android实现WebView点击拦截跳转原生

    1. 首先设置Web视图 webview.setWebViewClient(new MyWebViewClient()); webview1.setWebViewClient(new MyWebViewClient()); 2. 拦截点击的链接,跳转到对应的页面 // 监听 所有点击的链接,如果拦截到我们需要的,就跳转到相对应的页面. private class MyWebViewClient extends WebViewClient { @Override public boolean sh

  • 在Android环境下WebView中拦截所有请求并替换URL示例详解

    需求背景 接到这样一个需求,需要在 WebView 的所有网络请求中,在请求的url中,加上一个xxx=1的标志位. 例如 http://www.baidu.com 加上标志位就变成了 http://www.baidu.com?xxx=1 寻找解决方案 从 Android API 11 (3.0) 开始,WebView 开始在 WebViewClient 内提供了这样一条 API ,如下: public WebResourceResponse shouldInterceptRequest(Web

随机推荐