Android持久化保存cookie的方法

在解析网页信息的时候,需要登录后才能访问,所以使用httpclient模拟登录,然后把cookie保存下来,以供下一次访问使用,这时就需要持久化cookie中的内容。

在之前先科普一下基础知识:

什么是Cookies?

Cookies是一些小文件,它们被创建在客户端的系统里,或者被创建在客户端浏览器的内存中(如果是临时性的话)。用它可以实现状态管理的功能。我们可以存储一些少量信息到可以短的系统上,以便在需要的时候使用。最有趣的事情是,它是对用户透明的。在你的web应用程序中,你可以到处使用它,它极其得简单。Cookies是以文本形式存储的。如果一个web应用程序使用cookies,那么服务器负责发送cookies,客户端浏览器将存储它。浏览器在下次请求页面的时候,会返回cookies给服务器。最常用的例子是,使用一个cookie来存储用户信息,用户的喜好,“记住密码”操作等。Cookies有许多优点,当然也有许多缺点。我将在接下来讲述。

Cookies是如何创建的?

当一个客户端向服务器发出请求,服务器发送cookies给客户端。而相同的cookies可以被后续的请求使用。例如,如果codeproject.com将Session ID作为cookies存储。当一个客户端首次向web服务器请求页面,服务器生成Session ID,并将其作为cookies发送往客户端。

现在,所有来自相同客户端的后续请求,它将使用来自cookies的Session ID,就像下面这幅图片展示的那样。

浏览器和web服务器以交换cookies信息来作为响应。对不同的站点,浏览器会维护不同的cookies。如果一个页面需要cookies中的信息,当某个URL被“点击”,首先浏览器将搜索本地系统的cookies的信息,然后才转向服务器来获得信息。

Cookies的优势

下面是使用cookies的主要优势:

(1)    实现和使用都是非常简单的

(2)    由浏览器来负责维护发送过来的数据(cookies内容)

(3)    对来自多个站点的cookies来讲,浏览器自动管理它们

Cookies的劣势

下面是cookies的主要劣势:

(1)    它以简单的文本格式来存储数据,所以它一点也不安全

(2)    对于cookies数据,有大小限制(4kB)

(3)    Cookies最大数目也有限制。主流浏览器提供将cookies的个数限制在20条。如果新cookies到来,那么老的将被删除。有些浏览器能支持到300条的cookies数。

(4)    我们需要配置浏览器,cookies将不能工作在浏览器配置的高安全级别环境下。

什么是持久化的和非持久化的Cookies

我们可以将cookies分成两类:

(1)    持久化的cookies

(2)    非持久化的cookies

持久化的cookies:这可以被称为永久性的cookies,它被存储在客户端的硬盘内,直到它们失效。持久化的cookies应该被设置一个失效时间。有时,它们会一直存在直到用户删除它们。持久化的cookies通常被用来为某个系统收集一个用户的标识信息。

非持久化cookies:也可以被称之为临时性的cookies。如果没有定义失效时间,那么cookie将会被存储在浏览器的内存中。我上面展示的例子就是一个非持久的cookies。

修改一个持久化的cookies与一个非持久化的cookies并没有什么不同。它们唯一的区别是——持久化的cookies有一个失效时间的设置。

Cookie持久化

HttpClient可以和任意物理表示的实现了CookieStore接口的持久化cookie存储一起使用。默认的CookieStore实现称为BasicClientCookie,这是凭借java.util.ArrayList的一个简单实现。在BasicClientCookie对象中存储的cookie当容器对象被垃圾回收机制回收时会丢失。如果需要,用户可以提供更复杂的实现。

下载着重介绍在安卓中如何利用httpclient来实现对cookie的持久化操作:

一、请求网络获取cookie

先看一下下面的代码:

DefaultHttpClient httpclient = new DefaultHttpClient();
HttpGet httpget = new HttpGet("http://www.hlovey.com");
HttpResponse response = httpclient.execute(httpget);
HttpEntity entity = response.getEntity();
List<Cookie> cookies = httpclient.getCookieStore().getCookies(); 

Post模拟登录

HttpPost httpPost = new HttpPost(url);
List<NameValuePair> formparams = new ArrayList<NameValuePair>();
formparams.add(new BasicNameValuePair("id", userid));
formparams.add(new BasicNameValuePair("passwd", passwd));
UrlEncodedFormEntity entity;
try {
 entity = new UrlEncodedFormEntity(formparams, mobileSMTHEncoding);
} catch (UnsupportedEncodingException e1) {
 return 3;
}
httpPost.setEntity(entity);
httpPost.setHeader("User-Agent", userAgent);
HttpResponse response = httpClient.execute(httpPost); 

二、保存cookie

保存cookie有两种方式一种是数据库,另一种是SharedPreferences,其中//www.jb51.net/article/140423.htm是使用数据库来保存的,这里我是使用SharedPreferences保存。

 package com.smthbest.smth.util;
 import java.util.Locale;
 import android.content.Context;
 import android.content.SharedPreferences;
 import android.text.TextUtils;
 import android.util.Log;
 import org.apache.http.client.CookieStore;
 import org.apache.http.cookie.Cookie;
 import java.io.ByteArrayInputStream;
 import java.io.ByteArrayOutputStream;
 import java.io.ObjectInputStream;
 import java.io.ObjectOutputStream;
 import java.util.ArrayList;
 import java.util.Date;
 import java.util.List;
 import java.util.Locale;
 import java.util.concurrent.ConcurrentHashMap;
ic class PersistentCookieStore implements CookieStore {
private static final String LOG_TAG = "PersistentCookieStore";
private static final String COOKIE_PREFS = "CookiePrefsFile";
private static final String COOKIE_NAME_STORE = "names";
private static final String COOKIE_NAME_PREFIX = "cookie_";
private boolean omitNonPersistentCookies = false;
private final ConcurrentHashMap<String, Cookie> cookies;
private final SharedPreferences cookiePrefs; 

/**
 * Construct a persistent cookie store.
 *
 * @param context Context to attach cookie store to
 */
public PersistentCookieStore(Context context) {
 cookiePrefs = context.getSharedPreferences(COOKIE_PREFS, 0);
 cookies = new ConcurrentHashMap<String, Cookie>(); 

 // Load any previously stored cookies into the store
 String storedCookieNames = cookiePrefs.getString(COOKIE_NAME_STORE, null);
 if (storedCookieNames != null) {
 String[] cookieNames = TextUtils.split(storedCookieNames, ",");
 for (String name : cookieNames) {
  String encodedCookie = cookiePrefs.getString(COOKIE_NAME_PREFIX + name, null);
  if (encodedCookie != null) {
  Cookie decodedCookie = decodeCookie(encodedCookie);
  if (decodedCookie != null) {
   cookies.put(name, decodedCookie);
  }
  }
 } 

 // Clear out expired cookies
 clearExpired(new Date());
 }
} 

@Override
public void addCookie(Cookie cookie) {
 if (omitNonPersistentCookies && !cookie.isPersistent())
 return;
 String name = cookie.getName() + cookie.getDomain(); 

 // Save cookie into local store, or remove if expired
 if (!cookie.isExpired(new Date())) {
 cookies.put(name, cookie);
 } else {
 cookies.remove(name);
 } 

 // Save cookie into persistent store
 SharedPreferences.Editor prefsWriter = cookiePrefs.edit();
 prefsWriter.putString(COOKIE_NAME_STORE, TextUtils.join(",", cookies.keySet()));
 prefsWriter.putString(COOKIE_NAME_PREFIX + name, encodeCookie(new SerializableCookie(cookie)));
 prefsWriter.commit();
} 

@Override
public void clear() {
 // Clear cookies from persistent store
 SharedPreferences.Editor prefsWriter = cookiePrefs.edit();
 for (String name : cookies.keySet()) {
 prefsWriter.remove(COOKIE_NAME_PREFIX + name);
 }
 prefsWriter.remove(COOKIE_NAME_STORE);
 prefsWriter.commit(); 

 // Clear cookies from local store
 cookies.clear();
} 

@Override
public boolean clearExpired(Date date) {
 boolean clearedAny = false;
 SharedPreferences.Editor prefsWriter = cookiePrefs.edit(); 

 for (ConcurrentHashMap.Entry<String, Cookie> entry : cookies.entrySet()) {
 String name = entry.getKey();
 Cookie cookie = entry.getValue();
 if (cookie.isExpired(date)) {
  // Clear cookies from local store
  cookies.remove(name); 

  // Clear cookies from persistent store
  prefsWriter.remove(COOKIE_NAME_PREFIX + name); 

  // We've cleared at least one
  clearedAny = true;
 }
 } 

 // Update names in persistent store
 if (clearedAny) {
 prefsWriter.putString(COOKIE_NAME_STORE, TextUtils.join(",", cookies.keySet()));
 }
 prefsWriter.commit(); 

 return clearedAny;
} 

@Override
public List<Cookie> getCookies() {
 return new ArrayList<Cookie>(cookies.values());
} 

/**
 * Will make PersistentCookieStore instance ignore Cookies, which are non-persistent by
 * signature (`Cookie.isPersistent`)
 *
 * @param omitNonPersistentCookies true if non-persistent cookies should be omited
 */
public void setOmitNonPersistentCookies(boolean omitNonPersistentCookies) {
 this.omitNonPersistentCookies = omitNonPersistentCookies;
} 

/**
 * Non-standard helper method, to delete cookie
 *
 * @param cookie cookie to be removed
 */
public void deleteCookie(Cookie cookie) {
 String name = cookie.getName();
 cookies.remove(name);
 SharedPreferences.Editor prefsWriter = cookiePrefs.edit();
 prefsWriter.remove(COOKIE_NAME_PREFIX + name);
 prefsWriter.commit();
} 

/**
 * Serializes Cookie object into String
 *
 * @param cookie cookie to be encoded, can be null
 * @return cookie encoded as String
 */
protected String encodeCookie(SerializableCookie cookie) {
 if (cookie == null)
 return null;
 ByteArrayOutputStream os = new ByteArrayOutputStream();
 try {
 ObjectOutputStream outputStream = new ObjectOutputStream(os);
 outputStream.writeObject(cookie);
 } catch (Exception e) {
 return null;
 } 

 return byteArrayToHexString(os.toByteArray());
} 

/**
 * Returns cookie decoded from cookie string
 *
 * @param cookieString string of cookie as returned from http request
 * @return decoded cookie or null if exception occured
 */
protected Cookie decodeCookie(String cookieString) {
 byte[] bytes = hexStringToByteArray(cookieString);
 ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(bytes);
 Cookie cookie = null;
 try {
 ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream);
 cookie = ((SerializableCookie) objectInputStream.readObject()).getCookie();
 } catch (Exception exception) {
 Log.d(LOG_TAG, "decodeCookie failed", exception);
 } 

 return cookie;
} 

/**
 * Using some super basic byte array <-> hex conversions so we don't have to rely on any
 * large Base64 libraries. Can be overridden if you like!
 *
 * @param bytes byte array to be converted
 * @return string containing hex values
 */
protected String byteArrayToHexString(byte[] bytes) {
 StringBuilder sb = new StringBuilder(bytes.length * 2);
 for (byte element : bytes) {
 int v = element & 0xff;
 if (v < 16) {
  sb.append('0');
 }
 sb.append(Integer.toHexString(v));
 }
 return sb.toString().toUpperCase(Locale.US);
} 

/**
 * Converts hex values from strings to byte arra
 *
 * @param hexString string of hex-encoded values
 * @return decoded byte array
 */
protected byte[] hexStringToByteArray(String hexString) {
 int len = hexString.length();
 byte[] data = new byte[len / 2];
 for (int i = 0; i < len; i += 2) {
 data[i / 2] = (byte) ((Character.digit(hexString.charAt(i), 16) << 4) + Character.digit(hexString.charAt(i + 1), 16));
 }
 return data;
} 

使用PersistentCookieStore来存储cookie,首先最好把PersistentCookieStore放在Application获取其他的地方,取得唯一实例,保存cookie是在登录成功后,从下面代码获取保存。

PersistentCookieStore myCookieStore = App.getInstance().getPersistentCookieStore();
List<Cookie> cookies = httpClient.getCookieStore().getCookies();
for (Cookie cookie:cookies){
 myCookieStore.addCookie(cookie);
} 

三、cookie的使用

PersistentCookieStore cookieStore = new PersistentCookieStore(SmthBestApp.getInstance().getApplicationContext());
httpClient.setCookieStore(cookieStore);
HttpResponse response = httpClient.execute(httpget); 

这样就可以免再次登录了。

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

(0)

相关推荐

  • Android 登录Web 时对cookie 处理

    对于登录功能本身没有任何特别,使用httpclient向服务器post用户名密码即可. 但是为了保持登录的状态(在各个Activity之间切换时要让网站知道用户一直是处于登录的状态)就需要进行cookie的读写. httpclient相当强大,读写cookie非常容易: CookieStore cookies=((AbstractHttpClient)client).getCookieStore();//读cookie ((AbstractHttpClient) client).setCooki

  • Android 中cookie的处理详解

    android 客户端 Cookie处理 Cookie,有时也用其复数形式Cookies,指某些网站为了辨别用户身份.进行session跟踪而储存在用户本地终端上的数据(通常经过加密). Cookie最早是网景公司的前雇员Lou Montulli在1993年3月的发明. 发起 Cookie是由服务器端生成,发送给User-Agent(一般是浏览器,客户端),浏览器会将Cookie的key/value保存到某个目录下的文本文件内,下次请求同一网站时就发送该Cookie给服务器(前提是浏览器设置为启

  • AndroidHttpClient使用Cookie应用分析

    今天想把一个用使用了HttpClient的自动签到小程序移植到Android上,还好Android的SDK自带了HttpClient的包.翻Android的文档时发现官方还提供了一个实现了HttpClient接口的AndroidHttpClient,上网搜了下没发现关于AndroidHttpClient的文章.当然也可以继续使用DefaultHttpClient,但用为Android定制的AndroidHttpClient自然更好. 下面是2个测试用的HttpServlet: 复制代码 代码如下

  • android实现http中请求访问添加cookie的方法

    本文实例讲述了android实现http中请求访问添加cookie的方法.分享给大家供大家参考,具体如下: 第一种 HashMap<String, String> map = new HashMap<String, String>(); map.put("cookie","p1u_id=4eb591e73554db0f4d3300cb656113abfb968ef6b0ee2b5de0a35caa5217c51faa028b453576b35c&quo

  • Android持久化保存cookie的方法

    在解析网页信息的时候,需要登录后才能访问,所以使用httpclient模拟登录,然后把cookie保存下来,以供下一次访问使用,这时就需要持久化cookie中的内容. 在之前先科普一下基础知识: 什么是Cookies? Cookies是一些小文件,它们被创建在客户端的系统里,或者被创建在客户端浏览器的内存中(如果是临时性的话).用它可以实现状态管理的功能.我们可以存储一些少量信息到可以短的系统上,以便在需要的时候使用.最有趣的事情是,它是对用户透明的.在你的web应用程序中,你可以到处使用它,它

  • python实现12306登录并保存cookie的方法示例

    经过倒腾12306的登录,还是实现了,请求头很重要...各位感兴趣的可以继续写下去..... import sys import time import requests from PIL import Image import json import os import Headers import SessionUtil import UrlUtils class Ticket(object): def __init__(self): self.answer = { "1": &q

  • jQuery实现元素拖拽并cookie保存顺序的方法

    本文实例讲述了jQuery实现元素拖拽并cookie保存顺序的方法.分享给大家供大家参考,具体如下: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"&

  • Android编程实现录音及保存播放功能的方法【附demo源码下载】

    本文实例讲述了Android编程实现录音及保存播放功能的方法.分享给大家供大家参考,具体如下: 在android中进行录音相对来说是比较简单的,使用系统提供的MediaRecorder类进行录音并保存,然后调用MediaPlayer进行播放.以下为xml配置文件代码: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas

  • laravel 修改记住我功能的cookie保存时间的方法

    1.laravel的记住我功能默认cookie时间为 5年. 2. 我的思路是把cookie取出来,然后再加个过期时间. 开始演示代码. 这是我的登陆 所以只好写到中间件 以上这篇laravel 修改记住我功能的cookie保存时间的方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们.

  • Android 保存文件路径方法

    Android保存文件到本地路径问题 常见路径 例如: application 包名: com.my.company 项目名: chat /data/data == ///data/user/0 getExternalFilesDir()方法可以获取到 SDCard/Android/data/你的应用的包名/files/ 目录, 一般放一些长时间保存的数据 getExternalCacheDir()方法可以获取到 SDCard/Android/data/你的应用包名/cache/目录, 一般存放

  • Python3 Post登录并且保存cookie登录其他页面的方法

    如下所示: import urllib.request import sys import http.cookiejar import urllib.parse from bs4 import BeautifulSoup import codecs import re #登录页面 url = "http://www.abc.com/login.asp" #登录Post数据 postdata =urllib.parse.urlencode({ "username":&

  • android读写cookie的方法示例

    做了一个android网络应用,要求用自己实现的webview去访问web网站,并且在远程登录成功之后把cookie写入到手机,保留用作以后的自动登录.找了好多资料.发觉读取cookies倒还用的很普遍,可是通过程序写cookie却没有太多资料. 先来看一下如何读取cookie吧: try { DefaultHttpClient httpclient = new DefaultHttpClient(); HttpGet httpget = new HttpGet("http://www.hlov

  • Android数据共享 sharedPreferences 的使用方法

    Android数据共享 sharedPreferences 的使用方法 Android 中通过 sharedPreferences 来持久化存储数据并进行共享 在 Activity 或存在 Context 环境中即可使用 context.getSharedPreferences(name, Context.MODE_PRIVATE); 设置要保存的数据: mSp = context.getSharedPreferences(name, Context.MODE_PRIVATE); mEditor

随机推荐