转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/39943731,本文出自:【张鸿洋的博客】

1、概述

关于手机图片载入器,在当今像素随随便便破千万的时代。一张图片占领的内存都相当可观,作为高大尚程序员的我们。有必要掌握图片的压缩,缓存等处理,以到达纵使你有万张照片。纵使你的像素再高,我们也能正确的显示全部的图片。当然了,单纯显示图片没撒意思,我们决定高仿一下微信的图片选择器,在此,感谢微信!本篇博客将基于以下两篇博客:

Android 高速开发系列 打造万能的ListView GridView 适配器  将使用我们打造的CommonAdapter作为我们样例中GridView以及ListView的适配器

Android Handler 异步消息处理机制的妙用 创建强大的图片载入类 将使用我们自己写的ImageLoader作为我们的图片载入的核心类

假设你没看过也没关系,等看完本篇博客,能够结合以上两篇再进行充分理解一下。

好了。首先贴一下效果图:

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbG1qNjIzNTY1Nzkx/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="" />

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbG1qNjIzNTY1Nzkx/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt="" />

动态图实在是录不出来,大家自己打开微信点击发表图片,或者聊天窗体发送图片,大致和微信的效果一样~

简单描写叙述一下:

1、默认显示图片最多的目录图片,以及底部显示图片总数量。如上图1;

2、点击底部,弹出popupWindow,popupWindow包括全部含有图片的目录,以及显示每一个目录中图片数量。如上图2。注:此时Activity变暗

3、选择不论什么目录,进入该目录图片显示,能够点击选择图片,当然了,点击已选择的图片则会取消选择。如上图3。注:选中图片变暗

当然了,最重要的效果一定流畅,不能动不动OOM~~

本人測试手机小米2s,图片6802张,未出现OOM异常,效果也是非常流畅,堪比图库~

只是存在bug在所难免。大家能够留言说下自己发现的bug;文末会提供源代码下载。

好了,以下就能够代码的征程了~

2、图片的列表页

首先对手机中图片进行扫描,拿到图片数量最多的,直接显示在GridView上。而且扫描结束。得到一个全部包括图片的目录信息的List;

对于目录信息,我们单独创建了一个Bean:

  1. package com.zhy.bean;
  2.  
  3. public class ImageFloder
  4. {
  5. /**
  6. * 图片的目录路径
  7. */
  8. private String dir;
  9.  
  10. /**
  11. * 第一张图片的路径
  12. */
  13. private String firstImagePath;
  14.  
  15. /**
  16. * 目录的名称
  17. */
  18. private String name;
  19.  
  20. /**
  21. * 图片的数量
  22. */
  23. private int count;
  24.  
  25. public String getDir()
  26. {
  27. return dir;
  28. }
  29.  
  30. public void setDir(String dir)
  31. {
  32. this.dir = dir;
  33. int lastIndexOf = this.dir.lastIndexOf("/");
  34. this.name = this.dir.substring(lastIndexOf);
  35. }
  36.  
  37. public String getFirstImagePath()
  38. {
  39. return firstImagePath;
  40. }
  41.  
  42. public void setFirstImagePath(String firstImagePath)
  43. {
  44. this.firstImagePath = firstImagePath;
  45. }
  46.  
  47. public String getName()
  48. {
  49. return name;
  50. }
  51. public int getCount()
  52. {
  53. return count;
  54. }
  55.  
  56. public void setCount(int count)
  57. {
  58. this.count = count;
  59. }
  60.  
  61. }

用来存储当前目录的路径。当前目录包括多少张图片,以及第一张图片路径用于做目录的图标;注:目录的名称,我们在set目录的路径的时候。自己主动提取,细致看下setDir这种方法。

接下来就是扫描手机图片的代码了:

  1. @Override
  2. protected void onCreate(Bundle savedInstanceState)
  3. {
  4. super.onCreate(savedInstanceState);
  5. setContentView(R.layout.activity_main);
  6.  
  7. DisplayMetrics outMetrics = new DisplayMetrics();
  8. getWindowManager().getDefaultDisplay().getMetrics(outMetrics);
  9. mScreenHeight = outMetrics.heightPixels;
  10.  
  11. initView();
  12. getImages();
  13. initEvent();
  14.  
  15. }
  16.  
  17. /**
  18. * 利用ContentProvider扫描手机中的图片,此方法在执行在子线程中 完毕图片的扫描。终于获得jpg最多的那个目录
  19. */
  20. private void getImages()
  21. {
  22. if (!Environment.getExternalStorageState().equals(
  23. Environment.MEDIA_MOUNTED))
  24. {
  25. Toast.makeText(this, "暂无外部存储", Toast.LENGTH_SHORT).show();
  26. return;
  27. }
  28. // 显示运行进度条
  29. mProgressDialog = ProgressDialog.show(this, null, "正在载入...");
  30.  
  31. new Thread(new Runnable()
  32. {
  33. @Override
  34. public void run()
  35. {
  36.  
  37. String firstImage = null;
  38.  
  39. Uri mImageUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
  40. ContentResolver mContentResolver = MainActivity.this
  41. .getContentResolver();
  42.  
  43. // 仅仅查询jpeg和png的图片
  44. Cursor mCursor = mContentResolver.query(mImageUri, null,
  45. MediaStore.Images.Media.MIME_TYPE + "=?
  46.  
  47. or "
  48. + MediaStore.Images.Media.MIME_TYPE + "=?
  49.  
  50. ",
  51. new String[] { "image/jpeg", "image/png" },
  52. MediaStore.Images.Media.DATE_MODIFIED);
  53.  
  54. Log.e("TAG", mCursor.getCount() + "");
  55. while (mCursor.moveToNext())
  56. {
  57. // 获取图片的路径
  58. String path = mCursor.getString(mCursor
  59. .getColumnIndex(MediaStore.Images.Media.DATA));
  60.  
  61. Log.e("TAG", path);
  62. // 拿到第一张图片的路径
  63. if (firstImage == null)
  64. firstImage = path;
  65. // 获取该图片的父路径名
  66. File parentFile = new File(path).getParentFile();
  67. if (parentFile == null)
  68. continue;
  69. String dirPath = parentFile.getAbsolutePath();
  70. ImageFloder imageFloder = null;
  71. // 利用一个HashSet防止多次扫描同一个目录(不加这个推断,图片多起来还是相当恐怖的~~)
  72. if (mDirPaths.contains(dirPath))
  73. {
  74. continue;
  75. } else
  76. {
  77. mDirPaths.add(dirPath);
  78. // 初始化imageFloder
  79. imageFloder = new ImageFloder();
  80. imageFloder.setDir(dirPath);
  81. imageFloder.setFirstImagePath(path);
  82. }
  83.  
  84. int picSize = parentFile.list(new FilenameFilter()
  85. {
  86. @Override
  87. public boolean accept(File dir, String filename)
  88. {
  89. if (filename.endsWith(".jpg")
  90. || filename.endsWith(".png")
  91. || filename.endsWith(".jpeg"))
  92. return true;
  93. return false;
  94. }
  95. }).length;
  96. totalCount += picSize;
  97.  
  98. imageFloder.setCount(picSize);
  99. mImageFloders.add(imageFloder);
  100.  
  101. if (picSize > mPicsSize)
  102. {
  103. mPicsSize = picSize;
  104. mImgDir = parentFile;
  105. }
  106. }
  107. mCursor.close();
  108.  
  109. // 扫描完毕。辅助的HashSet也就能够释放内存了
  110. mDirPaths = null;
  111.  
  112. // 通知Handler扫描图片完毕
  113. mHandler.sendEmptyMessage(0x110);
  114.  
  115. }
  116. }).start();
  117.  
  118. }

ps:执行出现空指针的话,在81行的位置加入推断,if(parentFile.list()==null)continue , 切记~~~有些图片比較诡异~~;

initView就不看了。都是些findViewById;

getImages主要就是扫描图片的代码,我们开启了一个Thread进行扫描,扫描完毕以后。我们得到了图片最多目录路径(mImgDir),手机中图片数量(totalCount);以及全部包括图片目录信息(mImageFloders)

然后我们通过handler发送消息,在handleMessage里面:

1、创建GridView的适配器,为我们的GridView设置适配器,显示图片;

2、有了mImageFloders。就能够创建我们的popupWindow了

看一眼我们的Handler

  1. private Handler mHandler = new Handler()
  2. {
  3. public void handleMessage(android.os.Message msg)
  4. {
  5. mProgressDialog.dismiss();
  6. //为View绑定数据
  7. data2View();
  8. //初始化展示目录的popupWindw
  9. initListDirPopupWindw();
  10. }
  11. };

能够看到分别干了上述的两件事:

  1. /**
  2. * 为View绑定数据
  3. */
  4. private void data2View()
  5. {
  6. if (mImgDir == null)
  7. {
  8. Toast.makeText(getApplicationContext(), "擦,一张图片没扫描到",
  9. Toast.LENGTH_SHORT).show();
  10. return;
  11. }
  12.  
  13. mImgs = Arrays.asList(mImgDir.list());
  14. /**
  15. * 能够看到目录的路径和图片的路径分开保存,极大的降低了内存的消耗;
  16. */
  17. mAdapter = new MyAdapter(getApplicationContext(), mImgs,
  18. R.layout.grid_item, mImgDir.getAbsolutePath());
  19. mGirdView.setAdapter(mAdapter);
  20. mImageCount.setText(totalCount + "张");
  21. };

data2View就是我们当前Activity上全部的View设置数据了。

看到这里还用到了一个Adapter。我们GridView的:

  1. package com.zhy.imageloader;
  2.  
  3. import java.util.LinkedList;
  4. import java.util.List;
  5.  
  6. import android.content.Context;
  7. import android.graphics.Color;
  8. import android.view.View;
  9. import android.view.View.OnClickListener;
  10. import android.widget.ImageView;
  11.  
  12. import com.zhy.utils.CommonAdapter;
  13.  
  14. public class MyAdapter extends CommonAdapter<String>
  15. {
  16.  
  17. /**
  18. * 用户选择的图片。存储为图片的完整路径
  19. */
  20. public static List<String> mSelectedImage = new LinkedList<String>();
  21.  
  22. /**
  23. * 目录路径
  24. */
  25. private String mDirPath;
  26.  
  27. public MyAdapter(Context context, List<String> mDatas, int itemLayoutId,
  28. String dirPath)
  29. {
  30. super(context, mDatas, itemLayoutId);
  31. this.mDirPath = dirPath;
  32. }
  33.  
  34. @Override
  35. public void convert(final com.zhy.utils.ViewHolder helper, final String item)
  36. {
  37. // 设置no_pic
  38. helper.setImageResource(R.id.id_item_image, R.drawable.pictures_no);
  39. // 设置no_selected
  40. helper.setImageResource(R.id.id_item_select,
  41. R.drawable.picture_unselected);
  42. // 设置图片
  43. helper.setImageByUrl(R.id.id_item_image, mDirPath + "/" + item);
  44.  
  45. final ImageView mImageView = helper.getView(R.id.id_item_image);
  46. final ImageView mSelect = helper.getView(R.id.id_item_select);
  47.  
  48. mImageView.setColorFilter(null);
  49. // 设置ImageView的点击事件
  50. mImageView.setOnClickListener(new OnClickListener()
  51. {
  52. // 选择,则将图片变暗。反之则反之
  53. @Override
  54. public void onClick(View v)
  55. {
  56.  
  57. // 已经选择过该图片
  58. if (mSelectedImage.contains(mDirPath + "/" + item))
  59. {
  60. mSelectedImage.remove(mDirPath + "/" + item);
  61. mSelect.setImageResource(R.drawable.picture_unselected);
  62. mImageView.setColorFilter(null);
  63. } else
  64. // 未选择该图片
  65. {
  66. mSelectedImage.add(mDirPath + "/" + item);
  67. mSelect.setImageResource(R.drawable.pictures_selected);
  68. mImageView.setColorFilter(Color.parseColor("#77000000"));
  69. }
  70.  
  71. }
  72. });
  73.  
  74. /**
  75. * 已经选择过的图片。显示出选择过的效果
  76. */
  77. if (mSelectedImage.contains(mDirPath + "/" + item))
  78. {
  79. mSelect.setImageResource(R.drawable.pictures_selected);
  80. mImageView.setColorFilter(Color.parseColor("#77000000"));
  81. }
  82.  
  83. }
  84. }

能够看到我们GridView的Adapter继承了我们的CommonAdapter。假设不知道CommonAdapter为何物,能够去看看万能适配器那篇博文;

我们如今仅仅须要实现convert方法:

在convert中,我们设置图片。设置事件等,对于图片的变暗。我们使用的是ImageView的setColorFilter ;依据Url载入图片的操作封装在helper.setImageByUrl(view,url)中,内部使用的是我们自定义的ImageLoader,包括错乱处理都已经封装了。图片策略我们使用的是LIFO后进先出;不清楚的能够看文章一開始说明的那两篇博文,对于CommonAdapter以及ImageLoader都有从无到有的具体打造过程。

到此我们的第一个Activity的全部的任务就完毕了~~~

3、展现目录的PopupWindow

如今我们要实现。点击底部的布局弹出我们的目录选择框。而且我们弹出框后面的Activity要变暗;

不急着贴代码,我们先考虑下PopupWindow怎么用最好,我们的PopupWindow须要设置布局文件,须要初始化View,须要初始化事件,还须要和Activity交互~~

那么肯定的,我们使用独立的类,这个类和Activity非常类似。在里面initView(),initEvent()之类的。

我们创建了一个popupWindow使用的超类:

  1. package com.zhy.utils;
  2.  
  3. import java.util.List;
  4.  
  5. import android.content.Context;
  6. import android.graphics.drawable.BitmapDrawable;
  7. import android.view.MotionEvent;
  8. import android.view.View;
  9. import android.view.View.OnTouchListener;
  10. import android.widget.PopupWindow;
  11.  
  12. public abstract class BasePopupWindowForListView<T> extends PopupWindow
  13. {
  14. /**
  15. * 布局文件的最外层View
  16. */
  17. protected View mContentView;
  18. protected Context context;
  19. /**
  20. * ListView的数据集
  21. */
  22. protected List<T> mDatas;
  23.  
  24. public BasePopupWindowForListView(View contentView, int width, int height,
  25. boolean focusable)
  26. {
  27. this(contentView, width, height, focusable, null);
  28. }
  29.  
  30. public BasePopupWindowForListView(View contentView, int width, int height,
  31. boolean focusable, List<T> mDatas)
  32. {
  33. this(contentView, width, height, focusable, mDatas, new Object[0]);
  34.  
  35. }
  36.  
  37. public BasePopupWindowForListView(View contentView, int width, int height,
  38. boolean focusable, List<T> mDatas, Object... params)
  39. {
  40. super(contentView, width, height, focusable);
  41. this.mContentView = contentView;
  42. context = contentView.getContext();
  43. if (mDatas != null)
  44. this.mDatas = mDatas;
  45.  
  46. if (params != null && params.length > 0)
  47. {
  48. beforeInitWeNeedSomeParams(params);
  49. }
  50.  
  51. setBackgroundDrawable(new BitmapDrawable());
  52. setTouchable(true);
  53. setOutsideTouchable(true);
  54. setTouchInterceptor(new OnTouchListener()
  55. {
  56. @Override
  57. public boolean onTouch(View v, MotionEvent event)
  58. {
  59. if (event.getAction() == MotionEvent.ACTION_OUTSIDE)
  60. {
  61. dismiss();
  62. return true;
  63. }
  64. return false;
  65. }
  66. });
  67. initViews();
  68. initEvents();
  69. init();
  70. }
  71.  
  72. protected abstract void beforeInitWeNeedSomeParams(Object... params);
  73.  
  74. public abstract void initViews();
  75.  
  76. public abstract void initEvents();
  77.  
  78. public abstract void init();
  79.  
  80. public View findViewById(int id)
  81. {
  82. return mContentView.findViewById(id);
  83. }
  84.  
  85. protected static int dpToPx(Context context, int dp)
  86. {
  87. return (int) (context.getResources().getDisplayMetrics().density * dp + 0.5f);
  88. }
  89.  
  90. }

也就是封装了一下popupWindow经常使用的一些设置。然后使用了类似模版方法模式,约束子类,必须实现initView,initEvent,init等方法

  1. package com.zhy.imageloader;
  2.  
  3. import java.util.List;
  4.  
  5. import android.view.View;
  6. import android.widget.AdapterView;
  7. import android.widget.AdapterView.OnItemClickListener;
  8. import android.widget.ListView;
  9.  
  10. import com.zhy.bean.ImageFloder;
  11. import com.zhy.utils.BasePopupWindowForListView;
  12. import com.zhy.utils.CommonAdapter;
  13. import com.zhy.utils.ViewHolder;
  14.  
  15. public class ListImageDirPopupWindow extends BasePopupWindowForListView<ImageFloder>
  16. {
  17. private ListView mListDir;
  18.  
  19. public ListImageDirPopupWindow(int width, int height,
  20. List<ImageFloder> datas, View convertView)
  21. {
  22. super(convertView, width, height, true, datas);
  23. }
  24.  
  25. @Override
  26. public void initViews()
  27. {
  28. mListDir = (ListView) findViewById(R.id.id_list_dir);
  29. mListDir.setAdapter(new CommonAdapter<ImageFloder>(context, mDatas,
  30. R.layout.list_dir_item)
  31. {
  32. @Override
  33. public void convert(ViewHolder helper, ImageFloder item)
  34. {
  35. helper.setText(R.id.id_dir_item_name, item.getName());
  36. helper.setImageByUrl(R.id.id_dir_item_image,
  37. item.getFirstImagePath());
  38. helper.setText(R.id.id_dir_item_count, item.getCount() + "张");
  39. }
  40. });
  41. }
  42.  
  43. public interface OnImageDirSelected
  44. {
  45. void selected(ImageFloder floder);
  46. }
  47.  
  48. private OnImageDirSelected mImageDirSelected;
  49.  
  50. public void setOnImageDirSelected(OnImageDirSelected mImageDirSelected)
  51. {
  52. this.mImageDirSelected = mImageDirSelected;
  53. }
  54.  
  55. @Override
  56. public void initEvents()
  57. {
  58. mListDir.setOnItemClickListener(new OnItemClickListener()
  59. {
  60. @Override
  61. public void onItemClick(AdapterView<?> parent, View view,
  62. int position, long id)
  63. {
  64.  
  65. if (mImageDirSelected != null)
  66. {
  67. mImageDirSelected.selected(mDatas.get(position));
  68. }
  69. }
  70. });
  71. }
  72.  
  73. @Override
  74. public void init()
  75. {
  76. // TODO Auto-generated method stub
  77.  
  78. }
  79.  
  80. @Override
  81. protected void beforeInitWeNeedSomeParams(Object... params)
  82. {
  83. // TODO Auto-generated method stub
  84. }
  85.  
  86. }

好了,如今就是我们正在的popupWindow咯。布局目录主要是个ListView。所以在initView里面,我们得设置它的适配器;当然了。这里的适配器依旧用我们的CommonAdapter,几行代码搞定~~

然后我们须要和Activity交互,当我们点击某个目录的时候,外层的Activity须要改变它GridView的数据源,展示我们点击目录的图片;

关于交互,我们从Activity的角度去看弹出框,Activity想知道什么,仅仅想知道选择了别的目录来告诉我。所以我们创建一个接口OnImageDirSelected,对Activity设置回调。

这里还能够这么写:就是把popupWindow的ListView发布出去。然后在Activity里面使用popupWindow.getListView(),setOnItemClickListener,这么做,个人认为不好,耦合度太高。客户简单改下需求“这个目录展示,给我们换了,换成GridView”。呵呵,此时,你须要到处去改动Activity里面的代码。由于你Activity里面居然还有个popupWindow.getListView。

好了,扯多了。初始化事件的代码:

  1. @Override
  2. public void initEvents()
  3. {
  4. mListDir.setOnItemClickListener(new OnItemClickListener()
  5. {
  6. @Override
  7. public void onItemClick(AdapterView<?> parent, View view,
  8. int position, long id)
  9. {
  10.  
  11. if (mImageDirSelected != null)
  12. {
  13. mImageDirSelected.selected(mDatas.get(position));
  14. }
  15. }
  16. });
  17. }

假设有人设置了回调。我们就调用;

到此,整个popupWindow就出炉了,接下来就看啥时候让它展示了。

4、选择不同的目录

上面说道,当扫描图片完毕。拿到包括图片的目录信息列表;这个列表就是我们popupWindow所需的数据,所以我们的popupWindow的初始化在handleMessage(上面贴了handler的代码)里面:

在handleMessage里面调用initListDirPopupWindw

  1. /**
  2. * 初始化展示目录的popupWindw
  3. */
  4. private void initListDirPopupWindw()
  5. {
  6. mListImageDirPopupWindow = new ListImageDirPopupWindow(
  7. LayoutParams.MATCH_PARENT, (int) (mScreenHeight * 0.7),
  8. mImageFloders, LayoutInflater.from(getApplicationContext())
  9. .inflate(R.layout.list_dir, null));
  10.  
  11. mListImageDirPopupWindow.setOnDismissListener(new OnDismissListener()
  12. {
  13.  
  14. @Override
  15. public void onDismiss()
  16. {
  17. // 设置背景颜色变暗
  18. WindowManager.LayoutParams lp = getWindow().getAttributes();
  19. lp.alpha = 1.0f;
  20. getWindow().setAttributes(lp);
  21. }
  22. });
  23. // 设置选择目录的回调
  24. mListImageDirPopupWindow.setOnImageDirSelected(this);
  25. }

我们初始化我们的popupWindow。设置了关闭对话框的回调。已经设置了选择不同目录的回调;
这里仅仅是初始化,以下看我们合适将其弹出的。事实上整个Activity也就一个事件,点击弹出该对话框,所以看Activity的initEvents方法:

  1. private void initEvent()
  2. {
  3. /**
  4. * 为底部的布局设置点击事件。弹出popupWindow
  5. */
  6. mBottomLy.setOnClickListener(new OnClickListener()
  7. {
  8. @Override
  9. public void onClick(View v)
  10. {
  11. mListImageDirPopupWindow
  12. .setAnimationStyle(R.style.anim_popup_dir);
  13. mListImageDirPopupWindow.showAsDropDown(mBottomLy, 0, 0);
  14.  
  15. // 设置背景颜色变暗
  16. WindowManager.LayoutParams lp = getWindow().getAttributes();
  17. lp.alpha = .3f;
  18. getWindow().setAttributes(lp);
  19. }
  20. });
  21. }

能够看到。我们为底部布局设置点击事件。设置popupWindow的弹出与消失的动画;已经让Activity背景变暗变亮,通过改变Window alpha实现的。变亮在弹出框消息的监听里面~~

动画的文件就不贴了,大家自己看源代码;

popupWindow弹出了,用户此时能够选择不同的目录,那么如今该看选择后的回调的代码了:

我们的Activity实现了该接口,直接看实现的方法:

  1. @Override
  2. public void selected(ImageFloder floder)
  3. {
  4.  
  5. mImgDir = new File(floder.getDir());
  6. mImgs = Arrays.asList(mImgDir.list(new FilenameFilter()
  7. {
  8. @Override
  9. public boolean accept(File dir, String filename)
  10. {
  11. if (filename.endsWith(".jpg") || filename.endsWith(".png")
  12. || filename.endsWith(".jpeg"))
  13. return true;
  14. return false;
  15. }
  16. }));
  17. /**
  18. * 能够看到目录的路径和图片的路径分开保存,极大的降低了内存的消耗。
  19. */
  20. mAdapter = new MyAdapter(getApplicationContext(), mImgs,
  21. R.layout.grid_item, mImgDir.getAbsolutePath());
  22. mGirdView.setAdapter(mAdapter);
  23. // mAdapter.notifyDataSetChanged();
  24. mImageCount.setText(floder.getCount() + "张");
  25. mChooseDir.setText(floder.getName());
  26. mListImageDirPopupWindow.dismiss();
  27.  
  28. }

我们改变了GridView的适配器,以及底部的控件上的目录名称,文件数量等等;

好了,到此结束;整篇由于篇幅原因没有贴不论什么布局文件,大家自己通过源代码查看;

在此希望大家能够通过该案例,能够去其糟粕,取其精华,学习当中值得借鉴的代码风格,不要真的当作一个样例去学习~~

源代码点击下载

ps:请真机測试,反正我的模拟器扫描不到图片~

ps:执行出现空指针的话,在getImages中加入推断,if(parentFile.list()==null)continue , 切记~~~具体位置,上面有说;

---------------------------------------------------------------------------------------------------------

我建了一个QQ群。方便大家交流。群号:55032675

----------------------------------------------------------------------------------------------------------

博主部分视频已经上线。假设你不喜欢枯燥的文本,请猛戳(初录。期待您的支持):

1、高仿微信5.2.1主界面及消息提醒

2、高仿QQ5.0側滑

Android 超高仿微信图片选择器 图片该这么载入的更多相关文章

  1. [转]Android 超高仿微信图片选择器 图片该这么加载

    快速加载本地图片缩略图的方法: 原文地址:Android 超高仿微信图片选择器 图片该这么加载 其示例代码下载: 仿微信图片选择器 ImageLoader

  2. Android 超高仿微信图片选择器 图片该这么加载

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/39943731,本文出自:[张鸿洋的博客] 1.概述 关于手机图片加载器,在当今像 ...

  3. Android之仿微信图片选择器

    先上效果图.第一张图显示的是“相机”文件夹中的所有图片:通过点击多张图片可以到第二张图所示的效果(被选择的图片会变暗,同时选择按钮变亮):点击最下面的那一栏可以到第三张图所示的效果(显示手机中所有包含 ...

  4. Android高仿微信图片选择功能的PhotoPicker

    类似于微信修改头像的功能基本上每个app都会有,以前公司开发的项目就有修改头像的功能,但是用的Android系统自带的图片选择器.用Android系统的图片选择器有个好处就是稳定,不会有什么问题.但也 ...

  5. Android仿微信高效压缩图片(libjpeg)

    用过ios手机的同学应该很明显感觉到,ios拍照1M的图片要比安卓拍照排出来的5M的图片还要清晰.这是为什么呢? 这得了解android底层是如何对图片进行处理的. 当时谷歌开发Android的时候, ...

  6. Android NineGridLayout — 仿微信朋友圈和QQ空间的九宫格图片展示自定义控件

    NineGridLayout 一个仿微信朋友圈和QQ空间的九宫格图片展示自定义控件. GitHub:https://github.com/HMY314/NineGridLayout 一.介绍 1.当只 ...

  7. Android 使用ContentProvider扫描手机中的图片,仿微信显示本地图片效果

    版权声明:本文为博主原创文章,未经博主允许不得转载. 转载请注明本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/1873 ...

  8. Android高仿微信(一)——如何消除启动时的白屏

    默认情况下,APP启动时会先把屏幕刷成白色,然后才绘制第一个Activity中的View,这两个步骤之间的延迟会造成启动后先看到白屏(时间大概为1秒左右).时间不长,但是我们也看到,一般的APP时不存 ...

  9. Android 高仿微信头像截取 打造不一样的自定义控件

    转载请表明出处:http://blog.csdn.net/lmj623565791/article/details/39761281,本文出自:[张鸿洋的博客] 1.概述 前面已经写了关于检测手势识别 ...

随机推荐

  1. 354. Russian Doll Envelopes

    You have a number of envelopes with widths and heights given as a pair of integers (w, h). One envel ...

  2. function in Postgres

    CREATE or REPLACE FUNCTION fn_attr_category() RETURNS void AS $BODY$ declare v_tmp_rec record; begin ...

  3. 2018年东北农业大学春季校赛 B wyh的矩阵【找规律】

    链接:https://www.nowcoder.com/acm/contest/93/B来源:牛客网 题目描述 给你一个n*n矩阵,按照顺序填入1到n*n的数,例如n=5,该矩阵如下 1 2 3 4 ...

  4. Java爬虫系列之实战:爬取酷狗音乐网 TOP500 的歌曲(附源码)

    在前面分享的两篇随笔中分别介绍了HttpClient和Jsoup以及简单的代码案例: Java爬虫系列二:使用HttpClient抓取页面HTML Java爬虫系列三:使用Jsoup解析HTML 今天 ...

  5. codevs——1169 传纸条(棋盘DP)

    2008年NOIP全国联赛提高组  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题解  查看运行结果     题目描述 Description 小渊和小 ...

  6. [JLOI2015]有意义的字符串

    4002: [JLOI2015]有意义的字符串 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1000  Solved: 436[Submit][St ...

  7. Atom打开大文件卡死的问题替代方案

    无解,本身是网页的框架,所以直接换回ST或者Notepad++吧.

  8. 一款不错的编程字体Source Code Pro

    我以前一直是用的MS自家的是Consolas的字体,这个字体基本上具有编程字体所需的所有要素:等宽.支持ClearType.中文字体大小合适,l和1,o和0很容易区分.非要挑刺的话就是字体比较小,9号 ...

  9. MFC中 用Static控件做超链接(可以实现变手形、下划线、字体变色等功能)

    1.新建一个基于对话框的工程MyLink,在对话框中拖一个Static控件,ID可改为IDC_SLink. 2.在头文件中添加成员变量: private: CRect m_Rect; CFont* m ...

  10. 关于IIS的IUSER和IWAM帐户

    IUSER是Internet 来宾帐户匿名访问 Internet 信息服务的内置帐户 IWAM是启动 IIS 进程帐户用于启动进程外的应用程序的 Internet 信息服务的内置帐户 (在白吃点就是启 ...