Android之三种网络请求解析数据(最佳案例)

时间:2022-11-05 11:38:34

AsyncTask解析数据

AsyncTask主要用来更新UI线程,比较耗时的操作可以在AsyncTask中使用。

AsyncTask是个抽象类,使用时需要继承这个类,然后调用execute()方法。注意继承时需要设定三个泛型Params,Progress和Result的类型,如AsyncTask<Void,Inetger,Void>:

  • Params是指调用execute()方法时传入的参数类型和doInBackgound()的参数类型
  • Progress是指更新进度时传递的参数类型,即publishProgress()和onProgressUpdate()的参数类型
  • Result是指doInBackground()的返回值类型

上面的说明涉及到几个方法:

  • doInBackgound() 这个方法是继承AsyncTask必须要实现的,运行于后台,耗时的操作可以在这里做
  • publishProgress() 更新进度,给onProgressUpdate()传递进度参数
  • onProgressUpdate() 在publishProgress()调用完被调用,更新进度

1.下面我们来看看主函数的布局:activity_main.xml,其中,这几个方法除了主函数不一样之外,其他的代码都一样。

[html] view plain copy

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent" >
  5. <EditText
  6. android:id="@+id/et_name"
  7. android:layout_width="wrap_content"
  8. android:layout_height="wrap_content"
  9. android:layout_alignParentLeft="true"
  10. android:layout_alignParentTop="true"
  11. android:layout_marginLeft="15dp"
  12. android:layout_marginTop="28dp"
  13. android:ems="10" >
  14. <requestFocus />
  15. </EditText>
  16. <Button
  17. android:id="@+id/bt_search"
  18. android:layout_width="wrap_content"
  19. android:layout_height="wrap_content"
  20. android:layout_alignBottom="@+id/et_name"
  21. android:layout_toRightOf="@+id/et_name"
  22. android:onClick="search"
  23. android:text="搜索" />
  24. <ListView
  25. android:id="@+id/listView"
  26. android:layout_width="match_parent"
  27. android:layout_height="wrap_content"
  28. android:layout_alignParentLeft="true"
  29. android:layout_below="@+id/et_name"
  30. android:layout_marginTop="26dp" >
  31. </ListView>
  32. </RelativeLayout>

2.适配器的布局:list_item.xml

[html] view plain copy

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. >
  6. <ImageView
  7. android:id="@+id/imageView"
  8. android:layout_width="60dp"
  9. android:layout_height="60dp"
  10. android:layout_alignParentLeft="true"
  11. android:layout_alignParentTop="true"
  12. android:src="@drawable/ic_launcher" />
  13. <TextView
  14. android:id="@+id/textView1"
  15. android:layout_width="wrap_content"
  16. android:layout_height="wrap_content"
  17. android:layout_alignParentTop="true"
  18. android:layout_toRightOf="@+id/imageView"
  19. android:text="TextView" />
  20. <TextView
  21. android:id="@+id/textView2"
  22. android:layout_width="wrap_content"
  23. android:layout_height="wrap_content"
  24. android:layout_alignLeft="@+id/textView1"
  25. android:layout_below="@+id/textView1"
  26. android:text="TextView" />
  27. <TextView
  28. android:id="@+id/textView3"
  29. android:layout_width="wrap_content"
  30. android:layout_height="wrap_content"
  31. android:layout_alignLeft="@+id/textView2"
  32. android:layout_below="@+id/textView2"
  33. android:text="TextView" />
  34. </RelativeLayout>

3.  主函数 MainActivity.java

[java] view plain copy

  1. import java.io.IOException;
  2. import java.io.InputStream;
  3. import java.net.HttpURLConnection;
  4. import java.net.URL;
  5. import java.util.ArrayList;
  6. import android.app.Activity;
  7. import android.os.AsyncTask;
  8. import android.os.Bundle;
  9. import android.view.View;
  10. import android.view.View.OnClickListener;
  11. import android.widget.Button;
  12. import android.widget.EditText;
  13. import android.widget.ListView;
  14. import com.bwie.test.Data1.MyData;
  15. import com.google.gson.Gson;
  16. import com.lidroid.xutils.HttpUtils;
  17. import com.lidroid.xutils.exception.HttpException;
  18. import com.lidroid.xutils.http.ResponseInfo;
  19. import com.lidroid.xutils.http.callback.RequestCallBack;
  20. import com.lidroid.xutils.http.client.HttpRequest.HttpMethod;
  21. public class MainActivity extends Activity {
  22. private EditText et_search;
  23. private ListView listView;
  24. private String trim;
  25. protected static final int SUCEESS = 0;
  26. private MyBaseAdapter adapter;
  27. ArrayList<MyData> myList = new ArrayList<MyData>();
  28. private Button bt_search;
  29. @Override
  30. protected void onCreate(Bundle savedInstanceState) {
  31. super.onCreate(savedInstanceState);
  32. setContentView(R.layout.activity_main);
  33. et_search = (EditText) findViewById(R.id.et_name);
  34. bt_search = (Button) findViewById(R.id.bt_search);
  35. listView = (ListView) findViewById(R.id.listView);
  36. // "http://93.gov.cn/93app/get_search.do?key=九三"
  37. bt_search.setOnClickListener(new OnClickListener() {
  38. @Override
  39. public void onClick(View v) {
  40. // TODO Auto-generated method stub
  41. trim = et_search.getText().toString().trim();
  42. String path = "http://93.gov.cn/93app/get_search.do?key="
  43. + trim;
  44. MyAsyncTask task = new MyAsyncTask();
  45. task.execute(path);
  46. }
  47. });
  48. }
  49. class MyAsyncTask extends AsyncTask<String, Void, String> {
  50. @Override
  51. protected String doInBackground(String... params) {
  52. try {
  53. URL url = new URL(params[0]);
  54. HttpURLConnection openConnection = (HttpURLConnection) url
  55. .openConnection();
  56. openConnection.setConnectTimeout(5000);
  57. openConnection.setReadTimeout(5000);
  58. int responseCode = openConnection.getResponseCode();
  59. if (responseCode == 200) {
  60. InputStream inputStream = openConnection.getInputStream();
  61. StreamUtils streamUtils = new StreamUtils();
  62. String parseStream = streamUtils.parseSteam(inputStream);
  63. System.out.println("-----------parseStream------------"
  64. + parseStream);
  65. return parseStream;
  66. }
  67. } catch (IOException e) {
  68. // TODO Auto-generated catch block
  69. e.printStackTrace();
  70. }
  71. return null;
  72. }
  73. @Override
  74. protected void onPostExecute(String result) {
  75. super.onPostExecute(result);
  76. Gson gson = new Gson();
  77. Data1 fromJson = gson.fromJson(result,
  78. Data1.class);
  79. System.out.println("-----------fromJson------------" + fromJson);
  80. ArrayList<MyData> data = fromJson.getData();
  81. listView.setAdapter(new MyBaseAdapter(
  82. MainActivity.this, data));
  83. }
  84. }
  85. }

4.适配器 MyBaseAdapter.java

[java] view plain copy

  1. import java.util.ArrayList;
  2. import com.bwie.test.Data1.MyData;
  3. import com.nostra13.universalimageloader.core.DisplayImageOptions;
  4. import com.nostra13.universalimageloader.core.ImageLoader;
  5. import android.content.Context;
  6. import android.view.View;
  7. import android.view.ViewGroup;
  8. import android.widget.BaseAdapter;
  9. import android.widget.ImageView;
  10. import android.widget.TextView;
  11. public class MyBaseAdapter extends BaseAdapter {
  12. private Context context;
  13. private ArrayList<MyData> list;
  14. private DisplayImageOptions initOptions;
  15. public MyBaseAdapter(Context context, ArrayList<MyData> list) {
  16. this.context = context;
  17. this.list = list;
  18. //创建initOptions
  19. initOptions = ImageLoaderUtils.initOptions();
  20. }
  21. @Override
  22. public int getCount() {
  23. return list.size();
  24. }
  25. @Override
  26. public Object getItem(int arg0) {
  27. return null;
  28. }
  29. @Override
  30. public long getItemId(int arg0) {
  31. return 0;
  32. }
  33. @Override
  34. public View getView(int position, View convertView, ViewGroup viewGroup) {
  35. View view = View.inflate(context, R.layout.list_item, null);
  36. ImageView imageView = (ImageView) view.findViewById(R.id.imageView);
  37. TextView textView1 = (TextView) view.findViewById(R.id.textView1);
  38. TextView textView2 = (TextView) view.findViewById(R.id.textView2);
  39. TextView textView3 = (TextView) view.findViewById(R.id.textView3);
  40. textView1.setText(list.get(position).FROMNAME);
  41. textView2.setText(list.get(position).TITLE);
  42. textView3.setText(list.get(position).SHOWTIME);
  43. //图片缓存机制
  44. ImageLoader.getInstance().displayImage(list.get(position).IMAGEURL, imageView,initOptions);
  45. return view;
  46. }
  47. }

5.实体类
Data1.class

[java] view plain copy

  1. import java.util.ArrayList;
  2. public class Data1 {
  3. public ArrayList<MyData> data;
  4. public class MyData{
  5. public String FROMNAME;
  6. public String IMAGEURL;
  7. public String SHOWTIME;
  8. public String TITLE;
  9. }
  10. public ArrayList<MyData> getData() {
  11. return data;
  12. }
  13. public void setData(ArrayList<MyData> data) {
  14. this.data = data;
  15. }
  16. }

6.图片缓存工具类 ImageLoaderUtils.java

[java] view plain copy

  1. import android.content.Context;
  2. import com.nostra13.universalimageloader.core.DisplayImageOptions;
  3. import com.nostra13.universalimageloader.core.ImageLoader;
  4. import com.nostra13.universalimageloader.core.ImageLoaderConfiguration;
  5. import com.nostra13.universalimageloader.core.ImageLoaderConfiguration.Builder;
  6. public class ImageLoaderUtils {
  7. public static void initConfiguration(Context context) {
  8. Builder builder = new ImageLoaderConfiguration.Builder(context);
  9. ImageLoader.getInstance().init(builder.build());
  10. }
  11. public static DisplayImageOptions initOptions() {
  12. DisplayImageOptions displayImageOptions = new DisplayImageOptions.Builder()
  13. .showImageOnLoading(R.drawable.ic_launcher)
  14. .showImageForEmptyUri(R.drawable.ic_launcher)
  15. .cacheInMemory(true).cacheOnDisk(true).build();
  16. return displayImageOptions;
  17. }
  18. }

7.   MyApplication.java

[java] view plain copy

  1. import android.app.Application;
  2. public class MyApplication extends Application {
  3. public void onCreate() {
  4. //初始化操作
  5. ImageLoaderUtils.initConfiguration(getApplicationContext());
  6. };
  7. }

8. 工具类 StreamUtils.java

[java] view plain copy

  1. import java.io.ByteArrayOutputStream;
  2. import java.io.IOException;
  3. import java.io.InputStream;
  4. public class StreamUtils {
  5. public static String parseSteam(InputStream inputStream) {
  6. try {
  7. // 定义一个字节数组输出流
  8. ByteArrayOutputStream arrayOutputStream = new ByteArrayOutputStream();
  9. // 定义一个字节数组
  10. byte[] buffer = new byte[1024];
  11. // 定义初始长度
  12. int len = 0;
  13. while ((len = inputStream.read(buffer)) != -1) {
  14. // 将读的内容,写到字节数组输出流中
  15. arrayOutputStream.write(buffer, 0, len);
  16. }
  17. // 将字节输出流转成字符串
  18. return arrayOutputStream.toString("utf-8");
  19. // utf-8 大小写都可以,gbk 必须大写
  20. } catch (IOException e) {
  21. e.printStackTrace();
  22. }
  23. return null;
  24. }
  1. }

到这里,一个案例就结束啦,下面的两个主函数是其他两个网络请求方法,只是主函数代码不同,其他的代码是相同的哦。

二.HttpURLConnection解析数据

针对JDK中的URLConnection连接Servlet的问题,网上有虽然有所涉及,但是只是说明了某一个或几个问题,是以FAQ的方式来解决的,而且比较零散,现在对这个类的使用就本人在项目中的使用经验做如下总结: 

1:> URL请求的类别: 

分为二类,GET与POST请求。二者的区别在于: 

     a:) get请求可以获取静态页面,也可以把参数放在URL字串后面,传递给servlet, 

     b:) post与get的不同之处在于post的参数不是放在URL字串里面,而是放在http请求的正文内。

请求响应流程

设置连接参数的方法

  • setAllowUserInteraction
  • setDoInput
  • setDoOutput
  • setIfModifiedSince
  • setUseCaches
  • setDefaultAllowUserInteraction
  • setDefaultUseCaches

设置请求头或响应头

HTTP请求允许一个key带多个用逗号分开的values,但是HttpURLConnection只提供了单个操作的方法:

  • setRequestProperty(key,value)
  • addRequestProperty(key,value)

setRequestProperty和addRequestProperty的区别就是,setRequestProperty会覆盖已经存在的key的所有values,有清零重新赋值的作用。而addRequestProperty则是在原来key的基础上继续添加其他value。

发送URL请求

建立实际连接之后,就是发送请求,把请求参数传到服务器,这就需要使用outputStream把请求参数传给服务器:

  • getOutputStream

获取响应

请求发送成功之后,即可获取响应的状态码,如果成功既可以读取响应中的数据,获取这些数据的方法包括:

  • getContent
  • getHeaderField
  • getInputStream

对于大部分请求来说,getInputStream和getContent是用的最多的。

相应的信息头用以下方法获取:

  • getContentEncoding
  • getContentLength
  • getContentType
  • getDate
  • getExpiration
  • getLastModifed

HttpURLConnection

任何网络连接都需要经过socket才能连接,HttpURLConnection不需要设置socket,所以,HttpURLConnection并不是底层的连接,而是在底层连接上的一个请求。这就是为什么HttpURLConneciton只是一个抽象类,自身不能被实例化的原因。HttpURLConnection只能通过URL.openConnection()方法创建具体的实例。

虽然底层的网络连接可以被多个HttpURLConnection实例共享,但每一个HttpURLConnection实例只能发送一个请求。请求结束之后,应该调用HttpURLConnection实例的InputStream或OutputStream的close()方法以释放请求的网络资源,不过这种方式对于持久化连接没用。对于持久化连接,得用disconnect()方法关闭底层连接的socket。

1.主函数  MaActivity.class

[java] view plain copy

  1. import java.io.InputStream;
  2. import java.net.HttpURLConnection;
  3. import java.net.URL;
  4. import java.util.ArrayList;
  5. import android.app.Activity;
  6. import android.os.Bundle;
  7. import android.os.Handler;
  8. import android.view.View;
  9. import android.view.View.OnClickListener;
  10. import android.widget.Button;
  11. import android.widget.EditText;
  12. import android.widget.ListView;
  13. import com.bwie.test.Data1.MyData;
  14. import com.google.gson.Gson;
  15. import com.lidroid.xutils.HttpUtils;
  16. import com.lidroid.xutils.exception.HttpException;
  17. import com.lidroid.xutils.http.ResponseInfo;
  18. import com.lidroid.xutils.http.callback.RequestCallBack;
  19. import com.lidroid.xutils.http.client.HttpRequest.HttpMethod;
  20. public class MainActivity extends Activity {
  21. private EditText et_search;
  22. private ListView listView;
  23. private String trim;
  24. protected static final int SUCEESS = 0;
  25. private MyBaseAdapter adapter;
  26. ArrayList<MyData> myList = new ArrayList<MyData>();
  27. Handler handler = new Handler() {
  28. private MyBaseAdapter myBaseAdapter;
  29. public void handleMessage(android.os.Message msg) {
  30. if (msg.what == SUCEESS) {
  31. Data1 chatInfo = (Data1) msg.obj;
  32. // 展示到listView上边 数据成功
  33. myList.addAll(chatInfo.data);
  34. if (myBaseAdapter == null) {
  35. myBaseAdapter = new MyBaseAdapter(MainActivity.this, myList);
  36. listView.setAdapter(myBaseAdapter);
  37. } else {
  38. myBaseAdapter.notifyDataSetChanged();
  39. }
  40. }
  41. };
  42. };
  43. private Button bt_search;
  44. @Override
  45. protected void onCreate(Bundle savedInstanceState) {
  46. super.onCreate(savedInstanceState);
  47. setContentView(R.layout.activity_main);
  48. et_search = (EditText) findViewById(R.id.et_name);
  49. bt_search = (Button) findViewById(R.id.bt_search);
  50. listView = (ListView) findViewById(R.id.listView);
  51. // "http://93.gov.cn/93app/get_search.do?key=九三"
  52. bt_search.setOnClickListener(new OnClickListener() {
  53. @Override
  54. public void onClick(View v) {
  55. // TODO Auto-generated method stub
  56. trim = et_search.getText().toString().trim();
  57. String path = "http://93.gov.cn/93app/get_search.do?key="
  58. + trim;
  59. // 请求数据
  60. getHttp(path);
  61. }
  62. });
  63. }
  64. protected void getHttp(final String path) {
  65. // TODO Auto-generated method stub
  66. new Thread() {
  67. public void run() {
  68. try {
  69. // http://v.juhe.cn/weixin/query?key=您申请的KEY
  70. // 0917c812f187b92e025c9b7c2d4e59b5 pno
  71. URL url = new URL(path);
  72. HttpURLConnection openConnection = (HttpURLConnection) url
  73. .openConnection();
  74. openConnection.setConnectTimeout(5000);
  75. openConnection.setReadTimeout(5000);
  76. int responseCode = openConnection.getResponseCode();
  77. if (responseCode == 200) {
  78. InputStream inputStream = openConnection
  79. .getInputStream();
  80. String parseSteam = StreamUtils.parseSteam(inputStream);
  81. Gson gson = new Gson();
  82. Data1 weiChatInfo = gson.fromJson(parseSteam,
  83. Data1.class);
  84. // 展示到listView上
  85. handler.obtainMessage(SUCEESS, weiChatInfo)
  86. .sendToTarget();
  87. }
  88. } catch (Exception e) {
  89. e.printStackTrace();
  90. }
  91. };
  92. }.start();
  93. }

106.}

**其他的代码在一种复制即可,是通用的**

三.XUtils解析数据 (记得降版本,版本高会报错)

1. 功能介绍

xUtils一个Android公共库框架,主要包括四个部分:View,Db, Http, Bitmap 四个模块。

  • View模块主要的功能是通过注解绑定UI,资源,事件。
  • Db模块是一个数据库orm框架,
    简单的语句就能进行数据的操作。
  • Http模块主要访问网络,支持同步,异步方式的请求,支持文件的下载。
  • Bitmap模块是加载图片以及图片的处理, 支持加载本地,网络图片。而且支持图片的内存和本地缓存。

(具体介绍在下面...)

1.主函数  MaActivity.class

[java] view plain copy

  1. import java.util.ArrayList;
  2. import android.app.Activity;
  3. import android.os.Bundle;
  4. import android.view.View;
  5. import android.view.View.OnClickListener;
  6. import android.widget.Button;
  7. import android.widget.EditText;
  8. import android.widget.ListView;
  9. import com.bwie.test.Data1.MyData;
  10. import com.google.gson.Gson;
  11. import com.lidroid.xutils.HttpUtils;
  12. import com.lidroid.xutils.exception.HttpException;
  13. import com.lidroid.xutils.http.ResponseInfo;
  14. import com.lidroid.xutils.http.callback.RequestCallBack;
  15. import com.lidroid.xutils.http.client.HttpRequest.HttpMethod;
  16. public class MainActivity extends Activity {
  17. private EditText et_search;
  18. private ListView listView;
  19. private String trim;
  20. protected static final int SUCEESS = 0;
  21. private MyBaseAdapter adapter;
  22. ArrayList<MyData> myList = new ArrayList<MyData>();
  23. private Button bt_search;
  24. @Override
  25. protected void onCreate(Bundle savedInstanceState) {
  26. super.onCreate(savedInstanceState);
  27. setContentView(R.layout.activity_main);
  28. et_search = (EditText) findViewById(R.id.et_name);
  29. bt_search = (Button) findViewById(R.id.bt_search);
  30. listView = (ListView) findViewById(R.id.listView);
  31. // "http://93.gov.cn/93app/get_search.do?key=九三"
  32. bt_search.setOnClickListener(new OnClickListener() {
  33. @Override
  34. public void onClick(View v) {
  35. // TODO Auto-generated method stub
  36. trim = et_search.getText().toString().trim();
  37. String path = "http://93.gov.cn/93app/get_search.do?key="
  38. + trim;
  39. // 请求数据
  40. getHttp(path);
  41. }
  42. });
  43. }
  44. protected void getHttp(String path) {
  45. // TODO Auto-generated method stub
  46. HttpUtils utils = new HttpUtils();
  47. utils.send(HttpMethod.GET, path, new RequestCallBack<String>() {
  48. private MyBaseAdapter adapter;
  49. @Override
  50. public void onFailure(HttpException arg0, String arg1) {
  51. // TODO Auto-generated method stub
  52. }
  53. @Override
  54. public void onSuccess(ResponseInfo<String> arg0) {
  55. String result = arg0.result;
  56. Gson gson = new Gson();
  57. Data1 bean = gson.fromJson(result, Data1.class);
  58. ArrayList<MyData> list = new ArrayList<MyData>();
  59. list.addAll(bean.data);
  60. if (adapter == null) {
  61. adapter = new MyBaseAdapter(MainActivity.this, list);
  62. } else {
  63. adapter.notifyDataSetChanged();
  64. }
  65. listView.setAdapter(adapter);
  66. }
  67. });
  68. }
  69. }

最后的配置一定不要忘记啦:

[java] view plain copy

  1. android:name="com.bwie.test.MyApplication"

[java] view plain copy

  1. android:name="android.permission.INTERNET"/>

到这里三个网络请求解析数据的案例就完工啦,下面是XUtils 的详细介绍:

2. 详细设计

2.1 View模块

2.1.1 总体设计

流程和关系较少, 请看下面的详细分析

2.1.2 流程图

2.1.3 核心类功能介绍

请先了解注解 ,动态代理 可以帮助到您, 如果已经了解请忽略。

注解和反射知识是这个模块的主要内容

1.ViewUtils.java

View和各种事件的注入以及资源的注入。

(1)主要函数

[js] view plain copy

  1. private static void injectObject(Object handler, ViewFinder finder)

第一个参数Object handler代表的是需要注入的对象, 第二个参数是需要注入View(这个View就是handler的成员变量)所在的View或者Activity的包装对象。 该方法完成了View和各种事件的注入以及资源的注入。主要的原理就是通过反射和注解。

  • 完成Activity的setContentView。
  • 完成View的注入。
  • 完成资源的注入。
  • 完成各种事件的注入。

2.ViewFinder.java

(1)主要函数

[js] view plain copy

  1. public View findViewById(int id, int pid)
  2. public View findViewById(int id)

如果存在父View, 优先从父View寻找,否则从当前的View或者Activity中寻找。

3.ResLoader.java

[js] view plain copy

  1. public static Object loadRes(ResType type, Context context, int id)

获取资源文件值。支持多种资源的获取。

4.EventListenerManager.java

事件的注入, 其中的设计是通过动态代理。

[js] view plain copy

  1. private final static DoubleKeyValueMap<ViewInjectInfo, Class<?>, Object> listenerCache =            new DoubleKeyValueMap<ViewInjectInfo, Class<?>, Object>();

存放监听事件接口map。 因为有些接口有多个函数, 代理会判断事件接口是否存在, 如果存在只增加代理方法就够了, 避免重新设置监听事件接口。

[js] view plain copy

  1. public static void addEventMethod(
  2. ViewFinder finder,
  3. ViewInjectInfo info,
  4. Annotation eventAnnotation,
  5. Object handler,
  6. Method method)

代理监听事件

5.注解类

2.2 Db模块

2.2.1 总体设计

流程和关系较少, 请看下面的详细分析

2.2.2 流程图

2.2.3 核心类功能介绍

注解、反射和数据库操作知识这个模块的主要内容

1.DbUtils.java

主要功能数据库的创建,数据库的增删改查。

[js] view plain copy

  1. private static HashMap<String, DbUtils> daoMap = new HashMap<String, DbUtils>();

存放DbUtils实例对象的map,每个数据库对应一个实例, key为数据库的名称。

[js] view plain copy

  1. private synchronized static DbUtils getInstance(DaoConfig daoConfig)

采取的是单例模式,根据DaoConfig创建数据库, 中间还涉及到数据库升级。

delete;

findAll;

findById;

saveOrUpdate;// 当数据库没有时保存, 存在时修改。

update;

增删改查。

2.DaoConfig.java

private String dbName =
"xUtils.db"; // default db name数据库名称

private int dbVersion = 1; //数据库版本

private DbUpgradeListener
dbUpgradeListener; //升级监听事件

数据库配置类。

3.FindTempCache.java

在DbUtils的查询数据中

[js] view plain copy

  1. @SuppressWarnings("unchecked")
  2. public <T> List<T> findAll(Selector selector) throws DbException {
  3. ....
  4. String sql = selector.toString();
  5. long seq = CursorUtils.FindCacheSequence.getSeq();
  6. findTempCache.setSeq(seq);
  7. Object obj = findTempCache.get(sql);//优先从缓存读取
  8. if (obj != null) {
  9. return (List<T>) obj;
  10. }
  11. ...
  12. }

数据库查询数据的缓存。在查询中会优先调用缓存中的数据

4.SqlInfoBuilder.java

sql建表、增删改语句的组合。

[js] view plain copy

  1. public static SqlInfo buildCreateTableSqlInfo(DbUtils db, Class<?> entityType)
  2. public static SqlInfo buildDeleteSqlInfo(DbUtils db, Class<?> entityType, Object idValue)
  3. public static SqlInfo buildDeleteSqlInfo(DbUtils db, Class<?> entityType, WhereBuilder whereBuilder)
  4. public static SqlInfo buildDeleteSqlInfo(DbUtils db, Object entity)
  5. public static SqlInfo buildInsertSqlInfo(DbUtils db, Object entity)
  6. public static SqlInfo buildUpdateSqlInfo(DbUtils db, Object entity, String... updateColumnNames)
  1. public static SqlInfo buildUpdateSqlInfo(DbUtils db, Object entity, WhereBuilder whereBuilder, String... updateColumnNames)

5.SqlInfo.java

sql语句和值包装对象。

6.Table.java

表对象。

7.Column.java

表中列对象。

8.Id.java

表对应的主键对象。

9.Selector.java

sql查询语句的组合。

10.WhereBuilder.java

sql条件语句的组合。

2.3 Http模块

2.3.1 总体设计

2.3.2 流程图

2.3.3 类图

1.HttpUtils.java

支持异步同步访问网络数据,
断点下载文件。

[js] view plain copy

  1. //网络数据的缓存。
  2. public final static HttpCache sHttpCache = new HttpCache();
  3. //访问网络的HttpClient。
  4. private final DefaultHttpClient httpClient;
  5. private final HttpContext httpContext = new BasicHttpContext();
  6. //线程池。
  7. private final static PriorityExecutor EXECUTOR = new PriorityExecutor(DEFAULT_POOL_SIZE);

[js] view plain copy

  1. public HttpUtils(int connTimeout, String userAgent) {
  2. //配置超时时间,UserAgent, http版本信息协议等一些信息
  3. .....
  4. //将配置的参数统一放到httpClient中
  5. httpClient = new DefaultHttpClient(new ThreadSafeClientConnManager(params, schemeRegistry), params);
  6. ....
  7. //下面这个关键,设置拦截器。 默认加上gizp压缩。 通过gizp压缩后的数据传输效率高很多。
  8. httpClient.addRequestInterceptor(new HttpRequestInterceptor() {
  9. @Override
  1. public void process(org.apache.http.HttpRequest httpRequest, HttpContext httpContext) throws org.apache.http.HttpException, IOException {
  2. if (!httpRequest.containsHeader(HEADER_ACCEPT_ENCODING)) {
  3. httpRequest.addHeader(HEADER_ACCEPT_ENCODING, ENCODING_GZIP);
  4. }
  5. }
  6. });
  7. httpClient.addResponseInterceptor(new HttpResponseInterceptor() {
  8. @Override
  9. public void process(HttpResponse response, HttpContext httpContext) throws org.apache.http.HttpException, IOException {
  10. final HttpEntity entity = response.getEntity();
  11. if (entity == null) {
  12. return;
  13. }
  14. final Header encoding = entity.getContentEncoding();
  15. if (encoding != null) {
  16. for (HeaderElement element : encoding.getElements()) {
  17. if (element.getName().equalsIgnoreCase("gzip")) {
  18. //这里判断从服务器传输的数据是否需要通过gzip解压。
  19. response.setEntity(new GZipDecompressingEntity(response.getEntity()));
  20. return;
  21. }
  22. }
  23. }
  24. }
  25. });
  26. }

[js] view plain copy

  1. //访问网络数据
  2. private <T> HttpHandler<T> sendRequest(HttpRequest request, RequestParams params, RequestCallBack<T> callBack);
  3. //下载网络文件
  4. public HttpHandler<File> download(HttpRequest.HttpMethod method, String url, String target,
  1. RequestParams params, boolean autoResume, boolean autoRename, RequestCallBack<File> callback);

2.HttpRequest.java

网络请求的包装类。 包括url, 访问请求方法, 参数值等。

3.RequestCallBack.java

完成数据请求回调接口。

4.HttpHandler.java

获取网络数据逻辑的实现。这里可以理解为系统内部AsyncTask。 访问网络数据处理流程图

5.HttpCache.java

网络数据的缓存,内部包含LruMemoryCache。在获取数据的时候会判断是否过期。

6.StringDownLoadHandler.java

handleEntity()将网络io流转化为String。

7.FileDownLoadHandler.java

handleEntity()将网络io流转化为File。

8.HttpException.java

统一异常

2.4 Bitmap模块

2.4.1 总体设计

2.4.2 流程图

请查看http模块

2.4.3 类图

1.BitmapUtils.java

图片的异步加载,支持本地和网络图片,
图片的压缩处理, 图片的内存缓存已经本地缓存。

[js] view plain copy

  1. private BitmapGlobalConfig globalConfig; // 线程池,缓存,和网络的配置
  2. private BitmapDisplayConfig defaultDisplayConfig; //图片显示的配置

[js] view plain copy

  1. /**
  2. * @param container 表示需要显示图片的View
  3. * @param uri 图片的uri
  4. * @param displayConfig 图片显示的配置
  5. * @param callBack 图片加载的回调接口
  6. */
  7. public <T extends View> void display(T container, String uri, BitmapDisplayConfig displayConfig, BitmapLoadCallBack<T> callBack)

设置图片流程图

详细流程图

2.BitmapLoadTask.java

加载图片的异步任务。在doInBackground中读取图片资源

3.BitmapCache.java

[js] view plain copy

  1. private LruDiskCache mDiskLruCache; //闪存缓存
  2. private LruMemoryCache<MemoryCacheKey, Bitmap> mMemoryCache; //运存缓存

(1)主要函数

[js] view plain copy

  1. //下载网络图片, 然后根据配置压缩图片, 将图片缓存。
  2. public Bitmap downloadBitmap(String uri, BitmapDisplayConfig config, final BitmapUtils.BitmapLoadTask<?> task)
  3. //从运存缓存中读取bitmap 在获取的时候会判断是否过期
  4. public Bitmap getBitmapFromMemCache(String uri, BitmapDisplayConfig config)
  5. //从闪存缓存中读取bitmap
  6. public Bitmap getBitmapFromDiskCache(String uri, BitmapDisplayConfig config)

4.BitmapGlobalConfig.java

配置, 包括线程池, 缓存的大小。

[js] view plain copy

  1. //闪存缓存的路径
  2. private String diskCachePath;
  3. //运存缓存的最大值
  4. private int memoryCacheSize = 1024 * 1024 * 4; // 4MB
  5. //闪存缓存的最大值
  6. private int diskCacheSize = 1024 * 1024 * 50;  // 50M
  7. //从网络加载数据的线程池
  8. private final static PriorityExecutor BITMAP_LOAD_EXECUTOR = new PriorityExecutor(DEFAULT_POOL_SIZE);
  9. //从闪存读取数据的线程池
  10. private final static PriorityExecutor DISK_CACHE_EXECUTOR = new PriorityExecutor(2);
  11. //bitmap缓存的的时间
  12. private long defaultCacheExpiry = 1000L * 60 * 60 * 24 * 30; // 30 days
  13. //bitmap缓存
  14. private BitmapCache bitmapCache;

5.BitmapDisplayConfig.java

[js] view plain copy

  1. //图片显示的大小
  2. private BitmapSize bitmapMaxSize;
  3. //图片的动画
  4. private Animation animation;
  5. // 图片加载过程中的显示图片
  6. private Drawable loadingDrawable;
  7. // 图片加载失败的显示图片
  8. private Drawable loadFailedDrawable;
  9. // 图片显示的配置色彩
  10. private Bitmap.Config bitmapConfig = Bitmap.Config.RGB_565;

6.DefaultDownloader.java

获取bitmap, 支持三种获取路径, 本地文件,资产文件, 和网络图片。

7.DefaultBitmapLoadCallBack.java

图片加载完成的的回调, 默认回调将获取的bitmap值传递给view。

3. 杂谈

和Volley框架相比

相同点:

  • 1.采用了网络数据缓存机制。
  • 2.通过handler进行线程通信

不同点:

  • 2.Volley将Http请求数据先缓存进byte[], 然后是分配给不同的请求转化为需要的格式。xUtils是直接转化为想要的格式。 Volley:扩展性好, 但是不能存在大数据请求,否则就OOM。xUtils:不缓存入byte[]
    支持大数据的请求, 速度比Volley稍快,但扩展性就低。
  • 4.Volley访问网络数据时直接开启固定个数线程访问网络, 在run方法中执行死循环, 阻塞等待请求队列。 xUtils是开启线程池来管理线程。
  1. Volley的Http请求在 android 2.3 版本之前是通过HttpClient ,在之后的版本是通过URLHttpConnection。xUtils都是通过HttpClient请求网络(bitmap模块图片下载是通过 URLHttpConnection)。 URLHttpConnection默认支持GZIP压缩,api操作简单。
  1. 缓存失效策略,
    volley的所有网络数据支持从http响应头中控制是否缓存和读取缓存失效时间,每个请求可以控制是否缓存和缓存失效时间。 Xutils网络数据请求是统一自定义缓存失效时间。

Android之三种网络请求解析数据(最佳案例)的更多相关文章

  1. (转载)Android之三种网络请求解析数据&lpar;最佳案例&rpar;

    [置顶] Android之三种网络请求解析数据(最佳案例) 2016-07-25 18:02 4725人阅读 评论(0) 收藏 举报  分类: Gson.Gson解析(1)  版权声明:本文为博主原创 ...

  2. Android 几种网络请求的区别与联系

    HttpUrlConnection 最开始学android的时候用的网络请求是HttpUrlConnection,当时很多东西还不知道,但是在android 2.2及以下版本中HttpUrlConne ...

  3. Android网络请求与数据解析,使用Gson和GsonFormat解析复杂Json数据

    版权声明:未经博主允许不得转载 一:简介 [达叔有道]软件技术人员,时代作者,从 Android 到全栈之路,我相信你也可以!阅读他的文章,会上瘾!You and me, we are family ...

  4. 基于Android Volley的网络请求工具

    基于Android Volley的网络请求工具. 一.说明 AndroidVolley,Android Volley核心库及扩展工程.AndroidVolleySample,网络请求工具示例工程.Re ...

  5. Android进阶&lpar;一&rpar;几种网络请求方式详解

    Ref:http://blog.csdn.net/zuolongsnail/article/details/6373051 Android应用经常会和服务器端交互,这就需要手机客户端发送网络请求,下面 ...

  6. [转]Android各大网络请求库的比较及实战

    自己学习android也有一段时间了,在实际开发中,频繁的接触网络请求,而网络请求的方式很多,最常见的那么几个也就那么几个.本篇文章对常见的网络请求库进行一个总结. HttpUrlConnection ...

  7. Android 各大网络请求库的比较及实战

    自己学习android也有一段时间了,在实际开发中,频繁的接触网络请求,而网络请求的方式很多,最常见的那么几个也就那么几个.本篇文章对常见的网络请求库进行一个总结. HttpUrlConnection ...

  8. android -------- OkGo &lpar;让网络请求更简单的框架&rpar;

    项目地址:https://github.com/jeasonlzy 该库是封装了okhttp的网络框架,可以与RxJava完美结合,比Retrofit更简单易用.支持大文件上传下载,上传进度回调,下载 ...

  9. nginx android app 慢网络请求超时

    最近遇到了android 在慢网络下面请求服务器报 java.net.SocketException: recvfrom failed: ECONNRESET (Connection reset by ...

随机推荐

  1. Eclipse 调试Bug之使用断点的七大技巧

    原文链接:http://xiaohuafyle.iteye.com/blog/1705494 调试竟然有这么多技巧,亏我以前竟不知道...   Eclipse这个开发工具为我们调试bug提供了非常便利 ...

  2. GridView导出Excel(中文乱码)

    public void OUTEXCEL(string items,string where) { DataSet ds = new StudentBLL().GetTable(items,where ...

  3. vim开发环境配置

    一.大饱眼福 看了效果图,肯定有人说, 这都有啥功能?就花哨? 告诉你,你说花哨就错了,开玩笑?我们程序猿可都是实打实的人,说谎都不会,咋会忽悠人呢. 下面我来告诉你,这都有些什么功能: 文件索引功能 ...

  4. (原&plus;译)LUA调用C函数

    转载请注明出处: http://www.cnblogs.com/darkknightzh/p/5804924.html 原始网址: http://www.troubleshooters.com/cod ...

  5. linux下C&plus;&plus; STL hash&lowbar;map的使用以及使用char &ast;型变量作为Key值的一大&OpenCurlyDoubleQuote;坑”

    计算机编程中经常会用到hash表,而在C++中,使用STL编程更是少不了的.本文将介绍STL中hash_map的使用.在hash_map中使用自定义类型作为key值的方法以及在使用char *类型作为 ...

  6. 邮件应用Acompli和日历应用Sunrise(传微软曾考虑以80亿美元收购企业通讯公司Slack)

    http://tech.163.com/16/0305/10/BHCU8EHO000915BD.html http://www.cnbeta.com/articles/480835.htm

  7. Java连接数据库 &num;02&num; JDBC经典套路

    内容索引 LocalConnectionFactory.java LocalConnectionProxy.java ProfileDAO.java-2.0 ProfileDAOImpl.java-2 ...

  8. Python Django性能测试与优化指南

    摘要:本文通过一个简单的实例一步一步引导读者对其进行全方位的性能优化.以下是译文. 唐纳德·克努特(Donald Knuth)曾经说过:“不成熟的优化方案是万恶之源.”然而,任何一个承受高负载的成熟项 ...

  9. sql server 向mysql前移数据-单引号问题

    sql server中的数据导出来 用两个单引号表示一个单引号,这样的格式可以录入到mysql中: 但是遇到特殊的中文字符,例如顿号等,不能正确的显示两个单引号: mysql导出来的数据用反斜线和一个 ...

  10. MySQl中隔离级别和悲观锁乐观锁

    1.MySql的事物支持 MySQL的事务支持不是绑定在MySQL服务器本身,而是与存储引擎相关: MyISAM:不支持事务,用于只读程序提高性能 InnoDB:支持ACID事务.行级锁.并发 Ber ...