提问者:小点点

当向上导航到父activity时,回收器视图崩溃


我对Android,Java和Kotlin都很陌生,我正在尝试将我的应用程序移植到Swift上,它最初是为iOS编写的。

应用程序使用的数据是我的Speaker对象的ArrayList,来自一个JSON文件。

我的MainActivity显示一个由每个扬声器的“brand”参数填充的RecyclerView。 每一行的onClick启动BrandMenuActivity,并使用一个带有该品牌所有扬声器的RecycerView。 然后,它们的onClick显示了一个DetailView。

当使用模拟器中的“Hardware”back按钮导航时,一切都很正常,但是如果我尝试在detailView的操作栏中实现setDisplayHomeAsUpEnabled,应用程序就会崩溃,因为BrandMenuActivity的意图现在为null。

我的AndroidManifest:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.speakerskotlin">

<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:roundIcon="@mipmap/ic_launcher_round"
    android:supportsRtl="true"
    android:theme="@style/AppTheme">
    <activity android:name=".SpeakerDetailActivity"
        android:parentActivityName=".BrandMenuActivity">
    </activity>
    <activity android:name=".BrandMenuActivity"
        android:parentActivityName=".MainActivity" >
    </activity>
    <activity android:name=".MainActivity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
</application>

主要活动:

class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_main)

    setSupportActionBar(findViewById(R.id.speakerToolbar))
    supportActionBar?.title = "Speakers"
    
    val jsonFileString = getJsonDataFromAsset(applicationContext, "SpeakerData.json")

    val gson = Gson()
    val listSpeakerType = object : TypeToken<ArrayList<Speaker>>() {}.type
    val allSpeakers: ArrayList<Speaker> = gson.fromJson(jsonFileString, listSpeakerType)

    brandsRecView.layoutManager = LinearLayoutManager(this)
    brandsRecView.adapter = SpeakerBrandAdapter(allSpeakers){
        val intent = Intent(this, BrandMenuActivity::class.java)
        intent.putExtra("allSpeakers", allSpeakers)
        intent.putExtra("brand", it)
        startActivity(intent)
    }

}

品牌菜单活动:

    class BrandMenuActivity() : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_brand_menu)

    setSupportActionBar(findViewById(R.id.brandToolbar))
    supportActionBar?.title = intent.getStringExtra("brand")
    supportActionBar?.setDisplayHomeAsUpEnabled(true)

    val brand = intent.getStringExtra("brand")
    val allSpeakers: ArrayList<Speaker> = intent.getParcelableArrayListExtra("allSpeakers")

    brandMenuRecView.layoutManager = LinearLayoutManager(this.applicationContext)
    brandMenuRecView.adapter = BrandMenuAdapter(brand, allSpeakers) {
        val intent = Intent(this, SpeakerDetailActivity::class.java)
        intent.putExtra("speaker", it)
        startActivity(intent)
    }
}

详细信息活动:

    class SpeakerDetailActivity: AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    setContentView(R.layout.activity_speaker_detail)

    setSupportActionBar(findViewById(R.id.detailToolbar))
    supportActionBar?.setDisplayHomeAsUpEnabled(true)

    val speaker = intent.getParcelableExtra<Speaker>("speaker")

    val modelImage = findViewById<ImageView>(R.id.imgSpeakerImage)
    val brandText = findViewById<TextView>(R.id.txtBrandName)
    val modelText = findViewById<TextView>(R.id.txtModel)

    brandText.text = speaker.brand
    modelText.text = speaker.model
    val resID = resources.getIdentifier(speaker.imgString, "drawable", packageName)
    modelImage.setImageResource(resID)

    speakerDetailRecView.layoutManager = LinearLayoutManager(this.applicationContext)
    speakerDetailRecView.adapter = SpeakerDetailAdapter(speaker)
}

崩溃时的日志:

    E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.example.speakerskotlin, PID: 12099
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.speakerskotlin/com.example.speakerskotlin.BrandMenuActivity}: java.lang.IllegalStateException: intent.getParcelableArrayListExtra("allSpeakers") must not be null
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3270)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3409)
    at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:83)
    at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135)
    at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2016)
    at android.os.Handler.dispatchMessage(Handler.java:107)
    at android.os.Looper.loop(Looper.java:214)
    at android.app.ActivityThread.main(ActivityThread.java:7356)
    at java.lang.reflect.Method.invoke(Native Method)
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:492)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:930)
 Caused by: java.lang.IllegalStateException: intent.getParcelableArrayListExtra("allSpeakers") must not be null
    at com.example.speakerskotlin.BrandMenuActivity.onCreate(BrandMenuActivity.kt:20)
    at android.app.Activity.performCreate(Activity.java:7802)
    at android.app.Activity.performCreate(Activity.java:7791)
    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1299)
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:3245)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3409) 
    at android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:83) 
    at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:135) 
    at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:95) 
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2016) 
    at android.os.Handler.dispatchMessage(Handler.java:107) 
    at android.os.Looper.loop(Looper.java:214) 
    at android.app.ActivityThread.main(ActivityThread.java:7356) 
    at java.lang.reflect.Method.invoke(Native Method) 
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:492) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:930) 

共1个答案

匿名用户

有两种方法。 首先,您可以尝试在点击android.r.id.home时调用finish()

override fun onOptionsItemSelected(item: MenuItem): Boolean {
    if (item.itemId == android.R.id.home) {
        finish()
        return true
    }
    return super.onOptionsItemSelected(item)
}

这只是恢复您的BrandMenuActivity,这样onCreate就不会被调用。

第二个问题是,如果您希望从SpeakerDetailActivity中获得相同的意图数据,那么只需在活动之间传递数据,比如getIntent()。getExtras()在BackStack activity解决方案Android教程中返回null