1

アプリに LruCache を実装しようとしていますが、ドットを接続して異なるコンポーネント間でビットマップを渡すのが困難です。

アプリに LruCache を統合する方法を知りたいです。また、LruCache を実装するプロセスを理解しようとしているので、詳細が多いほど良い

最初のクラスは AsyncTask Image Loader で、2 番目のクラスはカスタム アダプターです。

非同期タスク

 public class GetImagesBitmap extends AsyncTask<ImageView, Void, Bitmap>{
    InputStream inputStream;
    ImageView imageView = null;
    String path;




    @Override
    protected Bitmap doInBackground(ImageView... imageViews) {
        this.imageView = imageViews[0];
        return download_Image((String)imageView.getTag());
    }

    @Override
    protected void onPostExecute(Bitmap result) {

        imageView.setImageBitmap(result);
    }

    private Bitmap download_Image(String url) {


        final AndroidHttpClient client = AndroidHttpClient.newInstance("Android");
        HttpGet httpGet = new HttpGet(url);
        try {
            HttpResponse httpResponse = client.execute(httpGet);
            final int statusCode = httpResponse.getStatusLine().getStatusCode();
            if (statusCode != HttpStatus.SC_OK) {
                Log.w("ImageDownloader", "Error " + statusCode
                        + " while retrieving bitmap from " + url);
                return null;
            }

            final HttpEntity entity = httpResponse.getEntity();
            if(entity != null){
                inputStream = null;
                inputStream = entity.getContent();
                final Bitmap bitmap = BitmapFactory.decodeStream(inputStream);
                return bitmap;
            }
                if(inputStream != null){
                    inputStream.close();
                }
                entity.consumeContent();

        } catch (IOException e) {

            httpGet.abort();
             Log.w("ImageDownloader", "Error while retrieving bitmap from " + url);
        }finally{
            if(client != null){
                client.close();
            }
        }

        return null;
        }



    }

カスタム ビュー アダプター クラス

 public class CustomListingListAdapter extends ArrayAdapter<HashMap<String, String>> {





    private ArrayList<HashMap<String, String>> data;
    private Context context;
    private LayoutInflater mInflater;
    private int viewId;
    private String[] tag;
    HashMap<String, String> currentData = new HashMap<String, String>();

    //private static final String TAG_CONTENT = "content";
    //private static final String TAG_RATING = "rating";
    private static final String TAG_NAME = "name";
    private static final String TAG_IMAGE_PATH = "image_path";
    private static final String TAG_PRODUCT_ID = "product_id";
    private static final String TAG_RATING = "rating";

    public CustomListingListAdapter(Context c,
            ArrayList<HashMap<String, String>> data,
            int viewId, String[] tag) {
        super( c, viewId, data);

        this.context = c;
        this.data= data;
        this.viewId = viewId ;
    }

    @Override
    public int getCount() {

        return data.size();
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {

        View vi = convertView;
        Holder holder;

        if (convertView == null) {

            // Inflate the view since it does not exist
            if (vi == null) {
                mInflater = (LayoutInflater) getContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                vi = mInflater.inflate(R.layout.list_item_layout, null);

            }

            holder = new Holder();

            holder.Name = (TextView) vi.findViewById(R.id.name);
            holder.pid = (TextView) vi.findViewById(R.id.pid);
            holder.imageView = (ImageView) vi.findViewById(R.id.list_image);
            holder.ratingView = (ImageView) vi.findViewById(R.id.num_stars);

            vi.setTag(holder);  
        }else {
            holder = (Holder) vi.getTag();
        }


        currentData = (HashMap<String, String>) data.get(position);
        holder.imageView.setTag(currentData.get(TAG_IMAGE_PATH));





        if (currentData != null) {

            holder.Name.setText(currentData.get(TAG_NAME));
            holder.pid.setText(currentData.get(TAG_PRODUCT_ID));
        }   

        new GetImagesBitmap().execute(holder.imageView);
    return vi;


    }
    private static class Holder {

        public ImageView imageView;
        public ImageView ratingView;
        public TextView pid;
        public TextView Name;

       } 
    }
4

1 に答える 1

5

あなたは正しい道を進んでいますが、List/GridView でのビューのリサイクルをコードのどこで説明しているのかわかりませんでした。確かに getView に渡された convertView を再利用していますが、すべての getView で新しい AsyncTask を起動すると、タスクが画像をロードすることにつながりますが、開始されたビューがリサイクルされると、間違ったリスト行に挿入されます。私のアドバイスは、Google の新しいネットワーキング ライブラリである Volley をチェックすることです。ImageView に画像を非同期的にロードする NetworkImageView クラスがあります。私がお勧めする理由は、ディスク上とメモリ内の両方でのキャッシングであり、ビューのリサイクルは自動的に処理されます。

ボレー: https://developers.google.com/events/io/sessions/325304728

自分でこれに取り組みたい場合は、ListView でのビットマップの読み込みに関する Android 開発者のブログ投稿を確認してください。やりたいように AsyncTasks を使用することを示しましたが、ビューのリサイクルを正しく処理します。

記事: http://developer.android.com/training/displaying-bitmaps/process-bitmap.html

最後のちょっとしたヒント: ネットワーク呼び出しごとに新しい HttpClient をインスタンス化しないでください。作成には時間がかかるため、推奨される方法は、マルチスレッド用に構成された 1 つの HttpClient をシングルトン インスタンスとして保持し、それをネットワーク呼び出しに使用することです。または、リクエストごとに新しい HttpUrlConnection を作成することもできます。それはそのクラスで問題ありません。

編集:申し訳ありませんが、私は実際に質問に答えていないことに気付きました. Bimap 用の LruCache を実装するには、サポート ライブラリの LruCache クラスをいくつか調整して拡張します。Volley ライブラリで使用する Bitamps をキャッシュするために使用したクラスを次に示します。

// From com.android.volley.toolbox.ImageLoader
public interface ImageCache {
    public Bitmap getBitmap(String url);
    public void putBitmap(String url, Bitmap bitmap);
}

// My ImageCache implementation
public class BitmapLruCache extends LruCache<String, Bitmap> implements ImageCache {
    private static final int DEFAULT_CACHE_SIZE = (int) (Runtime.getRuntime().maxMemory() / 1024) / 8;

    public BitmapLruCache() {
        this(DEFAULT_CACHE_SIZE);
    }

    public BitmapLruCache(int maxSize) {
        super(maxSize);
    }

    @Override
    public Bitmap getBitmap(String url) {
        return get(url);
    }

    @Override
    public void putBitmap(String url, Bitmap bitmap) {
        put(url, bitmap);
    }

    @Override
    protected int sizeOf(String key, Bitmap value) {
        return value == null ? 0 : value.getRowBytes() * value.getHeight() / 1024;
    }
}
于 2013-08-22T00:35:52.287 に答える