21

特定のデバイスで奇妙な問題に直面するまで、私のアプリは通常問題なく動作します。アプリには2つのアクティビティがあります。ActivityA 内で ActivityB を開始すると、ActivityA は問題なく開始されます。ただし、ハードウェア ボタンを押し戻すか、finish(); を呼び出して ActivityA に戻った後。ActivityB の closeButton の内部で、ActivityA が自身をリロードします。再び onCreate() をトリガーし、その内容をすべてリロードします。そして、私は電話の向きを変えていません。この奇妙な動作は、アプリのダウンロードが 1,000 回を超える 15 台の携帯電話でのみ発生します。

この問題は、Galaxy S3 Android OS 4.1.2 でのみ発生します。そして、これも奇妙です。

なぜこれが起こっているのか分かりますか?

次のように、ボタンリスナー内で新しいアクティビティを開始すると:

ActivityA.java (MesajlarListViewActivity)

    public class MesajlarListViewActivity extends TrackedActivity {

    Context context = null;

    // contacts JSONArray
    JSONArray contacts = null;

    ArrayList<Message> productArray = new ArrayList<Message>();

    private ProductAdapter adapter;
    private ListView productList;
    private Runnable viewOrders;
    private HoloProgressIndicator profilInfoProgress = null;

    ImageView kapatButton = null;

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

        context = this;

        kapatButton = (ImageView) findViewById(R.id.kapat_button);
        /* kapat button onclick listener. */
        // =================================================================================================================
        kapatButton.setOnClickListener(new View.OnClickListener() {
            public void onClick(View view)
            {
                // Set vibration on touch.
                KnetGenericClass.vibratePhone(context);

                finish();
            }

        });
        // =================================================================================================================
        //Progress bar.
        profilInfoProgress = (HoloProgressIndicator) findViewById(R.id.profil_info_progress);

        // cheking internet connectivity.
        if(KnetGenericClass.checkInternetConnection(context))
        {
            // start task!
            /* internet var ise web service baglantisi kurmaya baslayabiliriz. */
            startActivityIndicatorWithThread();
        }
        else
        {
            KnetGenericClass.printErrorMessage(context, "Bağlantı Hatası",
                    "Lütfen internet bağlantınızı kontrol ediniz.");
        }

        productList = (ListView) findViewById(R.id.product_list);
        adapter = new ProductAdapter(this, R.layout.message_row, productArray);
        productList.setAdapter(adapter);

        // When user click a view on list view new page is appearing.
        productList.setOnItemClickListener(new OnItemClickListener() {
            public void onItemClick(AdapterView<?> parent, View view, int position, long id)
            {

                // Set vibration on touch.
                KnetGenericClass.vibratePhone(context);

                /* Navigate to message detay activity class with ilan ID. */
                Intent myIntent = new Intent(view.getContext(), MesajDetayActivity.class);
                myIntent.putExtra("messageID", productArray.get(position).getId());
                startActivity(myIntent);

                // setting image of clicked message null.
                RelativeLayout relativeLayout = (RelativeLayout) view;
                ImageView unreadedImageView = (ImageView) relativeLayout.findViewById(R.id.unreaded_image);
                unreadedImageView.setImageResource(0);
            }
        });
    }

    public class ProductAdapter extends ArrayAdapter<Message> {
        ArrayList<Message> items;

        public ProductAdapter(Context context, int textViewResourceId, ArrayList<Message> objects) {
            super(context, textViewResourceId, objects);
            this.items = objects;
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent)
        {
            if(convertView == null)
            {
                LayoutInflater vi = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);
                convertView = vi.inflate(R.layout.message_row, null);
            }

            ImageView unreadedImageView = (ImageView) convertView.findViewById(R.id.unreaded_image);
            TextView productName = (TextView) convertView.findViewById(R.id.product_name);
            TextView productDetail = (TextView) convertView.findViewById(R.id.product_detail);
            // TextView productDate = (TextView)
            // convertView.findViewById(R.id.product_date);
            TextView sentDate = (TextView) convertView.findViewById(R.id.product_date);

            productName.setText(items.get(position).getSender());
            productDetail.setText(items.get(position).getTitle());
            // String bodyNoHTML = items.get(position).getBody();

            if(items.get(position).getIsReaded())
            {
                unreadedImageView.setImageResource(0);
            }
            else
            {
                unreadedImageView.setImageResource(R.drawable.bluedot);
            }

            String dateStr = items.get(position).getSentDate();
            try
            {
                sentDate.setText(dateStr.substring(6, 8) + "." + dateStr.substring(4, 6) + "." + dateStr.substring(0, 4)
                        +" "+dateStr.substring(8, 10)+":"+dateStr.substring(10, 12));
            }
            catch(Exception e)
            {
                sentDate.setText("");
            }


            return convertView;
        }

    }// @end of product adapter class.

    /* web service'e baglanti kurulan methodu threadin icerisinde cagiriyoruz. */
    public void startActivityIndicatorWithThread()
    {
        // ==============================================================================================
        // getting ilan details into arraylist.
        // setting up thread.
        viewOrders = new Runnable() {
            public void run()
            {
                getMessageListFromWebService();
            }
        };
        Thread thread = new Thread(null, viewOrders, "MagentoBackground");
        thread.start();
        profilInfoProgress.start();
        // ==============================================================================================
        // @end of the thread declaration.
    }

    public void getMessageListFromWebService()
    {
        // Creating JSON Parser instance
        JSONParser jParser = new JSONParser(context);

        // getting JSON string from URL
        JSONArray jsonArray = jParser.getAuthorizedInfoFromUrlToJSONArray(
                WebServiceInfo.getKnetWebServiceLink()+"/API/Member/GetInboxMessageList", MainActivity.getAccessToken());

        // if json is null then there is a problem.
        if(jsonArray == null)
        {
            // if json array is null then print error message.
            runOnUiThread(showAlertMessage);
            runOnUiThread(returnRes);
            return;
        }

        try
        {
            // Eger aranilan kritere gore ilan yok ise hata mesaji basiyoruz.
            if(jsonArray.length() == 0)
            {
                // if json array is null then print error message.
                runOnUiThread(showAlertIlanYokMessage);
                runOnUiThread(returnRes);
                return;
            }

            // looping through All Contacts
            for (int i = 0; i < jsonArray.length(); i++)
            {
                JSONObject c = jsonArray.getJSONObject(i);

                // Storing each json item in variable
                // String id = c.getString(TAG_ID);
                String id = c.getString("Id");
                String sender = c.getString("Sender");
                // String body = c.getString("Body");
                String title = c.getString("Title");
                String sentDate = c.getString("SentDate");
                Boolean isReaded = c.getBoolean("IsRead");

                Message productObject = new Message(id, sender, "", title, sentDate, isReaded);
                productArray.add(productObject);
            }
        }
        catch (Exception e)
        {
            Log.e("BACKGROUND_PROC", e.getMessage());
        }
        runOnUiThread(returnRes);
    }


    // @end of thread.
    private Runnable returnRes = new Runnable() {

        public void run()
        {
            profilInfoProgress.stop();
            adapter.notifyDataSetChanged();// refreshing data over adapter in
                                            // list view.
        }
    };

    // @end of thread.
    private Runnable showAlertMessage = new Runnable() {

        public void run()
        {
            // Bu hata genelde linkteki problemden, servera ulasilamamasindan
            // veya timeouttan meydana gelir.
            Toast.makeText(getApplicationContext(),
                    "Mesajlar alınamadı lütfen daha sonra tekrar deneyiniz.", 
                    Toast.LENGTH_LONG).show();
        }
    };

    private Runnable showAlertIlanYokMessage = new Runnable() {

        public void run()
        {
            // Bu hata aranilan kelimeye gore ilan bulunamazsa gelir.
            Toast.makeText(getApplicationContext(),
                    "Mesajlar bulunamadı.", 
                    Toast.LENGTH_LONG).show();
        }
    };

}

================================================== ======================

ActivityB.java (MesajDetayActivity.java)

public class MesajDetayActivity extends TrackedActivity {

    private HoloProgressIndicator profilInfoProgress = null;

    TextView titleTextView = null;
    TextView senderTextView = null;
    TextView dateTextView = null;
    WebView bodyWebView = null;

    Message messageObject = null;

    String messageID = null;

    ImageView kapatButton = null;

    Context context;

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

        context = this;

        kapatButton = (ImageView) findViewById(R.id.kapat_button);
        /* kapat button onclick listener. */
        // =================================================================================================================
        kapatButton.setOnClickListener(new View.OnClickListener() {
            public void onClick(View view)
            {
                // Set vibration on touch.
                KnetGenericClass.vibratePhone(context);

                finish();
            }

        });
        // =================================================================================================================
        //Progress bar.
        profilInfoProgress = (HoloProgressIndicator) findViewById(R.id.profil_info_progress);

        Bundle extras = getIntent().getExtras();
        if(extras != null)
        {
            messageID = extras.getString("messageID");
        }

        titleTextView = (TextView) findViewById(R.id.title_textview);
        senderTextView = (TextView) findViewById(R.id.sender_textview);
        dateTextView = (TextView) findViewById(R.id.date_textview);
        bodyWebView = (WebView) findViewById(R.id.mesaj_webView);

        // Show the ProgressDialog on this thread
        profilInfoProgress.start();

        // Start a new thread that will download all the data
        new MakeItTask().execute();

    }

    // Async task.
    private class MakeItTask extends AsyncTask<String, Void, Object> {
        protected Object doInBackground(String... args)
        {
            Log.i("MyApp", "Background thread starting");

            // This is where you would do all the work of downloading your data
            // getting message detay
            /* connect to web service */
            getMessageDetayFromWebService();

            return null;
        }

        protected void onPostExecute(Object result)
        {
            // Pass the result data back to the main activity
            // TakipListeActivity.this.data = result;
            try
            {
                titleTextView.setText("Başlık: " + messageObject.getTitle());
                senderTextView.setText("Gönderen: " + messageObject.getSender());
                dateTextView.setText("Tarih: " + messageObject.getSentDate().substring(6, 8) + "."
                        + messageObject.getSentDate().substring(4, 6) + "."
                        + messageObject.getSentDate().substring(0, 4));

                if(!messageObject.getBody().contains("img"))
                {
                    bodyWebView.loadDataWithBaseURL(null, messageObject.getBody(), "text/html", "UTF-8", null);
                }

            }
            catch (Exception e)
            {
                Log.e(CONNECTIVITY_SERVICE, "Mesaj Detayi bilgileri basilamadi.");
            }

            profilInfoProgress.stop();
        }
    }

    /* web service'e baglanti kurulan methodu threadin icerisinde cagiriyoruz. */
    public void getMessageDetayFromWebService()
    {
        // Creating JSON Parser instance
        JSONParser jParser = new JSONParser(context);

        // getting JSON string from URL
        JSONObject jsonObject = jParser.getAuthorizedInfoFromUrlToJSONObject(
                WebServiceInfo.getKnetWebServiceLink()+"/API/Member/GetInboxMessage/" + messageID, MainActivity.getAccessToken());

        // if json is null then there is a problem.
        if(jsonObject == null)
        {
            return;
        }

        try
        {
            String title = jsonObject.getString("Title");
            String id = jsonObject.getString("Id");
            String sender = jsonObject.getString("Sender");
            String date = jsonObject.getString("SentDate");
            String body = jsonObject.getString("Body");

            messageObject = new Message(id, sender, body, title, date, true);

        }
        catch (Exception e)
        {
            Log.e("BACKGROUND_PROC", e.getMessage());
        }

    }// @end of getIlanDetayFromWebService.

}

編集: この問題は、これら 2 つのアクティビティだけでなく、一部の電話ではすべてのアクティビティが同じ動作をします。

4

13 に答える 13

48

[設定] > [システム] > [開発者向けオプション] > [アプリ] で [アクティビティを保持しない]が有効になっているかどうかを確認します。

于 2013-03-01T22:57:58.007 に答える
11

アクティビティのドキュメント ( http://developer.android.com/reference/android/app/Activity.html ) には、バックグラウンド アクティビティのライフサイクルについて次のように記載されています。

バックグラウンド アクティビティ (ユーザーには表示されず、一時停止されたアクティビティ) はもはや重要ではないため、システムはそのプロセスを安全に強制終了して、他のフォアグラウンドまたは表示されているプロセスのためにメモリを再利用できます。そのプロセスを強制終了する必要がある場合、ユーザーがアクティビティに戻ったときに (再び画面に表示されるように)、その onCreate(Bundle) メソッドが、以前に onSaveInstanceState(Bundle) で提供されていた savedInstanceState で呼び出されます。ユーザーが最後に終了したときと同じ状態で再起動できます。

つまり、ActivityB がアクティブである間に、ActivityA がオペレーティング システムによって破棄される場合と破棄されない場合があるため、この状況はコードで処理する必要があります。ActivityA が破棄された場合、ユーザーが ActivityB で戻るボタンを押すと、onCreate(Bundle) が呼び出されます。

于 2013-02-21T12:59:11.433 に答える
9

「アクティビティを保持しない」という Android 開発者向け設定があります。このオプションの説明は、「ユーザーが離れるとすぐにすべてのアクティビティを破棄する」です。これは、あなたが見ていることの良い説明のように思えます.少数の携帯電話でしか見られないので、これがデフォルト以外のシステム設定によって引き起こされているという考えはもっともらしい.

最適ではなくても、このシナリオでもアプリが機能することが理想的です。ただし、この設定がアプリの問題である場合は、ユーザーのためにこの問題を文書化することをお勧めします。

于 2013-03-01T23:18:43.437 に答える
7

launchmodeAndroid マニフェストで を変更してみましたか? これを Activity 宣言に追加してみてください。

android:launchMode="singleTask"

次に、startActivityForResultの代わりに を使ってみてくださいstartActivityonActivityResult(int, int, Intent)これにより、アクティビティ B が終了したときにアクティビティ A がそのメソッドを強制的に呼び出すようになりonCreateます。次に、アクティビティ A で、何かを行うメソッドを実装します (デバッグ ステートメントの出力など)。

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data)
{
    Log.i("Test", "Did this work???");
    //TODO send notification to your server to verify this works?
}
于 2013-02-27T02:32:47.570 に答える
5

この動作に問題はありません。

の状態を保持したい場合ActivityAは、メソッドonSaveInstanceStateとを使用してonRestoreInstanceStateください。詳細については、 http://developer.android.com/reference/android/app/Activity.html#ActivityLifecycleのアクティビティライフサイクルを参照してください。

より深い理解については、https://stackoverflow.com/a/10492967/332210も参照してください。

于 2013-03-01T03:57:00.507 に答える
3

レイアウトを提供する1つのことを試しonCreate()て、残りの作業をで行うことができますonStart()か?それがうまくいくなら??

お気に入り:

 public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.show);
   }

      @Override
       protected void onStart() {
       // TODO Auto-generated method stub
    super.onStart();
    Log.i(TAG, "On Start .....");

      }

アクティビティライフサイクルを参照してください ここに画像の説明を入力してください

于 2013-03-01T05:07:43.893 に答える
1

おそらくあなたは使うべきです

Intent startIntent = new Intent(view.getContext(), ActivityB.class); 
startActivity(startIntent); 
finish() ;

Intent startIntent = new Intent(view.getContext(), ActivityA.class); 
startActivity(startIntent); 
finish() ;

戻るか進むたびに。

于 2013-02-21T14:50:05.793 に答える
1

アクティビティ A で onStart() および onResume メソッドをオーバーライドし、問題が解決しないかどうかを確認します。可能であれば、アクティビティ A と B のコードをここに入力してください。

于 2013-03-01T06:10:28.097 に答える
1

アクティビティ A は、レイアウト R.layout.mesajlar_list を使用します

アクティビティ B は、レイアウト R.layout.mesajdetaylari を使用します

ただし、どちらにも次のコード行があります。

kapatButton = (ImageView) findViewById(R.id.kapat_button);

R.id.kapat_button はどのレイアウトにありますか? 異なるレイアウトで同じ ID を使用することは、非常に危険です。それがあなたが見ているものを引き起こしていることを保証することはできませんが、それは奇妙な動作を引き起こす可能性のあるものです.

于 2013-03-01T19:06:50.977 に答える
1

android:launchMode="standard"それも正確な問題に直面し、 in activityofを使用して問題を解決しましたmanifest

于 2013-02-28T07:50:52.900 に答える