質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.34%

  • Android Studio

    4107questions

    Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。

取得した画像をgridViewで表示する際に、スレッドを使って高速化したい

受付中

回答 2

投稿

  • 評価
  • クリップ 0
  • VIEW 3,113

nanshime_2621

score 20

端末内に保存されている画像を取得して、gridViewを使ってサムネイル表示するコードを書きましたが、画像が多くなると、取得に時間がかかり、表示までかなり待たされる状態です。
スレッドを使って、画像を取得しながら同時に取得した画像を表示していくコードに書きかえたいと思っていますが、うまくコードにおとせません。

詳しい方がいらっしゃいましたらご教授お願いします。

MainActivity.java

package com.example.xx.gridlayout_3;

import android.app.Activity;
import android.content.ContentResolver;
import android.content.Intent;
import android.database.Cursor;
import android.graphics.Bitmap;
import android.net.Uri;
import android.provider.MediaStore;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.GridView;
import java.util.ArrayList;
import java.util.List;

public class MainActivity extends Activity implements AdapterView.OnItemClickListener {

    private GridView mGridView = null;

    private List<String> imagePaths = new ArrayList<String>();

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mGridView = (GridView)findViewById(R.id.gridView1);

        //グリッド4列表示
        mGridView.setNumColumns(4);
        //表示する画像を取得(DBへアクセス)
        Uri uri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; //SDカード
        //queryメソッドでカーソルを取得
        Cursor cursor = getContentResolver().query(uri, null, null, null, null);
        ContentResolver cr = getContentResolver();
        ArrayList<Bitmap> lstBitmap = new ArrayList<Bitmap>();
        //カーソルの一番目に移動
        cursor.moveToFirst();
        for (int i = 0; i < cursor.getCount(); i++){
            //画像iD取得
            long id = cursor.getLong(cursor.getColumnIndexOrThrow("_id"));
            //画像パスを取得
            imagePaths.add(cursor.getString(1));
            //サムネイルを取得
            Bitmap bmp = MediaStore.Images.Thumbnails.getThumbnail(cr, id, MediaStore.Images.Thumbnails.MICRO_KIND, null);
            lstBitmap.add(bmp);
            cursor.moveToNext();
//           System.out.println(imagePaths);

        }
        //アダプター作成
        BitmapAdapter adapter = new BitmapAdapter(getApplicationContext(), lstBitmap);
        //グリッドにアダプタを設定
        mGridView.setAdapter(adapter);

        //画像をクリックしたときの処理
        mGridView.setOnItemClickListener(this);
    }

    @Override
    public void onItemClick(AdapterView<?> parent,View v,int position,long id){
        Intent intent = new Intent(this.getApplicationContext(),SubActivity.class);
        String selectedImagePath = imagePaths.get(position);
        //「position」を、"imagePath"をキーワードにしてインテントにセット
        intent.putExtra("imagePath",selectedImagePath);
        //アクティビティをスイッチ
        startActivity(intent);
    }
}

BitmapAdapter.java

package com.example.xx.gridlayout_3;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.media.Image;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.ImageView;
import java.util.List;
/**
 * Created by xx on 2015/12/16.
 */

/*"implements Runnable"を追加*/
public class BitmapAdapter extends ArrayAdapter<Bitmap> /*implements Runnable */{

    //GridView内で画像を表示するために作成したレイアウト
    private static final int RESOURCE_ID = R.layout.grid_item;

    private LayoutInflater mInflater;

    public BitmapAdapter(Context context, List<Bitmap> objects) {
        super(context, RESOURCE_ID, objects);
        mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);

    }

   // @Override
    //public void run() {

    @Override

    public View getView(int position, View convertView, ViewGroup parent) {
        if (convertView == null) {
            convertView = mInflater.inflate(RESOURCE_ID, null);
        }
        ImageView imageView = (ImageView) convertView.findViewById(R.id.imageView);
        imageView.setImageBitmap(getItem(position));
        return convertView;
    }
//}
}

SubActivity.java

import android.app.Activity;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Bundle;
import android.widget.ImageView;

/**
 * Created by xx on 2015/12/16.
 */
public class SubActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState){
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_sub);

        Intent intent = getIntent();
        String imagePath = intent.getStringExtra("imagePath");
        ImageView imageView = (ImageView)findViewById(R.id.selectedImage);
        Bitmap bmp = BitmapFactory.decodeFile(imagePath);
        imageView.setImageBitmap(bmp);
    }

}

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 2

0

とりあえずUIスレッドの進行を邪魔している処理を調べることから始めましょう。
実装方法(Tread、AsyncTask、AsyncTaskLoaderなど)はユーザーにどのように見せたいのかが決まらないと決定できないですね。

今回で言えば下記の処理を非同期にすべきでしょうね。

uri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; //SDカード 
        //queryメソッドでカーソルを取得 
        Cursor cursor = getContentResolver().query(uri, null, null, null, null); 
        ContentResolver cr = getContentResolver(); 
        ArrayList<Bitmap> lstBitmap = new ArrayList<Bitmap>(); 
        //カーソルの一番目に移動 
        cursor.moveToFirst(); 
        for (int i = 0; i < cursor.getCount(); i++){ 
            //画像iD取得 
            long id = cursor.getLong(cursor.getColumnIndexOrThrow("_id")); 
            //画像パスを取得 
            imagePaths.add(cursor.getString(1)); 
            //サムネイルを取得 
            Bitmap bmp = MediaStore.Images.Thumbnails.getThumbnail(cr, id, MediaStore.Images.Thumbnails.MICRO_KIND, null); 
            lstBitmap.add(bmp); 
            cursor.moveToNext(); 
//           System.out.println(imagePaths);

}

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/01/29 17:27

    Threadを使用して、
    MainActivity.javaを書き換え、SubThread.javaを追加しましたが、アプリが落ちてしまいます。


    MainActivity.java

    package com.example.ohmura.gridlayout_3;

    import android.app.Activity;
    import android.content.ContentResolver;
    import android.content.Intent;
    import android.database.Cursor;
    import android.graphics.Bitmap;
    import android.net.Uri;
    import android.provider.MediaStore;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.view.View;
    import android.widget.AdapterView;
    import android.widget.GridView;
    import java.util.ArrayList;
    import java.util.List;

    public class MainActivity extends Activity implements AdapterView.OnItemClickListener {

    public GridView mGridView = null;

    public List<String> imagePaths = new ArrayList<String>();
    public ArrayList<Bitmap> lstBitmap;

    public ContentResolver content = getContentResolver();

    @Override
    public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    mGridView = (GridView)findViewById(R.id.gridView1);


    SubThread sub = new SubThread();
    Thread th = new Thread(sub);
    th.run();

    //アダプター作成
    BitmapAdapter adapter = new BitmapAdapter(getApplicationContext(), lstBitmap);
    //グリッドにアダプタを設定
    mGridView.setAdapter(adapter);

    mGridView.setOnItemClickListener(this);

    }

    @Override
    public void onItemClick(AdapterView<?> parent,View v,int position,long id){
    Intent intent = new Intent(this.getApplicationContext(),SubActivity.class);
    String selectedImagePath = imagePaths.get(position);
    //「position」を、"imagePath"をキーワードにしてインテントにセット
    intent.putExtra("imagePath",selectedImagePath);
    //アクティビティをスイッチ
    startActivity(intent);
    }
    }




    SubThread.java


    package com.example.xx.gridlayout_3;

    import android.app.Activity;
    import android.content.ContentResolver;
    import android.database.Cursor;
    import android.graphics.Bitmap;
    import android.net.Uri;
    import android.os.Bundle;
    import android.provider.MediaStore;
    import android.content.ContentResolver;
    import com.google.android.gms.appindexing.Action;
    import com.google.android.gms.appindexing.AppIndex;
    import com.google.android.gms.common.api.GoogleApiClient;

    import java.util.ArrayList;

    /**
    * Created by xx on 2016/01/29.
    */
    public class SubThread implements Runnable {

    /**
    * ATTENTION: This was auto-generated to implement the App Indexing API.
    * See https://g.co/AppIndexing/AndroidStudio for more information.
    */
    private GoogleApiClient client;

    @Override
    public void run() {

    //グリッド4列表示
    MainActivity main = new MainActivity();
    main.mGridView.setNumColumns(4);
    //表示する画像を取得(DBへアクセス)
    Uri uri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; //SDカード
    //queryメソッドでカーソルを取得
    //Cursor cursor = getContentResolver().query(uri, null, null, null, null);
    Cursor cursor = main.content.query(uri, null, null, null, null);
    // ContentResolver cr = getContentResolver();
    ContentResolver cr = main.content;
    main.lstBitmap = new ArrayList<Bitmap>();
    //カーソルの一番目に移動
    cursor.moveToFirst();
    for (int i = 0; i < cursor.getCount(); i++) {
    //画像iD取得
    long id = cursor.getLong(cursor.getColumnIndexOrThrow("_id"));
    //画像パスを取得
    main.imagePaths.add(cursor.getString(1));
    Bitmap bmp = MediaStore.Images.Thumbnails.getThumbnail(cr, id, MediaStore.Images.Thumbnails.MICRO_KIND, null);
    main.lstBitmap.add(bmp);
    cursor.moveToNext();

    }
    }

    }

    キャンセル

  • 2016/01/29 17:42

    アプリが落ちますではわかりません。ログを貼るべきです。

    また、プログラム上でActivityをnewできるとどこかで学びましたか?

    キャンセル

0

会員登録したばかりなので、不備があったらすみません。

一覧する情報だけは最初に取っておいて、画像のロードなど重いところだけ
非同期にするというのはよくやります。
こんな感じではどうでしょうか。

MainActivityではカーソルからの情報取得だけ行い、画像のロードは
BitmapAdapterのImageLoadTaskで行います。
なお、Bitmapのキャッシュ処理などは省いています。

MainActivity

public class MainActivity extends Activity implements AdapterView.OnItemClickListener {

    private GridView mGridView = null;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mGridView = (GridView) findViewById(R.id.gridView1);

        //グリッド4列表示
        mGridView.setNumColumns(4);
        //表示する画像を取得(DBへアクセス)
        Uri uri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; //SDカード
        //queryメソッドでカーソルを取得
        Cursor cursor = getContentResolver().query(uri, null, null, null, null);
        ArrayList<Pair<Long,String>> targets = new ArrayList<Pair<Long,String>>();

        //カーソルの一番目に移動
        cursor.moveToFirst();
        for (int i = 0; i < cursor.getCount(); i++) {
            long id = cursor.getLong(cursor.getColumnIndexOrThrow("_id"));
            targets.add(new Pair<Long, String>(Long.valueOf(id), cursor.getString(1)));
            cursor.moveToNext();
        }
        //アダプター作成
        BitmapAdapter adapter = new BitmapAdapter(getApplicationContext(), targets);
        //グリッドにアダプタを設定
        mGridView.setAdapter(adapter);

        //画像をクリックしたときの処理
        mGridView.setOnItemClickListener(this);
    }

    @Override
    public void onItemClick(AdapterView<?> parent, View v, int position, long id) {
        Intent intent = new Intent(this.getApplicationContext(), SubActivity.class);
        Pair<Long,String> item = (Pair<Long,String>)parent.getAdapter().getItem(position);
        String selectedImagePath = item.second;
        //「position」を、"imagePath"をキーワードにしてインテントにセット
        intent.putExtra("imagePath", selectedImagePath);
        //アクティビティをスイッチ
        startActivity(intent);
    }
}

BitmapAdapter

import android.content.ContentResolver;
import android.content.Context;
import android.graphics.Bitmap;
import android.os.AsyncTask;
import android.os.Handler;
import android.provider.MediaStore;
import android.util.Pair;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.ImageView;

import java.util.List;
/**
 * Created by xx on 2015/12/16.
 */
public class BitmapAdapter extends ArrayAdapter<Pair<Long,String>> {

    //GridView内で画像を表示するために作成したレイアウト
    private static final int RESOURCE_ID = R.layout.grid_item;

    private LayoutInflater mInflater;
    private final ContentResolver resolver;
    private final Handler handler;

    public BitmapAdapter(Context context, List<Pair<Long,String>> objects) {
        super(context, RESOURCE_ID, objects);
        mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        resolver = context.getContentResolver();
        handler = new Handler();
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        if (convertView == null) {
            convertView = mInflater.inflate(RESOURCE_ID, null);
        }
        ImageView imageView = (ImageView) convertView.findViewById(R.id.imageView);
        // TODO: imageViewにロード中的な画像を貼る
        Pair<Long,String> item = getItem(position);
        ImageLoadTask task = new ImageLoadTask(resolver, handler, imageView);
        task.execute(item.first);

        return convertView;
    }

    private static class ImageLoadTask extends AsyncTask<Long,Void, Pair<Long,Bitmap>> {
        private final ImageView view;
        private final ContentResolver resolver;
        private final Handler handler;

        ImageLoadTask(ContentResolver resolver, Handler handler, ImageView view) {
            this.resolver = resolver;
            this.view = view;
            this.handler = handler;
            view.setTag(this);
        }

        private boolean isTargetChanged() {
            return    view.getTag() != this;
        }

        @Override
        protected void onPostExecute(final Pair<Long,Bitmap> pair) {
            if (pair == null || pair.second == null) {
                return;
            }
            if (isTargetChanged()) {
                return;
            }
            Runnable r = new Runnable() {
                @Override
                public void run() {
                    view.setImageBitmap(pair.second);
                }
            };
            handler.post(r);
        }

        @Override
        protected Pair<Long,Bitmap> doInBackground(Long... longs) {
            if (isTargetChanged()) {
                return    null;
            }
            long id = longs[0].longValue();
            return    new Pair<Long,Bitmap>(Long.valueOf(id),
                            MediaStore.Images.Thumbnails.getThumbnail(resolver, id,
                                MediaStore.Images.Thumbnails.MICRO_KIND, null));
        }
    }
}

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.34%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

同じタグがついた質問を見る

  • Android Studio

    4107questions

    Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。