HttpCient:
HttpClient是Apache开源组织提供的HTTP网络访问接口(一个开源的项目),从名字上就可以看出,它是一个简单的HTTP客户端(并不是浏览器),可以发送HTTP请求,接受HTTP响应。但是不会缓存服务器的响应,不能执行HTTP页面中签入嵌入的JS代码,自然也不会对页面内容进行任何解析、处理,这些都是需要开发人员来完成的。
现在Android已经成功集成了HttpClient,所以开发人员在Android项目中可以直接使用HttpClient来想Web站点提交请求以及接受响应,如果使用其他的Java项目,需要引入进相应的Jar包。HttpClient可以在官网上下载。官网链接:http://hc.apache.org/downloads.cgi
HttpClient其实是一个interface类型,HttpClient封装了对象需要执行的Http请求、身份验证、连接管理和其它特性。既然HttpClient是一个接口,因此无法创建它的实例。从文档上看,HttpClient有三个已知的实现类分别是:AbstractHttpClient, AndroidHttpClient, DefaultHttpClient,会发现有一个专门为Android应用准备的实现类AndroidHttpClient,当然使用常规的DefaultHttpClient也可以实现功能。
从两个类包所有在位置就可以看出区别,AndroidHttpClient定义在android.net.http.AndroidHttpClient包下,属于Android原生的http访问,而DefaultHttpClient定义在org.apache.http.impl.client.DefaultHttpClient包下,属于对apche项目的支持。而AndroidHttpClient没有公开的构造函数,只能通过静态方法newInstance()方法来获得AndroidHttpClient对象。
简单来说,用HttpClient发送请求、接收响应都很简单,只需要五大步骤即可:(要牢记)
- 创建代表客户端的HttpClient对象。
- 创建代表请求的对象,如果需要发送GET请求,则创建HttpGet对象,如果需要发送POST请求,则创建HttpPost对象。注:对于发送请求的参数,GET和POST使用的方式不同,GET方式可以使用拼接字符串的方式,把参数拼接在URL结尾;POST方式需要使用setEntity(HttpEntity entity)方法来设置请求参数。
- 调用HttpClient对象的execute(HttpUriRequest request)发送请求,执行该方法后,将获得服务器返回的HttpResponse对象。服务器发还给我们的数据就在这个HttpResponse相应当中。调用HttpResponse的对应方法获取服务器的响应头、响应内容等。
- 检查相应状态是否正常。服务器发给客户端的相应,有一个相应码:相应码为200,正常;相应码为404,客户端错误;相应码为505,服务器端错误。
- 获得相应对象当中的数据
DefaultHttpClient:
根据上面的五大步骤,我们就拿DefaultHttpClient举例来访问网络。注意这里有一条原则:在主线程中不能访问网络(在android4.0之后的加入的概念)。
package com.example.myapplication; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.widget.ListView; public class MainActivity extends AppCompatActivity { private ListView lv; String path="http://api.tianapi.com/wxnew/?key=8d6e3228d25298f13af4fc40ce6c9679&num=10"; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); init(); initgetData(path); } private void init(){ lv = (ListView) findViewById(R.id.lv); } private void initgetData(String path){ MyTask task = new MyTask(MainActivity.this, path, "GET", lv); task.execute(path); } }
网络请求异步加载
package com.example.myapplication; import android.content.Context; import android.os.AsyncTask; import android.widget.ListView; import com.google.gson.Gson; import org.apache.http.HttpEntity; import org.apache.http.HttpResponse; import org.apache.http.client.HttpClient; import org.apache.http.client.methods.HttpGet; import org.apache.http.impl.client.DefaultHttpClient; import org.apache.http.util.EntityUtils; import java.io.IOException; import java.util.List; /** * 网络请求 异步加载 * Created by Wangrx on 2017/9/6. */ class MyTask extends AsyncTask<String, Integer, String> { private Context context; private String path; private String get; private ListView lv; private List<JavaBean.NewslistBean> list; public MyTask(MainActivity mainActivity, String path, String get, ListView lv) { this.context = mainActivity; this.path = path; this.get = get; this.lv = lv; } @Override protected String doInBackground(String... strings) { HttpClient httpClient = new DefaultHttpClient(); HttpGet httpGet = new HttpGet(strings[0]); try { HttpResponse httpResponse = httpClient.execute(httpGet); if (httpResponse.getStatusLine().getStatusCode() == 200) { HttpEntity entity = httpResponse.getEntity(); String s = EntityUtils.toString(entity, "utf-8"); return s; } } catch (IOException e) { e.printStackTrace(); } return null; } @Override protected void onPostExecute(String s) { super.onPostExecute(s); //获得Gson对象 Gson gson = new Gson(); //将json串 转化成bean JavaBean javaBean = gson.fromJson(s, JavaBean.class); //得到Newslist list = javaBean.getNewslist(); //设置适配器 MyAdapter myAdapter = new MyAdapter(list, context); lv.setAdapter(myAdapter); } }设置适配器
package com.example.myapplication; import android.content.Context; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.ImageView; import android.widget.TextView; import com.nostra13.universalimageloader.core.DisplayImageOptions; import com.nostra13.universalimageloader.core.ImageLoader; import com.nostra13.universalimageloader.core.display.RoundedBitmapDisplayer; import java.util.List; /** * //适配器 * Created by Wangrx on 2017/9/6. */ class MyAdapter extends BaseAdapter{ private List<JavaBean.NewslistBean> list; private Context context; public MyAdapter(List<JavaBean.NewslistBean> list, Context context) { this.list = list; this.context = context; } @Override public int getCount() { return list.size(); } @Override public Object getItem(int i) { return list.get(i); } @Override public long getItemId(int i) { return i; } @Override public View getView(int i, View view, ViewGroup viewGroup) { ViewHolder holder = null; if (view==null){ view = View.inflate(context, R.layout.item, null); holder = new ViewHolder(); holder.text_title = view.findViewById(R.id.text_title); holder.text_time = view.findViewById(R.id.text_time); holder.text_author = view.findViewById(R.id.text_author); holder.img_01 = view.findViewById(R.id.img_01); holder.img_02 = view.findViewById(R.id.img_02); holder.img_03 = view.findViewById(R.id.img_03); view.setTag(holder); }else { holder = (ViewHolder) view.getTag(); } holder.text_title.setText(list.get(i).getTitle()); holder.text_time.setText(list.get(i).getCtime()); holder.text_author.setText(list.get(i).getDescription()); //图片的缓存 DisplayImageOptions options = new DisplayImageOptions.Builder() .showImageOnLoading(R.mipmap.img_in) //加载图片时的图片 .showImageForEmptyUri(R.mipmap.img_er) //没有图片资源时的默认图片 .showImageOnFail(R.mipmap.img_on) //加载失败时的图片 .cacheInMemory(true) //启用内存缓存 .cacheOnDisk(true) //启用外存缓存 .considerExifParams(true) //启用EXIF和JPEG图像格式 .displayer(new RoundedBitmapDisplayer(20)) //设置显示风格这里是圆角矩形 .build(); //得到图片地址 并赋值 ImageLoader.getInstance().displayImage(list.get(i).getPicUrl().toString(), holder.img_01, options); ImageLoader.getInstance().displayImage(list.get(i).getPicUrl().toString(), holder.img_02, options); ImageLoader.getInstance().displayImage(list.get(i).getPicUrl().toString(), holder.img_03, options); return view; } class ViewHolder{ TextView text_title; TextView text_time; TextView text_author; ImageView img_01; ImageView img_02; ImageView img_03; } }图片缓存
package com.example.myapplication; import android.app.Application; import android.content.Context; import android.content.res.Configuration; import com.nostra13.universalimageloader.cache.disc.impl.UnlimitedDiskCache; import com.nostra13.universalimageloader.cache.disc.naming.Md5FileNameGenerator; import com.nostra13.universalimageloader.core.ImageLoader; import com.nostra13.universalimageloader.core.ImageLoaderConfiguration; import com.nostra13.universalimageloader.core.assist.QueueProcessingType; import java.io.File; /** * Created by Wangrx on 2017/9/6. * 把图片缓存到本地磁盘,拿到图片,写到SD卡中 */ public class MyApplic extends Application{ @Override public void onCreate() { super.onCreate(); //初始化IamgeLoader //获取sd卡根目录路径 File files = new File("/sdcard/Rimg"); initImageLoader(getApplicationContext(),files); } public static void initImageLoader(Context context, File file) { ImageLoaderConfiguration.Builder config = new ImageLoaderConfiguration.Builder(context); config.threadPriority(Thread.NORM_PRIORITY - 2); config.denyCacheImageMultipleSizesInMemory(); config.diskCacheFileNameGenerator(new Md5FileNameGenerator()); config.diskCacheSize(50 * 1024 * 1024); // 50 MiB config.tasksProcessingOrder(QueueProcessingType.LIFO); config.writeDebugLogs(); // Remove for release app config .diskCache(new UnlimitedDiskCache(file));//UnlimitedDiskCache 限制这个图片的缓存路径 config .diskCacheFileCount(50);//配置sdcard缓存文件的数量 // Initialize ImageLoader with configuration. ImageLoader.getInstance().init(config.build()); } @Override public void onConfigurationChanged(Configuration newConfig) { super.onConfigurationChanged(newConfig); } @Override public void onLowMemory() { super.onLowMemory(); } }权限
<!-- 网络权限--> <uses-permission android:name="android.permission.INTERNET"/> <!-- 写入权限--> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> <!-- sd创建删除文件--> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"></uses-permission>引入:
布局文件
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.example.myapplication.MainActivity"> <ListView android:id="@+id/lv" android:layout_width="match_parent" android:layout_height="match_parent"></ListView> </RelativeLayout>
item
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <TextView android:id="@+id/text_title" android:layout_width="match_parent" android:layout_height="wrap_content" /> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <ImageView android:id="@+id/img_01" android:layout_width="0dp" android:layout_height="100dp" android:layout_weight="1" /> <ImageView android:id="@+id/img_02" android:layout_width="0dp" android:layout_height="100dp" android:layout_weight="1" /> <ImageView android:id="@+id/img_03" android:layout_width="0dp" android:layout_height="100dp" android:layout_weight="1" /> </LinearLayout> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:id="@+id/text_author" android:layout_width="wrap_content" android:layout_height="wrap_content" /> <TextView android:id="@+id/text_time" android:layout_width="wrap_content" android:layout_height="wrap_content" /> </LinearLayout> </LinearLayout>注意:
Android Studio
HttpClient的jar包 可能会与ImageLoader的jar包起冲突 解决方案参考如下: