3

サーバーから 2 つの画像をダウンロードし、SD カードに保存して表示するアプリを作成しています (15 秒ごとに循環して画像を変更します)。

完了しましたが、数分後にプログラムがクラッシュします: OutOfMemoryError: ビットマップが VM 予算のサイズを超えています。

したがって、どこかでメモリ リークが発生しているはずです。私はまだAndroidプログラミングにかなり慣れていません.VMが実行しないようにするために必要なクリーンアップについてはよくわかりません。

何か助けはありますか?

private ImageView mImageView;
private Bitmap mImageBitmap;
public static String rslt="";
/** Called when the activity is first created. */
@SuppressLint("NewApi")
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    //Remove title bar
    this.requestWindowFeature(Window.FEATURE_NO_TITLE);

    //Remove notification bar
    this.getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
    setContentView(R.layout.main);

    mImageView = (ImageView) findViewById(R.id.imageView1);

    mImageBitmap=null;

    final Handler handler = new Handler();
    Runnable runnable = new Runnable() {
        int count = 1;
        public void run() {
            Bitmap bm = getNewImages();
            if (bm != null){
                mImageView.setImageBitmap(bm);
                bm=null;
                System.gc();
                handler.postDelayed(this, 15* 1000);
            }else if(count==1){
                File img1 = new File(Environment.getExternalStorageDirectory(), "1.jpg");
                if(img1.exists()){
                    bm = BitmapFactory.decodeFile(img1.getAbsolutePath());
                    if (bm != null){
                        mImageView.setImageBitmap(bm);
                        bm=null;
                    }
                }

                count = 2;
                System.gc();
                handler.postDelayed(this, 15* 1000);
            }else if (count==2){
                File img2 = new File(Environment.getExternalStorageDirectory(), "2.jpg");
                if(img2.exists()){
                    bm = BitmapFactory.decodeFile(img2.getAbsolutePath());
                    if (bm != null){
                        mImageView.setImageBitmap(bm);

                        bm=null;
                    }
                }

                count=1;
                System.gc();
                handler.postDelayed(this, 15* 1000);
            }

        }
        private Bitmap getNewImages() {
            ConnectivityManager connManager = (ConnectivityManager) getSystemService(CONNECTIVITY_SERVICE);
            NetworkInfo mWifi = connManager.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
            mImageBitmap=null;
            if (mWifi.isConnected()) {
                try{
                    //              EditText ed1=(EditText)findViewById(R.id.editText1);
                    //              String rego=ed1.getText().toString();
                    String rego="AGR905";
                    rslt="START";
                    Caller c=new Caller();
                    c.rego=rego;
                    c.join();
                    c.start();
                    while(rslt=="START")
                    {
                        try
                        {
                            Thread.sleep(10);
                        }
                        catch(Exception ex)
                        {
                        }
                    }
                }
                catch(Exception ex){
                }

                if (rslt.length()<=1600){
                }else{
                    byte[] decodedString = Base64.decode(rslt, Base64.DEFAULT);
                    mImageBitmap = BitmapFactory.decodeByteArray(decodedString, 0, decodedString.length);

                    ByteArrayOutputStream bytes = new ByteArrayOutputStream();
                    mImageBitmap.compress(Bitmap.CompressFormat.JPEG, 100, bytes);



                    File from = new File(Environment.getExternalStorageDirectory(), "1.jpg");
                    if(from.exists()){
                        File to = new File(Environment.getExternalStorageDirectory(), "2.jpg");
                        from.renameTo(to);
                        from.delete();

                    }
                    //you can create a new file name "test.jpg" in sdcard folder.
                    File f = new File(Environment.getExternalStorageDirectory()+ File.separator + "1.jpg");
                    try {
                        f.createNewFile();

                        //write the bytes in file
                        FileOutputStream fo = new FileOutputStream(f);
                        fo.write(bytes.toByteArray());

                        // remember close de FileOutput
                        fo.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }
            return mImageBitmap;
        }
    };
    handler.postDelayed(runnable, 2000); //for initial delay..
4

2 に答える 2

3

ここで、記憶をスムーズにするのに役立ついくつかのことがあります。

  1. ビットマップの処理が完了したら、bitmap.recycle() を実行します。
  2. その後、それらを null に設定します。
  3. パススルーごとにガベージ コレクターを実行します。System.gc().
  4. マニフェストから大きなヒープをリクエストしていることを確認してください。android:largeHeap="true"アプリケーションに。
  5. アプリケーションのメモリ プロファイリングに取り組みます。これを達成する方法の詳細については、この回答を参照してください。
于 2013-01-31T21:07:02.553 に答える
1

問題はメモリリークではないと思います。おそらく、両方の画像を同時にメモリに保持することで、メモリの予算を超えています。各ビットマップはheight_in_px * width_in_px * 4バイトを使用するのに対し、アプリのメモリ バジェットは24...48 MB (デバイスによって異なります) であることに注意してください。

ビットマップの効率的な使用法に関する優れたチュートリアルがあります。これを使用して、問題の原因を特定し、修正します。

于 2013-01-31T21:14:07.613 に答える