Android(java)学习笔记201:网络图片浏览器的实现(ANR)

时间:2021-08-08 11:23:42

1.我们在Android下,实现使用http协议进行网络通信,请求网络数据。这里是获取网络上的图片信息,让它可以显示在手机上;

但是我们这个手机连接网络是很费时间,如果我们在主线程(UI线程)中写这个网络连接的逻辑,这是很容易报一个错误:android.os.NetworkOnMainThreadException(Android4.0之后引入的异常)

主线程很重要,它负责监听系统的各种事件,如果主线程在一段时间内没有响应,系统就会这个应用程序无响应,就会产生ANR的异常。下面有必要说明一下这个ANR异常:

2.ANRs ("Application Not Responding"),意思是"应用没有响应".

在如下情况下,Android会报出ANR错误:
•主线程 ("事件处理线程"/ "UI线程") 在5秒内没有响应输入事件.
• BroadcastReceiver 没有在10秒内完成返回.

通常情况下,下面这些做法会导致ANR:
(1).在主线程内进行网络操作
(2).在主线程内进行一些缓慢的磁盘操作(例如执行没有优化过的SQL查询)

应用应该在5秒或者10秒内响应,否则用户会觉得“这个应用很垃圾”“烂”“慢”…等等

逻辑应该是:
(1). new出一个新的线程,进行数据请求.
(2). 获取数据后,调用handler.sendMessage方法.

(3). 在handler的handle()方法中更新UI.

案例:

activity_main.xml:

 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity" > <Button
android:onClick="click"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerHorizontal="true"
android:layout_centerVertical="true"
android:text="点我" /> </RelativeLayout>

MainActivity.java:

package com.itheima.anr;

import android.app.Activity;
import android.os.Bundle;
import android.view.View; public class MainActivity extends Activity { @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
} public void click(View view){
try {
System.out.println(Thread.currentThread().getName());
Thread.sleep(6000);//当前线程。 } catch (InterruptedException e) {
e.printStackTrace();
}
} }

这里的Thread.sleep(6000),这里的表示Thread线程是Main线程,也就是UI主线程,我们之前说过了UI主线程在5s不能响应,系统就会认为这个应用程序是不响应的。这里6s>5s,所以会报错ANRs ("Application Not Responding")

3.前面说了这么多,也就是说了关于在Android下进行网络连接注意事项,网络连接是耗时程序代码:下面开始实现网络图片浏览器程序:

MainActivity.java:

package com.itheima.netimageviewer;

import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL; import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Bundle;
import android.text.TextUtils;
import android.util.Base64;
import android.view.View;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.Toast; public class MainActivity extends Activity {
private EditText et_path;
private ImageView iv; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
et_path = (EditText) findViewById(R.id.et_path);
iv = (ImageView) findViewById(R.id.iv);
} /**
* 点击查看网络上的图片
*
* @param view
*/
public void click(View view) {
String path = et_path.getText().toString().trim();// http://www.baidu.com/aa.png
if (TextUtils.isEmpty(path)) {
Toast.makeText(this, "图片路径不能为空", 0).show();
return;
}
//把文件名编码之后+缓冲的路径变成file文件
File file = new File(getCacheDir(), Base64.encodeToString(
path.getBytes(), Base64.DEFAULT));
if (file.exists() && file.length() > 0) {
System.out.println("图片存在,拿缓存");
Bitmap bitmap = BitmapFactory.decodeFile(file
.getAbsolutePath());
iv.setImageBitmap(bitmap);
} else {
System.out.println("图片不存在,获取数据生成缓存");
// 通过http请求把图片获取下来。
try {
// 1.声明访问的路径, url 网络资源 http ftp rtsp
URL url = new URL(path);
// 2.通过路径得到一个连接 http的连接
HttpURLConnection conn = (HttpURLConnection) url
.openConnection();
// 3.判断服务器给我们返回的状态信息。
// 200 成功 302 重定向 404资源没找到 5xx 服务器内部错误
int code = conn.getResponseCode();
if (code == 200) {
//这个流是用来接收图片的
InputStream is = conn.getInputStream();// png的图片
//这个输出流是吧图片写入安卓系统的储存区的
FileOutputStream fos = new FileOutputStream(file);
//读写操作
byte[] buffer = new byte[1024];
int len = -1;
while ((len = is.read(buffer)) != -1) {
fos.write(buffer, 0, len);
}
is.close();
fos.close();
//通过图片工厂来获取文件路径, 然后变成图片传递给控件
Bitmap bitmap = BitmapFactory.decodeFile(file
.getAbsolutePath());
iv.setImageBitmap(bitmap);
} else {
// 请求失败
Toast.makeText(this, "请求失败", 0).show();
}
} catch (Exception e) {
e.printStackTrace();
Toast.makeText(this, "发生异常,请求失败", 0).show();
}
}
}
}

activity_main.xml:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity" > <EditText
android:text="http://www.baidu.com/img/bd_logo1.png"
android:id="@+id/et_path"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="请输入网络图片的路径" /> <Button
android:onClick="click"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="查看" /> <ImageView
android:id="@+id/iv"
android:layout_width="wrap_content"
android:layout_height="wrap_content" /> </LinearLayout>

AndroidManifest.xml:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.himi.webpicturewatch"
android:versionCode="1"
android:versionName="1.0" >
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/> <uses-sdk
android:minSdkVersion="15"
android:targetSdkVersion="17" /> <application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name=".MainActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application> </manifest>