Я адаптировал этот учебник , чтобы составить список «избранных» элементов для записи и чтения из общих настроек. В моем действии DetailView есть кнопка для добавления / удаления элемента из этого списка, и, похоже, здесь проблема. Я всегда получаю исключение NullPointerException при запуске этого действия. Я попытался сделать список по умолчанию пустым массивом, когда он не может найти данные, но при доступе он всегда, кажется, возвращает значение null.
Мой класс FavoriteSpeakers:
class FavoriteSpeakers (context: Context) {
val PREFS_FILENAME = "com.soundspecs.prefs"
val FAVORITE_SPEAKERS = "favorite_speakers"
val prefs: SharedPreferences = context.getSharedPreferences(PREFS_FILENAME, 0);
var favoriteSpeakers: ArrayList<String>
get() {
if (loadFavorites() == null) {
return ArrayList()
} else {
return loadFavorites()
}
}
set(list: ArrayList<String>){
if (favoriteSpeakers == null) {
favoriteSpeakers = arrayListOf()
}
}
fun contains (speaker: Speaker): Boolean {
return favoriteSpeakers.contains(speaker.model)
}
fun addToFavorites(speaker: Speaker) {
favoriteSpeakers.add(speaker.model)
save(favoriteSpeakers)
}
fun removeFromFavorites(speaker: Speaker) {
if (favoriteSpeakers.contains(speaker.model)) {
favoriteSpeakers.remove(speaker.model)
}
save(favoriteSpeakers)
}
fun save(list: ArrayList<String>) {
val editor: SharedPreferences.Editor = prefs.edit()
val gson = Gson()
val json: String = gson.toJson(list)
editor.putString(FAVORITE_SPEAKERS, json)
editor.apply()
}
fun loadFavorites(): ArrayList<String> {
val gson = Gson()
val json: String? = this.prefs.getString(FAVORITE_SPEAKERS, null)
val type: Type = object : TypeToken<ArrayList<String?>?>() {}.getType()
return gson.fromJson(json, type)
}
}
Расширение My App Class:
val favoriteSpeakersClass: FavoriteSpeakers by lazy {
App.favoriteSpeakersClass!!
}
class App: Application() {
companion object {
var favoriteSpeakersClass: FavoriteSpeakers? = null
}
override fun onCreate() {
favoriteSpeakersClass = FavoriteSpeakers(applicationContext)
super.onCreate()
}
}
Соответствующие части моего подробного представления:
class SpeakerDetailActivity: AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_speaker_detail)
intent.getExtras()?.isEmpty()
var 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)
val favoritesButton = findViewById<ImageButton>(R.id.btnFavorite)
if (favoriteSpeakers!= null) {
if (favoriteSpeakersClass.favoriteSpeakers.contains(speaker.model)) {
favoritesButton.setImageResource(R.drawable.outline_star_24)
} else {
favoritesButton.setImageResource(R.drawable.outline_star_border_24)
}
} else {
favoritesButton.setImageResource(R.drawable.outline_star_border_24)
}
btnFavorite.setOnClickListener {
if (favoriteSpeakers != null) {
if (!favoriteSpeakersClass.contains(speaker)) {
favoriteSpeakersClass.addToFavorites(speaker)
} else {
favoriteSpeakersClass.removeFromFavorites(speaker)
}
} else {
favoriteSpeakers?.addToFavorites(speaker)
}
}
И журнал при возникновении исключения при запуске SpeakerDetail Activity:
E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.example.speakerskotlin, PID: 22270
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.speakerskotlin/com.example.speakerskotlin.SpeakerDetailActivity}: kotlin.KotlinNullPointerException
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: kotlin.KotlinNullPointerException
at com.example.speakerskotlin.AppKt$favoriteSpeakersClass$2.invoke(App.kt:7)
at com.example.speakerskotlin.AppKt$favoriteSpeakersClass$2.invoke(Unknown Source:0)
at kotlin.SynchronizedLazyImpl.getValue(LazyJVM.kt:74)
at com.example.speakerskotlin.AppKt.getFavoriteSpeakersClass(Unknown Source:2)
at com.example.speakerskotlin.SpeakerDetailActivity.onCreate(SpeakerDetailActivity.kt:45)
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)