- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试制作一个用户可以登录并将其用户数据保存到 mysql-db 的应用程序。我正在使用 PreferenceActivity 和 PreferenceFragment 来处理这个问题。通常这工作正常,直到我改变用户......
现在当用户 A 注销和用户 B 登录时发生错误。(共享首选项在注销时使用 clear() 和 commit() 清除..)
应用程序停止
java.lang.NullPointerException: Attempt to invoke virtual method 'java.io.File android.content.Context.getCacheDir()'
on a null object reference
at com.android.volley.toolbox.Volley.newRequestQueue(Volley.java: 43)
at com.android.volley.toolbox.Volley.newRequestQueue(Volley.java: 78)
at com.jamp.jamp.UserProfileSettingsFragment.onSharedPreferenceChanged(UserProfileSettingsFragment.java: 171)
at android.app.SharedPreferencesImpl$EditorImpl.notifyListeners(SharedPreferencesImpl.java: 479)
at android.app.SharedPreferencesImpl$EditorImpl.apply(SharedPreferencesImpl.java: 387)
at android.preference.Preference.tryCommit(Preference.java: 1415)
at android.preference.Preference.persistString(Preference.java: 1448)
at android.preference.EditTextPreference.setText(EditTextPreference.java: 94)
at android.preference.EditTextPreference.onDialogClosed(EditTextPreference.java: 148)
at android.preference.DialogPreference.onDismiss(DialogPreference.java: 395)
at android.app.Dialog$ListenersHandler.handleMessage(Dialog.java: 1337)
at android.os.Handler.dispatchMessage(Handler.java: 102)
at android.os.Looper.loop(Looper.java: 154)
at android.app.ActivityThread.main(ActivityThread.java: 6077)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java: 866)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java: 756)
现在这是 fragment 的代码
package com.jamp.jamp;
import android.content.Context;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.preference.Preference;
import android.preference.PreferenceFragment;
import android.preference.PreferenceManager;
import android.support.v7.app.AlertDialog;
import android.widget.Toast;
import com.android.volley.RequestQueue;
import com.android.volley.Response;
import com.android.volley.toolbox.Volley;
import org.json.JSONException;
import org.json.JSONObject;
import java.io.IOException;
import java.util.HashMap;
/**
* Created by Erikken on 05.09.2017.
*/
public class UserProfileSettingsFragment extends PreferenceFragment implements SharedPreferences.OnSharedPreferenceChangeListener {
SessionManager session;
SharedPreferences prefs;
String prefs_realname;
String prefs_email;
String prefs_streetname;
String prefs_postcode;
String prefs_city;
String prefs_state;
@Override
public void onCreate(final Bundle savedInstanceState) {
prefs = this.getActivity().getSharedPreferences("JampSharedPrefs", Context.MODE_PRIVATE);
final SharedPreferences.Editor editor = prefs.edit();
PreferenceManager.getDefaultSharedPreferences(getActivity()).registerOnSharedPreferenceChangeListener(this);
super.onCreate(savedInstanceState);
addPreferencesFromResource(R.xml.usersettings);
session = new SessionManager(this.getActivity().getApplicationContext());
HashMap < String, String > user = session.getUserDetails();
final String sessionUsername = user.get(SessionManager.KEY_USERNAME);
// ResponseListener um Request Nutzerdaten auszulesen.
Response.Listener < String > UserDataResponseListener = new Response.Listener < String > () {
@Override
public void onResponse(String response) {
try {
JSONObject jsonResponse = new JSONObject(response);
boolean success = jsonResponse.getBoolean("success");
// Wenn Datenabfrage erfolgreich, JSONResponse auswerten. // If successful, import data from mysql database
if (success) {
String responseRealName = jsonResponse.getString("realname");
if (responseRealName.equals("")) {
responseRealName = getResources().getString(R.string.MissingRealName);
}
String responseEmail = jsonResponse.getString("email");
if (responseEmail.equals("")) {
responseEmail = getResources().getString(R.string.MissingEmail);
}
String responseStreetName = jsonResponse.getString("streetname");
if (responseStreetName.equals("")) {
responseStreetName = getResources().getString(R.string.MissingStreetName);
}
String responsePostcode = jsonResponse.getString("postcode");
if (responsePostcode.equals("")) {
responsePostcode = getResources().getString(R.string.MissingPostcode);
}
String responseCity = jsonResponse.getString("city");
if (responseCity.equals("")) {
responseCity = getResources().getString(R.string.MissingCity);
}
String responseState = jsonResponse.getString("state");
if (responseState.equals("")) {
responseState = getResources().getString(R.string.MissingState);
}
int responseAge = jsonResponse.getInt("age");
int responseIsPremium = jsonResponse.getInt("isPremium"); // BOOLEAN
// Add data to shared prefs
editor.putString("realname", responseRealName);
editor.putString("email", responseEmail);
editor.putString("streetname", responseStreetName);
editor.putString("postcode", responsePostcode);
editor.putString("city", responseCity);
editor.putString("state", responseState);
editor.commit();
prefs_realname = prefs.getString("realname", "");
prefs_email = prefs.getString("email", "");
prefs_streetname = prefs.getString("streetname", "");
prefs_postcode = prefs.getString("postcode", "");
prefs_city = prefs.getString("city", "");
prefs_state = prefs.getString("state", "");
Preference prefUserData = (Preference) findPreference("preferencescreen_userdata");
prefUserData.setTitle(sessionUsername);
prefUserData.setSummary(prefs_realname + "\n" +
prefs_email + "\n" +
prefs_streetname + "\n" +
prefs_postcode + " " + prefs_city + "\n" +
prefs_state);
findPreference("settings_username").setTitle(sessionUsername);
findPreference("settings_realname").setSummary(prefs_realname);
findPreference("settings_email").setSummary(prefs_email);
findPreference("settings_streetname").setSummary(prefs_streetname);
findPreference("settings_postcode").setSummary(prefs_postcode);
findPreference("settings_city").setSummary(prefs_city);
findPreference("settings_state").setSummary(prefs_state);
} else {
AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
builder.setMessage("Konnte Nutzerdaten nicht abrufen.")
.setNegativeButton("Nochmal", null)
.create()
.show();
}
} catch (JSONException e) {
e.printStackTrace();
}
}
};
// Request an userdatarequest.php senden / Send request to userdatarequest.php
UserDataRequest userDataRequest = new UserDataRequest(sessionUsername, UserDataResponseListener);
RequestQueue queue = Volley.newRequestQueue(this.getActivity());
queue.add(userDataRequest);
}
@Override
public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) {
HashMap < String, String > user = session.getUserDetails();
final String sessionUsername = user.get(SessionManager.KEY_USERNAME);
final String sessionPassword = user.get(SessionManager.KEY_PASSWORD);
Context mContext = getActivity().getApplicationContext();
Response.Listener < String > UpdateUserDataResponseListener = new Response.Listener < String > () {
@Override
public void onResponse(String response) {
try {
JSONObject jsonResponse = new JSONObject(response);
boolean success = jsonResponse.getBoolean("success");
if (success) {
Toast.makeText(getActivity().getApplicationContext(), "Change Successful", Toast.LENGTH_LONG).show();
} else {
AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
builder.setMessage("Konnte Nutzerdaten nicht abrufen.")
.setNegativeButton("Nochmal", null)
.create()
.show();
}
} catch (JSONException e) {
e.printStackTrace();
}
}
};
if (key.equals("settings_email")) {
// An mySQLDB senden
UpdateUserDataRequest updateUserDataRequest = new UpdateUserDataRequest(sessionUsername, sessionPassword, "email", sharedPreferences.getString(key, ""), UpdateUserDataResponseListener);
RequestQueue queue = Volley.newRequestQueue(mContext);
queue.add(updateUserDataRequest);
// Summary sofort erneuern. // Refresh Summaries
prefs_email = sharedPreferences.getString(key, "");
findPreference(key).setSummary(prefs_email);
findPreference("preferencescreen_userdata").setSummary(prefs_realname + "\n" +
prefs_email + "\n" +
prefs_streetname + "\n" +
prefs_postcode + " " + prefs_city + "\n" +
prefs_state);
}
// LOTS OF SIMILAR BUTTONS FOLLOWING HERE...
}
}
因此错误仅在用户重新登录后发生一次。当一切似乎都已重置时它崩溃了,应用程序会重新启动,他可以根据需要随时更改他的任何数据。
我使用的 mContext 变量错了吗?是否有一些我没有清除的剩余值会导致崩溃?
有什么建议吗? =/
最佳答案
好的,我知道了。发现其他一些有类似问题的线程。
我的 Context mContext = getActivity().getApplicationContext();
在 OnPreferenceChange 方法中调用。但这似乎还早,因为尚未调用 Fragment 中的 onCreate 方法,并且我的上下文应该指向的地方没有 Activity ,因此返回 null。我将 mContext 变量移至 onCreate 方法,将其从 onPreferenceChange 中删除(还进行了一些其他调整..),现在 onPreferenceChange 似乎在等待 onCreate 方法并且它运行良好。
package com.jamp.jamp;
import android.content.Context;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.preference.Preference;
import android.preference.PreferenceFragment;
import android.preference.PreferenceManager;
import android.support.v7.app.AlertDialog;
import android.widget.Toast;
import com.android.volley.RequestQueue;
import com.android.volley.Response;
import com.android.volley.toolbox.Volley;
import org.json.JSONException;
import org.json.JSONObject;
import java.io.IOException;
import java.util.HashMap;
/**
* Created by Erikken on 05.09.2017.
*/
public class UserProfileSettingsFragment extends PreferenceFragment implements SharedPreferences.OnSharedPreferenceChangeListener
{
SessionManager session;
Context mContext;
String prefs_realname;
String prefs_email;
String prefs_streetname;
String prefs_postcode;
String prefs_city;
String prefs_state;
@Override
public void onCreate(final Bundle savedInstanceState)
{
final SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(getActivity().getApplicationContext());
final SharedPreferences.Editor editor = prefs.edit();
prefs.registerOnSharedPreferenceChangeListener(this);
mContext = getActivity().getApplicationContext();
super.onCreate(savedInstanceState);
addPreferencesFromResource(R.xml.usersettings);
session = new SessionManager(getActivity().getApplicationContext());
session.CheckLogin();
HashMap<String,String> user = session.getUserDetails();
final String sessionUsername = user.get(SessionManager.KEY_USERNAME);
final String sessionPassword = user.get(SessionManager.KEY_PASSWORD);
// ResponseListener um Request Nutzerdaten auszulesen.
Response.Listener<String> UserDataResponseListener = new Response.Listener<String>(){
@Override
public void onResponse(String response) {
try {
JSONObject jsonResponse = new JSONObject(response);
boolean success = jsonResponse.getBoolean("success");
// Wenn Datenabfrage erfolgreich, JSONResponse auswerten. // If successful, import data from mysql database
if (success) {
String responseRealName = jsonResponse.getString("realname"); if (responseRealName.equals("")) {responseRealName = getResources().getString(R.string.MissingRealName);}
String responseEmail = jsonResponse.getString("email"); if (responseEmail.equals("")) {responseEmail= getResources().getString(R.string.MissingEmail);}
String responseStreetName = jsonResponse.getString("streetname"); if (responseStreetName.equals("")) {responseStreetName = getResources().getString(R.string.MissingStreetName);}
String responsePostcode = jsonResponse.getString("postcode"); if (responsePostcode.equals("")) {responsePostcode = getResources().getString(R.string.MissingPostcode);}
String responseCity = jsonResponse.getString("city"); if (responseCity.equals("")) {responseCity = getResources().getString(R.string.MissingCity);}
String responseState = jsonResponse.getString("state"); if (responseState.equals("")) {responseState = getResources().getString(R.string.MissingState);}
int responseAge = jsonResponse.getInt ("age");
int responseIsPremium = jsonResponse.getInt ("isPremium"); // BOOLEAN
// Add data to shared prefs
editor.putString("realname",responseRealName);
editor.putString("email",responseEmail);
editor.putString("streetname",responseStreetName);
editor.putString("postcode",responsePostcode);
editor.putString("city",responseCity);
editor.putString("state",responseState);
editor.commit();
prefs_realname = prefs.getString("realname","0");
prefs_email = prefs.getString("email","0");
prefs_streetname = prefs.getString("streetname","0");
prefs_postcode = prefs.getString("postcode","0");
prefs_city = prefs.getString("city","0");
prefs_state = prefs.getString("state","0");
Preference prefUserData = (Preference) findPreference("preferencescreen_userdata");
prefUserData.setTitle(sessionUsername);
prefUserData.setSummary(prefs_realname+"\n"
+prefs_email+"\n"
+prefs_streetname+"\n"
+prefs_postcode + " " + prefs_city +"\n"
+prefs_state);
findPreference("settings_username").setTitle(sessionUsername);
findPreference("settings_realname").setSummary(prefs_realname);
findPreference("settings_email").setSummary(prefs_email);
findPreference("settings_streetname").setSummary(prefs_streetname);
findPreference("settings_postcode").setSummary(prefs_postcode);
findPreference("settings_city").setSummary(prefs_city);
findPreference("settings_state").setSummary(prefs_state);
}else{
AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
builder.setMessage("Konnte Nutzerdaten nicht abrufen.")
.setNegativeButton("Nochmal",null)
.create()
.show();
}
} catch (JSONException e) {
e.printStackTrace();
}
}
};
// Request an userdatarequest.php senden / Send request to userdatarequest.php
UserDataRequest userDataRequest = new UserDataRequest(sessionUsername, UserDataResponseListener);
RequestQueue queue = Volley.newRequestQueue(getActivity());
queue.add(userDataRequest);
}
@Override
public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) {
session = new SessionManager(mContext);
HashMap<String,String> user = session.getUserDetails();
final String sessionUsername = user.get(SessionManager.KEY_USERNAME);
final String sessionPassword = user.get(SessionManager.KEY_PASSWORD);
Response.Listener<String> UpdateUserDataResponseListener = new Response.Listener<String>(){
@Override
public void onResponse(String response) {
try {
JSONObject jsonResponse = new JSONObject(response);
boolean success = jsonResponse.getBoolean("success");
if (success){
Toast.makeText(mContext,"Change Successful",Toast.LENGTH_LONG).show();
}else{
AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
builder.setMessage("Konnte Nutzerdaten nicht abrufen.")
.setNegativeButton("Nochmal",null)
.create()
.show();
}
} catch (JSONException e) {
e.printStackTrace();
}
}
};
if (key.equals("settings_email")){ // Adressenprüfung mussnoch implementiert werden.
// An mySQLDB senden
//CELLNAME // CELLVALUE
UpdateUserDataRequest updateUserDataRequest = new UpdateUserDataRequest(sessionUsername, sessionPassword, "email", sharedPreferences.getString(key, ""), UpdateUserDataResponseListener);
RequestQueue queue = Volley.newRequestQueue(mContext);
queue.add(updateUserDataRequest);
// Summary sofort erneuern. // Refresh Summaries instantly
prefs_email = sharedPreferences.getString(key, "");
findPreference(key).setSummary(prefs_email);
findPreference("preferencescreen_userdata").setSummary(prefs_realname + "\n"
+ prefs_email + "\n"
+ prefs_streetname + "\n"
+ prefs_postcode + " " + prefs_city + "\n"
+ prefs_state);
}
if (key.equals("settings_streetname")){
UpdateUserDataRequest updateUserDataRequest = new UpdateUserDataRequest(sessionUsername,sessionPassword,"streetname",sharedPreferences.getString(key,""), UpdateUserDataResponseListener);
RequestQueue queue = Volley.newRequestQueue(mContext);
queue.add(updateUserDataRequest);
// Summary sofort erneuern. // Refresh Summaries instantly
prefs_streetname = sharedPreferences.getString(key,"");
findPreference(key).setSummary(prefs_streetname);
findPreference("preferencescreen_userdata").setSummary(prefs_realname+"\n"
+prefs_email+"\n"
+prefs_streetname+"\n"
+prefs_postcode + " " + prefs_city +"\n"
+prefs_state);
}
if (key.equals("settings_realname")){
UpdateUserDataRequest updateUserDataRequest = new UpdateUserDataRequest(sessionUsername,sessionPassword,"realname",sharedPreferences.getString(key,""), UpdateUserDataResponseListener);
RequestQueue queue = Volley.newRequestQueue(mContext);
queue.add(updateUserDataRequest);
// Summary sofort erneuern. // Refresh Summaries instantly
prefs_realname = sharedPreferences.getString(key,"");
findPreference(key).setSummary(prefs_realname);
findPreference("preferencescreen_userdata").setSummary(prefs_realname+"\n"
+prefs_email+"\n"
+prefs_streetname+"\n"
+prefs_postcode + " " + prefs_city +"\n"
+prefs_state);
}
if (key.equals("settings_postcode")){
UpdateUserDataRequest updateUserDataRequest = new UpdateUserDataRequest(sessionUsername,sessionPassword,"postcode",sharedPreferences.getString(key,""), UpdateUserDataResponseListener);
RequestQueue queue = Volley.newRequestQueue(mContext);
queue.add(updateUserDataRequest);
// Summary sofort erneuern. // Refresh Summaries instantly
prefs_postcode = sharedPreferences.getString(key,"");
findPreference(key).setSummary(prefs_postcode);
findPreference("preferencescreen_userdata").setSummary(prefs_realname+"\n"
+prefs_email+"\n"
+prefs_streetname+"\n"
+prefs_postcode + " " + prefs_city +"\n"
+prefs_state);
}
if (key.equals("settings_city")) {
UpdateUserDataRequest updateUserDataRequest = new UpdateUserDataRequest(sessionUsername,sessionPassword,"city",sharedPreferences.getString(key,""), UpdateUserDataResponseListener);
RequestQueue queue = Volley.newRequestQueue(mContext);
queue.add(updateUserDataRequest);
// Summary sofort erneuern. // Refresh Summaries instantly
prefs_city = sharedPreferences.getString(key,"");
findPreference(key).setSummary(prefs_city);
findPreference("preferencescreen_userdata").setSummary(prefs_realname+"\n"
+prefs_email+"\n"
+prefs_streetname+"\n"
+prefs_postcode + " " + prefs_city +"\n"
+prefs_state);
}
if (key.equals("settings_state")) {
UpdateUserDataRequest updateUserDataRequest = new UpdateUserDataRequest(sessionUsername,sessionPassword,"state",sharedPreferences.getString(key,""), UpdateUserDataResponseListener);
RequestQueue queue = Volley.newRequestQueue(mContext);
queue.add(updateUserDataRequest);
// Summary sofort erneuern. // Refresh Summaries instantly
prefs_state = sharedPreferences.getString(key,"");
findPreference(key).setSummary(prefs_state);
findPreference("preferencescreen_userdata").setSummary(prefs_realname+"\n"
+prefs_email+"\n"
+prefs_streetname+"\n"
+prefs_postcode + " " + prefs_city +"\n"
+prefs_state);
}
}
}
关于java - 使用 PreferenceFragment 时从 Volley 获取空指针错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46130951/
PreferenceFragment 首选项直到我加载首选项 fragment 后才添加到我的 SharedPreference 映射中。 这是 PreferenceFragment 的预期行为吗?
背景 在以前版本的支持库中,我们可以使用标题来获得设置的主菜单屏幕,每个屏幕都会打开一个新的设置屏幕( fragment )。 问题 现在 header 消失了(如写的 here )一段时间,我认为它
我有一个 PreferenceFragment,我需要在其中添加自定义首选项。现在问题不在于自定义首选项本身,我已经扩展了一些 DialogPreference 创建我自己的首选项,并且我已经处理了保
我需要创建一个settings screen ,其中只有前半部分包含首选项。 我正在考虑添加 PreferenceFragment ,但是我需要支持较旧的设备(最低支持的 API 级别为 9),并且我
我正在创建一个带有主菜单的应用程序,这些菜单项之一将打开一个 PreferenceFragment。为了在应用程序内创建统一性,我希望主菜单看起来尽可能接近 PreferenceFragment。基本
public class MyPrefs extends PreferenceActivity { public static class MyPreferenceFragment exten
在首选项 fragment 中,您可以添加一个图标以显示在首选项的左侧。然而,没有图标的首选项对齐方式很时髦。我如何添加空格以使所有首选项具有相同的对齐方式。我在下面附上了一张图片。 我试过添加 an
当我的小部件添加到主屏幕时,我使用 PreferenceFragment 对其进行配置。用户根据自己的喜好编辑首选项后,如何关闭首选项 fragment ,以便随后可以根据设置添加小部件? 我正在考虑
我想在我的 PreferenceActivity 中获得一个选择文件对话框,我该如何实现?我可以以某种方式覆盖 PreferenceActivity 的 onClick 吗?安卓 API 14。 这是
我正在尝试使用 PreferenceFragment 而不是 PreferenceActivity,因为我不喜欢在我的代码中使用“已弃用”一词。 我已经构建了我的首选项布局和 PreferenceFr
在我的应用程序中,我有偏好:“pref1”,可能的值为 0、1、2。我在 SharedPreferences 类的帮助下保存它们。 我想提供一个用户界面来改变它。我创建类 SettingFragmen
我尝试使用 PreferenceFragment 从 Activity 调用它,当我从 ActionBar 单击一个图标时,我只是调用 FragmentPreferences whit Intent:
我在 Android 中加载 PreferenceFragment 时遇到问题。在这一点上,它是一个非常基本的实现,我想稍后在它工作时对其进行扩展。 问题是导航到 SettingsFragment 后
我又搜索了几个小时,但没有找到我理解/正在寻找的答案。 我有一个首选项屏幕,当用户单击菜单中的设置时它会打开。这行得通。但是,当用户完成设置时,我如何最好地让他关闭此屏幕。 我喜欢 Chrome 中的
我正在开发一个应用程序,我一直在使用支持库 fragment ,我偶然发现了这个问题,我似乎无法使用它添加 PreferencesFragment(用于设置)图书馆? 我找到了一些使用 v7 Pref
如标题所示,我的 PreferenceFragment 与我的工具栏重叠。我已经尝试过不同的解决方案,但问题仍然存在。希望有任何帮助。这是我的代码。 Activity : import android
我有包含左右 fragment 的多 Pane View 。在右边的 fragment 上,我正在启动一个 PreferenceFragment。问题是 fragment 看起来完全扭曲,没有任何风格
我创建了一个包含两个类别和一个复选框的 PreferenceFragment,但是,当我在我的应用程序中显示它时,背景似乎是透明的。 我可以看到主要 Activity 、字段和 PreferenceF
我正在尝试实现“设置”应用的外观,这意味着我正在寻找在类别之间添加分隔线的解决方案。 所以我以为我找到了解决方案,但不幸的是它对我不起作用。正如所建议的那样,我应该在布局中添加空首选项:
我想在我的首选项屏幕底部添加几个按钮,用于设置默认值和恢复默认值。 This答案不包括如何使用 PreferenceFragment 执行此操作。推荐的方法是什么? 加载首选项 fragment 的
我是一名优秀的程序员,十分优秀!