7

我正在尝试创建一个基本的“查找我最近的...”应用程序作为学习练习。

我有 3 个活动:

  1. 一个主活动,用户输入他的地址,这通过意图将一组对象传递给列表活动。
  2. 一个列表活动,显示最近的 10 个地点。当一个项目被选中时,这会将一个对象通过意图传递给细节活动。
  3. 详细信息活动,显示所选地点的详细信息。

我的大部分东西都在工作,但是我注意到一个小问题,它与后退按钮(在我的手机屏幕底部)和向上按钮(操作栏)不同。在研究了两者之间的差异之后,看起来它们在这种情况下的行为应该相同。当点击后退按钮时,一切都按预期工作,我们仍然看到相同的列表。但是,当点击向上按钮时,它似乎以不同的方式启动/恢复列表活动,并且列表中的数据已经消失。

谁能向我解释为什么会发生这种情况以及如何防止它?

我的代码如下:(欢迎对我的 Android / Java 技术提出任何批评!)

MainActivity.java

public class MainActivity extends Activity {

    public final static String TRACKS = "com.example.firstapp.TRACKS";

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

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }

    /** Called when the user clicks the 'Search' button */
    public void search(View view) {

        // Get Address Text
        // EditText editText = (EditText) findViewById(R.id.edit_address);
        // String address = editText.getText().toString();

        // Get tracks near to address
        Track[] trackArray = new Track[20];

        for(int i=0; i < trackArray.length; i++) {
            trackArray[i] = new Track("Track " + i, "Address" + i, (float) Math.random()*10);
        }

        // Fire intent to show results
        Intent intent = new Intent(this, TrackListActivity.class);
        intent.putExtra(TRACKS, trackArray);
        startActivity(intent);
    }

}

TrackListActivity.java

public class TrackListActivity extends ListActivity {

    public final static String TRACK = "com.mycroft.runtrackdir.TRACK";

    @Override
    protected void onCreate(Bundle savedInstanceState){
        super.onCreate(savedInstanceState);
        Intent intent = getIntent();

        if (intent.hasExtra(MainActivity.TRACKS)) {
            Parcelable[] tempTrackArray = intent.getParcelableArrayExtra(MainActivity.TRACKS);        

            // Cast Parcelable to Track
            Track[] trackArray = new Track[tempTrackArray.length];
            for(int i=0; i < trackArray.length; i++) {
                trackArray[i] = (Track) tempTrackArray[i];
            }

            ArrayAdapter<Track> arrayAdapter = new ArrayAdapter<Track>(this, android.R.layout.simple_list_item_1, trackArray);
            setListAdapter(arrayAdapter);
        }
    }

    @Override
    protected void onListItemClick (ListView l, View v, int position, long id) {
        Track clickedTrack = (Track) l.getItemAtPosition(position);

        // Fire intent to show detail
        Intent intent = new Intent(this, TrackDetailActivity.class);
        intent.putExtra(TRACK, clickedTrack);
        startActivity(intent);
    }
}

TrackDetailActivity.java

public class TrackDetailActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        Intent intent = getIntent();
        Track track = intent.getParcelableExtra(TrackListActivity.TRACK);

        setContentView(R.layout.activity_track_detail);

        TextView track_name = (TextView) findViewById(R.id.track_name);
        TextView track_address = (TextView) findViewById(R.id.track_address);
        TextView track_distance = (TextView) findViewById(R.id.track_distance);

        track_name.setText(track.name);
        track_address.setText(track.address);
        track_distance.setText(track.distance.toString());

    }
}

清单.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.firstapp"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="17" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="com.example.firstapp.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity
            android:name="com.example.firstapp.TrackListActivity"
            android:label="@string/title_activity_track_list"
            android:parentActivityName="com.example.firstapp.MainActivity" >
            <meta-data
                android:name="android.support.PARENT_ACTIVITY"
                android:value="com.example.firstapp.MainActivity" />
        </activity>
        <activity
            android:name="com.example.firstapp.TrackDetailActivity"
            android:label="@string/title_activity_track_detail"
            android:parentActivityName="com.example.firstapp.TrackListActivity" >
            <meta-data
                android:name="android.support.PARENT_ACTIVITY"
                android:value="com.example.firstapp.TrackListActivity" />
        </activity>
    </application>

</manifest>
4

3 回答 3

4

我认为你应该修改你的Manifest.xml文件并告诉你Manifest file你的活动之间的关系。这应该可以解决问题。

例如:

<application ... >
...
<!-- The main/home activity (has no parent activity) -->
<activity
    android:name="com.example.myfirstapp.MainActivity" ...>
    ...
</activity>
<!-- A child of the main activity -->
<activity
    android:name="com.example.myfirstapp.DisplayMessageActivity"
    android:label="@string/title_activity_display_message"
    android:parentActivityName="com.example.myfirstapp.MainActivity" >
    <!-- Parent activity meta-data to support API level 7+ -->
    <meta-data
        android:name="android.support.PARENT_ACTIVITY"
        android:value="com.example.myfirstapp.MainActivity" />
</activity>

您也可以查看文档。

编辑:
如果您在某些活动中收到空意图,您也可以使用数据持久性。对于 Android 中的数据持久性,您可以使用Shared PreferencesSQLite DatabaseSerialization

在学习 Android 开发时,您可以简单地将trackArray值放在应用程序的 Shared Preferences 中,同时从 Main Activity 启动 List Activity 并trackArray在 List Activity 中加载来自 Shared Preferences的数据onResume并将其填充到onCreate. 这样,您的 Action bar 的向上按钮不会使 ListView Activity 的数据消失。

PS:共享首选项的实际目的是存储大小较小且偶尔更改的值,例如特定应用程序的设置数据。因此,请不要滥用它。

于 2013-08-08T21:13:18.047 回答
1
  • 向上按钮和返回按钮之间的区别。

后退按钮 - 后退按钮将您带到您之前处于活动状态的最后一个活动。

向上按钮 - 向上按钮以分层或祖传方式将您提升一级。无论您的最后一个活动活动是什么,您将始终被重定向到您在清单中设置的父活动。

尽管 Back 和 Up Button 在许多情况下几乎做同样的事情,但他们在幕后做了完全不同的步骤。

于 2017-12-06T04:57:54.957 回答
0

将 launchMode = “singleTop” 添加到清单文件中的 TrackListActivity

<activity
        android:name="com.example.firstapp.TrackListActivity"
        android:label="@string/title_activity_track_list"
        android:launchMode="singleTop"
        android:parentActivityName="com.example.firstapp.MainActivity" >
        <meta-data
            android:name="android.support.PARENT_ACTIVITY"
            android:value="com.example.firstapp.MainActivity" />
</activity>

如果您不想在按下向上按钮时杀死并重新启动 Activity,则需要使用 launchMode="singleTop" 。

于 2019-02-23T09:14:45.957 回答