写真を表示するとメモリ不足になる

Showing photos runs out of memory


質問 written by TMB87 @2012-10-24 12:36:50Z

: 6 : 1 : 2

ダイアログボックスを使用して、Androidプロジェクトの画像を表示しています。 最初のものは正常に開きますが、閉じて別のプロセスを表示するプロセスを再度実行すると、メモリエラーでアプリが倒れます(samsung galaxy s3で実行されているため、問題になりません)。

エラー:

10-24 11:25:45.575: E/dalvikvm-heap(29194): Out of memory on a 31961104-byte allocation.
10-24 11:25:45.580: E/AndroidRuntime(29194): FATAL EXCEPTION: main
10-24 11:25:45.580: E/AndroidRuntime(29194): java.lang.OutOfMemoryError
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.graphics.BitmapFactory.nativeDecodeStream(Native Method)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:587)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.graphics.BitmapFactory.decodeFile(BitmapFactory.java:389)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.graphics.BitmapFactory.decodeFile(BitmapFactory.java:418)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.graphics.drawable.Drawable.createFromPath(Drawable.java:882)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.widget.ImageView.resolveUri(ImageView.java:569)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.widget.ImageView.setImageURI(ImageView.java:340)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at com.directenquiries.assessment.tool.AddAsset.loadPhoto(AddAsset.java:771)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at com.directenquiries.assessment.tool.AddAsset$11.onClick(AddAsset.java:748)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at com.android.internal.app.AlertController$AlertParams$3.onItemClick(AlertController.java:936)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.widget.AdapterView.performItemClick(AdapterView.java:292)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.widget.AbsListView.performItemClick(AbsListView.java:1359)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.widget.AbsListView$PerformClick.run(AbsListView.java:2988)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.widget.AbsListView$1.run(AbsListView.java:3783)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.os.Handler.handleCallback(Handler.java:605)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.os.Handler.dispatchMessage(Handler.java:92)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.os.Looper.loop(Looper.java:137)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at android.app.ActivityThread.main(ActivityThread.java:4517)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at java.lang.reflect.Method.invokeNative(Native Method)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at java.lang.reflect.Method.invoke(Method.java:511)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:993)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:760)
10-24 11:25:45.580: E/AndroidRuntime(29194):    at dalvik.system.NativeStart.main(Native Method)

コードの読み込み:

public void loadPhotoList(){

    Cursor f = db.rawQuery("select * from stationphotos where StationObjectID =  '"+ checkStationObjectID + "'", null);  
    final ArrayList<String> mHelperNames= new ArrayList<String>();

            if(f.getCount() != 0) {
              f.moveToFirst();

                f.moveToFirst();
                while(!f.isAfterLast()) {
                    mHelperNames.add(f.getString(f.getColumnIndex("FilePath")));
                    f.moveToNext();
                }
            }
     f.close();
     final String [] nameStrings = new String [mHelperNames.size()];

     for(int i=0; i<mHelperNames.size(); i++)
        nameStrings[i] = mHelperNames.get(i).toString();


    AlertDialog.Builder builder = new AlertDialog.Builder(this);

    builder.setTitle("Select Picture");
    builder.setItems(nameStrings, new DialogInterface.OnClickListener() {

       public void onClick(DialogInterface dialog, int item) {

          loadPhoto(mHelperNames.get(item).toString());

       }

    });

    AlertDialog alert = builder.create();

    alert.show();

 }


public void loadPhoto(String imagepath){

    Dialog dialog = new Dialog(this);
    dialog.setContentView(R.layout.activity_show_image);
    dialog.setTitle("Image");
    dialog.setCancelable(true);

    ImageView img = (ImageView) dialog.findViewById(R.id.imageView1);
    img.setImageResource(R.drawable.ico_partial);
    Uri imgUri = Uri.parse(imagepath);
    img.setImageURI(imgUri);


    dialog.show();
}

編集:私はそれを今動作させるために使用しているもの:

public static Bitmap decodeSampledBitmapFromFile(String imagePath, int reqWidth, int reqHeight) {

    // First decode with inJustDecodeBounds=true to check dimensions
    final BitmapFactory.Options options = new BitmapFactory.Options();
    options.inJustDecodeBounds = true;
    BitmapFactory.decodeFile(imagePath, options);

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

    // Decode bitmap with inSampleSize set
    options.inJustDecodeBounds = false;
    return BitmapFactory.decodeFile(imagePath, options);
}
    public static 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) {
        if (width > height) {
            inSampleSize = Math.round((float)height / (float)reqHeight);
        } else {
            inSampleSize = Math.round((float)width / (float)reqWidth);
        }
    }
    return inSampleSize;
}



img.setImageBitmap(decodeSampledBitmapFromFile(imagepath, 500, 500));
コメント 1

メモリプロファイラーは何を示していますか?ヒープサイズ、ヒープ空きメモリとは何ですか?

written by サンダースター @2012-10-24 10:37:04Z

コメント 2

写真をロードすると、ヒープは最大44MBを使用しました。別の方法がありますか?1 45.258 MB 44.830 MB 438.242 KB 99.05%57,128

written by TMB87 @2012-10-24 10:50:19Z

コメント 3

高さと幅が500であることがわかりますか?

written by ジョンジョー @2017-01-11 09:09:48Z

コメント 4

私は本当に覚えていません(4年前!?)が、どんな目的であれ、常に500500であるように見えます-最終行

written by x @2017-01-18 15:01:19Z

回答 1 written by Community @2017-05-23 12:01:55Z
5

私はこれを経験しましたが、私の答えがベストプラクティスであるかどうかはわかりません...

これはおそらくあなたの問題です:

Uri imgUri = Uri.parse(imagepath);
    img.setImageURI(imgUri);

写真をロードすると、ヒープは最大44MBを使用しました。

大きな画像をロードすると、おそらくメモリ不足になります。 また、一連のイメージを読み込むときに適切なメモリ管理を使用しないと、メモリが不足します。

BitmapFactoryを使用してファイルをデコードし、 BitmapFactory.Optionsを使用して新しいスクレイプビットマップを作成し、画面に表示しました。

その考え方は、BitmapFactory.Optionsを使用すると、イメージが使い果たす可能性のあるメモリをより詳細に制御できるようになるということです。 たとえば、これらの解像度をサポートしていない画面にフルサイズで1920x1080の画像を表示する必要はありません。

あなたに役立つかもしれないいくつかの関連するスタックオーバーフローの質問があります、特に:BitmapFactory.OptionsオプションinJustDecodeBounds焦点を当てた大きなビットマップの処理

最後に、 Aqueryと呼ばれるこの素晴らしい軽量ライブラリを見つけました。 大きなファイルを処理するための多くの方法があり、これまでのところうまく機能しています。 一見の価値があります。 Web開発に関してJqueryと同様のパラダイムを使用し(名前の由来)、ビューを操作するための短い構文を導入します。 ドキュメントの次のセクションでは、フォールバックイメージを含む、ローカルおよびリモートソースからのイメージの読み込みについて説明します。

http://code.google.com/p/android-query/wiki/ImageLoading

コメント 1

このおかげで、ビットマップファクトリを調べただけで、現在は圧縮中です

written by 。TMB8712年 @2012-10-24 12:35:56Z