繁体   English   中英

android:在viewpager中使用gridview显示156个图像

[英]android: showing 156 images using gridview inside viewpager

我想使用18页的Viewpager显示156个缩略图,每个页面有9个缩略图。 我已经实现如下:

Index_gridview类:

public void init() 
{  
    PageCount = (int) Math.ceil(mThumbIds.length / PAGE_SIZE);  
    mLists = new ArrayList<GridView>();  

    for (int i = 0; i < PageCount; i++) 
    {  
        GridView gv = new GridView(this);  
        gv.setAdapter(new MyGridViewAdapter(this, mStrs, mThumbIds, i, icon_dimension));  
        gv.setGravity(Gravity.CENTER);  
        gv.setClickable(true);  
        gv.setFocusable(true);  
        gv.setNumColumns(NUMOFCOLUMN);  
        gv.setStretchMode(GridView.STRETCH_COLUMN_WIDTH);
        gv.setHorizontalSpacing(gridview_horizontal_padding);
        gv.setVerticalSpacing(gridview_vertical_padding);
        mLists.add(gv);  
    }         
    page_number.setText("1/"+PageCount);
}  

GridViewAdapter类:

public class MyGridViewAdapter extends BaseAdapter 
{   
    private Animation zoom     = null;
    // view
    int icon_width, icon_height;
    Typeface tf;
    private Context mContext;  
    private List<String> mLists;  
    public static final int PAGE_SIZE = 9; // showing how many items in 1 page  

    public MyGridViewAdapter(Context pContext, String[] pStrs, Integer[] image_ref, int page, int width) 
    {  
        this.mContext = pContext;  
        mLists = new ArrayList<String>();  
        int i = page * PAGE_SIZE;  
        int end = i + PAGE_SIZE;  
        while ((i < image_ref.length) && (i < end)) 
        {  
            mLists.add(pStrs[i]); 
            i++;  
        }  

        icon_width = width;
        icon_height = icon_width;

        tf = Typeface.createFromAsset(mContext.getAssets(),"fonts/HomegirlKiddo.ttf");   
    }  

    @Override  
    public int getCount() 
    {           
        return mLists.size();  
    }  
    @Override  
    public Object getItem(int position) 
    {           
        return mLists.get(position);  
    }  
    @Override  
    public long getItemId(int position) 
    {     
        return position;  
    }  

    @Override  
    public View getView(int position, View convertView, ViewGroup parent) 
    {
        ViewHolder holder;
        if (convertView == null)
        {
            LayoutInflater inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
            convertView = inflater.inflate(R.layout.menugrid, parent, false);
            holder = new ViewHolder();

            holder.img = (ImageView) convertView.findViewById(R.id.imageicon);
            convertView.setTag(holder);
        } 
        else
        {
            holder = (ViewHolder) convertView.getTag();
        }


        final int myNum = Integer.parseInt(mLists.get(position)) -1;
        ImageView imageView = holder.img;

        try
        {
            imageView.setImageBitmap(decodeSampledBitmapFromResource(mContext.getResources(), mThumbIds[myNum], icon_width, icon_width));   
        }
        catch (OutOfMemoryError e)
        {
            ((Index_Gridview)mContext).custom_toast("System Out-of-Error! Restarting...");
            ((Index_Gridview)mContext).restart_app();  
        }


        imageView.setOnClickListener(new OnImageClickListener(myNum));       
        return convertView;
    }

private Bitmap decodeSampledBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight) 
{
    // First decode with inJustDecodeBounds=true to check dimensions
    final BitmapFactory.Options options = new BitmapFactory.Options();
    options.inPreferredConfig = Bitmap.Config.ARGB_8888;
    options.inJustDecodeBounds = true;
    BitmapFactory.decodeResource(res, resId, options);

    // Calculate inSampleSize
    options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);

    // Decode bitmap with inSampleSize set
    options.inJustDecodeBounds = false;
    return BitmapFactory.decodeResource(res, resId, options);
}

private int calculateInSampleSize (BitmapFactory.Options options, int reqWidth, int reqHeight) 
{
    // Raw height and width of image
    final int height = options.outHeight;
    final int width = options.outWidth;
    int inSampleSize = 1;

    if (height > reqHeight || width > reqWidth) 
    {   
        // Calculate ratios of height and width to requested height and width
        final int heightRatio = Math.round((float) height / (float) reqHeight);
        final int widthRatio = Math.round((float) width / (float) reqWidth);

        // Choose the smallest ratio as inSampleSize value, this will guarantee
        // a final image with both dimensions larger than or equal to the requested height and width.
        inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;
    }   
    return inSampleSize;
} 

class ViewHolder 
{
    ImageView img;
}

// References to our images in res > drawable
public Integer[] mThumbIds =  new Integer[] 
    {
        R.drawable.t_aaa, 
        R.drawable.t_bbb, 
                    ....listing of 156 image references here
    }

题:

我已经尝试通过'decodeSampledBitmapFromResource'导入缩略图。 原始缩略图各约为75KB。

它很容易遇到OOM。 如何增强代码以减轻OOM错误?

谢谢!!

您可以尝试使用RAM和磁盘缓存来重用您的位图。

这是android支持v4中LRUCache的链接:

http://developer.android.com/reference/android/support/v4/util/LruCache.html

这是Jake Wharton对DiskLRUCache项目的链接

https://github.com/JakeWharton/DiskLruCache

你必须自己做一些记忆管理 - 毫无疑问。 您可以回收屏幕外页面并在必要时重新加载它们。 LRUCache非常好,但它会将所有内存用于图像。

您的画廊不是那么大,您可以考虑将其保存在本机内存中。 你需要30MB的hdpi 800x480屏幕到150MB的高清屏幕。 使用JNI和C调用,您可以向android询问安装在硬件中的整个内存(512 - 1GB)。 这样的内存不是Java堆的一部分,因此您不必担心应用程序中其他内容缺少内存。

当然,这不是完美的解决方案,因为占用的内存比系统提供的内存多,这将导致分配失败。 你必须照顾它。

链接: http//code.google.com/p/native-buffer/

NativeBuffer是一种缓存,它使用C分配的内存来存储图像数据。 它比重新加载图像或在SD卡上缓存它们要快得多。

https://code.google.com/p/android-query/wiki/AsyncAPI?tm=6在一个示例中使用我知道https://play.google.com/store/apps/details?id=com.app .ive它可以处理任意数量的图像....

我认为重新发明轮子并不好,希望它可以帮助你实现你想要的。

谢谢

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM