2

让我先解释一下我在做什么,然后再进入所有内容

基本上,我给用户一个选项来从 SD 卡中选择图像,将所选图像的 Uri 存储在我的数据库中,并将其作为列表项的头像,但是带有头像的列表项越多,滚动的速度就越慢清单是

我用一个加载列表Loader

    @Override
public Loader<Cursor> onCreateLoader(int arg0, Bundle arg1) {

    return new CursorLoader(getActivity(),BowlersDB.CONTENT_URI,
            new String[] {BowlersDB.ID,BowlersDB.FIRST_NAME,BowlersDB.LAST_NAME,BowlersDB.PHOTO_URI},null,null,BowlersDB.LAST_NAME + " COLLATE LOCALIZED ASC");
}

我有一个自定义适配器,可以调整图像大小设置文本视图等。

@Override
    public void bindView(final View view,final Context context,final Cursor cursor){
        final int id = cursor.getInt(0);;
        final QuickContactBadge image = (QuickContactBadge)view.findViewById(R.id.quickContactBadge1);
        image.setOnClickListener(new View.OnClickListener(){

            @Override
            public void onClick(View v) {
                if(!fromGame){
                    bowlerID = id;
                    Intent i = new Intent(Intent.ACTION_PICK,android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
                    startActivityForResult(i,1);
                }
            }

        });
        String uri = cursor.getString(3);
        if(uri != null){

                    InputStream input=null;
                    try {
                        input = getActivity().getContentResolver().openInputStream(Uri.parse(cursor.getString(3)));
                        BitmapFactory.Options options = new BitmapFactory.Options();
                        options.inJustDecodeBounds = false;
                        Bitmap og = BitmapFactory.decodeStream(input,null,options);
                        int height = options.outHeight;
                        int width = options.outWidth;
                        BitmapFactory.decodeResource(getResources(), R.drawable.ic_contact_picture, options);

                        int imageHeight = options.outHeight;
                        int imageWidth = options.outWidth;
                        try {
                            input.close();
                        } catch (IOException e) {
                            e.printStackTrace();
                        }

                        float scaledHeight = ((float)imageHeight)/height;
                        float scaledWidth = ((float)imageWidth)/width;

                        Matrix matrix = new Matrix();
                        matrix.postScale(scaledWidth, scaledHeight);

                        final Bitmap resized = Bitmap.createBitmap(og, 0, 0, width, height, matrix, true);
                        try {
                            input.close();
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                                image.setImageBitmap(resized);


                    } catch (FileNotFoundException e) {
                        e.printStackTrace();
                    }
                }else{
                    image.setImageResource(R.drawable.ic_contact_picture);
                }

        TextView first = (TextView)view.findViewById(R.id.bListTextView);
        TextView last = (TextView)view.findViewById(R.id.bListTextView2);
        first.setText(cursor.getString(1));
        last.setText(cursor.getString(2));
    }

}

现在我知道它可能是图像的大小导致它如此缓慢但是当我把那段代码放在一个单独的线程中并且没有任何效果但我认为LoaderManager无论如何都会在后台加载所有内容

这是我的行 xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:minHeight="60dp"
android:id="@+id/names_layout"
android:background="@drawable/list_item_state">

<QuickContactBadge
    android:id="@+id/quickContactBadge1"
    android:layout_width="70dp"
    android:layout_height="70dp"
    android:layout_alignParentLeft="true"
    android:layout_alignParentTop="true"
    android:src="@drawable/ic_contact_picture"
    style="?android:attr/quickContactBadgeStyleWindowLarge" />



<TextView
    android:id="@+id/bListTextView"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:layout_alignParentTop="true"
    android:layout_toRightOf="@+id/quickContactBadge1"
    android:background="@drawable/list_item_state"
    android:paddingLeft="20dp"
    android:paddingTop="2dp"
    android:textAppearance="?android:attr/textAppearanceLarge"
    android:textSize="20dp"
    android:textColor="?android:attr/textColorPrimary" />

<TextView
    android:id="@+id/bListTextView2"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:layout_toRightOf="@+id/quickContactBadge1"
    android:layout_below="@+id/bListTextView"
    android:paddingLeft="20dp"
    android:background="@drawable/list_item_state"
    android:textAppearance="?android:attr/textAppearanceMedium"
    android:textSize="15dp"
    android:textColor="?android:attr/textColorSecondary" 
    android:layout_alignBottom="@+id/quickContactBadge1"/>

</RelativeLayout>

Quick Contact Badge保存图像,如果没有图像集,我会设置默认图像。

那么我该如何解决这个问题呢?

我正在尝试做的一个例子是谷歌音乐应用程序,其中专辑列表包含所有这些专辑头像并且滚动没有问题,它们似乎根据需要出现,但显然它有点不同,因为图像是可能是从服务器下载的

4

2 回答 2

2

您可以延迟加载列表视图的图像。这是一个完整的例子:

http://codehenge.net/blog/2011/06/android-development-tutorial-asynchronous-lazy-loading-and-caching-of-listview-images/

于 2012-08-01T23:27:36.027 回答
1

您正在 bindView 线程中流式传输图像,这通常不是一个好主意。我通常在 asyncTask 中下载和构建位图 - 或者更确切地说,在 RoboAsyncTask 中。我将 ImageView 与图像 URL 一起传递,因此图像是在下载或位图创建完成后在 asynctask 的 onSuccess() 中设置的。您在屏幕上看到的是在其他内容之后不久出现的图像。您还可以将占位符资源传递给异步任务,以便在下载完成时显示。

在您的活动或片段中:

    mediaManager.setImage(imageUrl, imageView, R.drawable.placeholder_image);

在媒体管理器中:

    @Override
public void setImage(String imageUrl, ImageView imageView, int placeholderResource) {
    new SetImageTask(context, imageUrl, imageView, placeholderResource).execute();
}

@Override
public Bitmap getImage(String imageUrl) throws IllegalStateException, IOException, SQLException {

    // put your code to fetch image here....

}

在 SetImageTask(扩展 RoboAsyncTask)中:

    @Override
public void onPreExecute() {
    imageView.setImageResource(placeholderResource);
}

@Override
public Bitmap call() throws Exception {
    return mediaManager.getImage(imageUrl);
}

@Override
public void onSuccess(Bitmap bitmap) {
    imageView.setImageBitmap(bitmap);
}
于 2012-08-01T22:40:15.170 回答