【Android】读取sdcard卡上的所有图片并且显示,读取的过程有进度条显示

虽然下面的app还没有做到快图浏览、ES文件浏览器的水平,遇到大sdcard还是会存在读取过久、内存溢出等问题,但是基本思想是这样的。

如下图,在sdcard卡上有4张图片,


打开app,则会吧sd卡上的所有图片读取,并显示出来,读取的过程有进度条显示。


制作过程如下:

1、首先,res\values\strings.xml对字符设置如下,没有什么特别的。

<?xml version="1.0" encoding="utf-8"?>
<resources>

    <string name="app_name">读取sdcard下的所有图片</string>
    <string name="action_settings">Settings</string>
    <string name="imageView_description">大图</string>  

</resources>
2、之后,修改res\layout\activity_main.xml形成布局,这里采用布局与《【Android】画廊式的图片浏览器,使用HorizontalScrollView取代Gallery,OnClickListener的参数传递》( 点击打开链接)是一模一样的,只是这里,HorizontalScrollView旗下的LinearLayout改为用Java代码生成,故对HorizontalScrollView赋予id。

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <ImageView
        android:id="@+id/imageView1"
        android:layout_width="match_parent"
        android:layout_height="200dp"
        android:contentDescription="@string/imageView_description"
        android:paddingTop="10dp" />

    <HorizontalScrollView
        android:id="@+id/horizontalScrollView1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="center_vertical"
        android:scrollbars="horizontal" >
    </HorizontalScrollView>

</LinearLayout>

3、然后,与《【Android】读取sdcard上的图片》(点击打开链接)一样,在AndroidManifest.xml插入<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />与<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />要求对sdcard读取与写入的权限,这个文件修改之后如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.asyncread"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="18" />

    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <!-- 要求向SDCard读取数据权限 -->
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> <!-- 要求向SDCard写入数据权限 -->

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="com.example.asyncread.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>
4、最后,才是整个程序的关键,对MainActivity.java的编写。这里首先插入了两个工具类,一个在《【Java】读取其下所有文件夹与文件的路径》( 点击打开链接)中讲过,用于遍历sdcard上面的文件的,当然,遍历的时候剔除sdcard上Android这个系统文件夹,不剔除的话会把系统的图标等遍历出来。之后是一个图片缩放类,不对读取到的图片进行缩放,会在app运行时,出现内存溢出的错误。

程序一开始,在onCreate方法,在app头部生成一个进度条,之后调用异步类AsyncTask对sdcard卡文件的读取,与《【Android】画廊式的图片浏览器,使用HorizontalScrollView取代Gallery,OnClickListener的参数传递》(点击打开链接)中的方法一样,把图片资源一张张加载出来。只是作用对象从app的图片资源,变成从sdcard读取出来的图片。

package com.example.asyncread;

import java.io.File;
import java.util.ArrayList;

import android.os.AsyncTask;
import android.os.Bundle;
import android.os.Environment;
import android.view.Gravity;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.view.Window;
import android.widget.HorizontalScrollView;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.LinearLayout.LayoutParams;
import android.widget.TextView;
import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;

public class MainActivity extends Activity {
	// 安卓组件
	private HorizontalScrollView horizontalScrollView1;
	private ImageView imageView1;
	// 用于存放sdcard卡上的所有图片路径
	public static ArrayList<String> dirAllStrArr = new ArrayList<String>();

	// 用于遍历sdcard卡上所有文件的类
	public static void DirAll(File dirFile) throws Exception {
		if (dirFile.exists()) {
			File files[] = dirFile.listFiles();
			for (File file : files) {
				if (file.isDirectory()) {
					String fileName = file.getName();
					// 除sdcard上Android这个文件夹以外。
					if (!fileName.endsWith("Android")) {
						// 如果遇到文件夹则递归调用。
						DirAll(file);
					}
				} else {
					// 如果是图片文件压入数组
					String fileName = file.getName();
					if (fileName.endsWith(".jpg") || fileName.endsWith(".jpeg")
							|| fileName.endsWith(".bmp")
							|| fileName.endsWith(".gif")
							|| fileName.endsWith(".png")) {
						// 如果遇到文件则放入数组
						if (dirFile.getPath().endsWith(File.separator)) {
							dirAllStrArr
									.add(dirFile.getPath() + file.getName());
						} else {
							dirAllStrArr.add(dirFile.getPath() + File.separator
									+ file.getName());
						}
					}
				}
			}
		}
	}

	// 图片加载的类
	public static BitmapFactory.Options getHeapOpts(File file) {
		BitmapFactory.Options opts = new BitmapFactory.Options();
		// 数字越大读出的图片占用的heap必须越小,不然总是溢出
		if (file.length() < 20480) { // 0-20k
			opts.inSampleSize = 1;// 这里意为缩放的大小
		} else if (file.length() < 51200) { // 20-50k
			opts.inSampleSize = 2;
		} else if (file.length() < 307200) { // 50-300k
			opts.inSampleSize = 4;
		} else if (file.length() < 819200) { // 300-800k
			opts.inSampleSize = 6;
		} else if (file.length() < 1048576) { // 800-1024k
			opts.inSampleSize = 8;
		} else {
			opts.inSampleSize = 10;
		}
		return opts;
	}

	class MyTask extends AsyncTask<Void, Integer, LinearLayout> {
		@Override
		// 任务开始前做什么
		protected void onPreExecute() {
			setProgressBarVisibility(true);// 显示进度条
			super.onPreExecute();
		}

		@Override
		protected LinearLayout doInBackground(Void... arg0) {
			// 生成一个水平线性布局
			LinearLayout linearLayout1 = new LinearLayout(MainActivity.this);
			LayoutParams layoutParams = new LayoutParams(
					ViewGroup.LayoutParams.WRAP_CONTENT,
					ViewGroup.LayoutParams.WRAP_CONTENT);
			linearLayout1.setOrientation(LinearLayout.HORIZONTAL);
			linearLayout1.setLayoutParams(layoutParams);// 应用到新生成的线性布局
			/* 遍历sdcard旗下的所有文件夹开始 */
			String sdpath = Environment.getExternalStorageDirectory()
					.getAbsolutePath();// 获取sdcard的根路径
			File dirFile = new File(sdpath);
			try {
				DirAll(dirFile);
			} catch (Exception e) {
				e.printStackTrace();
			}
			/* 遍历sdcard旗下的所有文件夹结束 */
			// 得到sdcard旗下的所有图片路径之后,再对这个数组进行遍历
			for (int i = 0; i < dirAllStrArr.size(); i++) {
				String filePath = dirAllStrArr.get(i);
				File file = new File(filePath);
				Bitmap bm = BitmapFactory.decodeFile(filePath,
						getHeapOpts(file));
				LinearLayout linearLayout2 = new LinearLayout(MainActivity.this);
				// 设置新生成线性布局的参数,宽度为100,高度为匹配父组件,也就是水平滚动视图的高度
				LayoutParams layoutParams1 = new LayoutParams(100,
						ViewGroup.LayoutParams.MATCH_PARENT);
				layoutParams1.gravity = Gravity.CENTER_HORIZONTAL;// 设置线性布局内的组件水平居中
				linearLayout2.setOrientation(LinearLayout.VERTICAL);// 设置新生成的线性布局android:orientation="vertical"
				linearLayout2.setLayoutParams(layoutParams1);// 应用到新生成的线性布局
				ImageView imageView2 = new ImageView(MainActivity.this);
				imageView2.setId(i + 20000);// 这里由于id不能为字符的缘故,所有对图片的id分别设为20000,20001,20002,...便于下面的图片点击监听器所控制
				imageView2.setImageBitmap(bm);// 将数组中的第i张图片放到图片视图
				imageView2.setAdjustViewBounds(true);// 自动缩放为宽高比
				imageView2.setMaxHeight(80);// 图片的高度为80dp
				imageView2.setPadding(10, 10, 10, 10);
				imageView2.setOnClickListener(new OnClickListener() {
					@Override
					public void onClick(View view) {
						String filePath = dirAllStrArr.get(view.getId() - 20000);
						File file = new File(filePath);
						Bitmap bm = BitmapFactory.decodeFile(filePath,
								getHeapOpts(file));
						imageView1.setImageBitmap(bm);
						// 把点击的图片id取出之后,减20000就是要显示的图片在图片数组的位置了。
					}
				});
				// 将图片视图加载到新生成的线性布局之中
				linearLayout2.addView(imageView2);
				// 新生成一个标签文本
				TextView textView = new TextView(MainActivity.this);
				textView.setText(filePath);
				textView.setTextSize(15);
				// 标签文本在水平位置居中
				textView.setGravity(Gravity.CENTER);
				// 添加到新生成的线性布局之后
				linearLayout2.addView(textView);
				linearLayout1.addView(linearLayout2);
				publishProgress(i);// 调用onProgressUpdate方法
			}

			return linearLayout1;
		}

		@Override
		protected void onProgressUpdate(Integer... values) {
			setProgress(values[0] * 2500);// 更新进度条
			super.onProgressUpdate(values);
		}

		// 任务放完之后
		@Override
		protected void onPostExecute(LinearLayout result) {
			setProgressBarVisibility(false);
			horizontalScrollView1.addView(result);// 把doInBackground()最终搞出来的水平线性布局放到横向滚动布局中
			super.onPostExecute(result);
		}

	}

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		requestWindowFeature(Window.FEATURE_PROGRESS);// 在头部生成一个进度条,必须在setContentView(R.layout.activity_main);这条语句之前
		setContentView(R.layout.activity_main);
		// 获取各种组件
		horizontalScrollView1 = (HorizontalScrollView) findViewById(R.id.horizontalScrollView1);
		imageView1 = (ImageView) findViewById(R.id.imageView1);
		new MyTask().execute();// 开始MyTask任务
	}

}


评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值