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

论坛 期权论坛 脚本     
匿名技术用户   2021-1-5 23:16   324   0

虽然下面的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任务
 }

}


分享到 :
0 人收藏
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

积分:7942463
帖子:1588486
精华:0
期权论坛 期权论坛
发布
内容

下载期权论坛手机APP