![](/img/trans.png)
[英]First fragment appears for a second when other fragments are called in tablayout
[英]SharedPreferences are empty when called for the first time in Fragment
我有一個示例Android應用程序,其中根據設置(SharedPreference)中設置的位置(郵政編碼)和溫度單位,該應用程序顯示7天天氣。
似乎當應用程序首次獲取溫度並檢查在SharedPreference中設置的溫度單位是什么時,它為空並且isMetric設置為TRUE。 Utility.isMetric可以進行改進以表示沒有從SharedPreference提取數據,但是我的問題是,為什么從ForecastFragment的onCreateView首次在Utility.isMetric中調用SharedPreference為空?
Utility.isMetric訪問SharedPreference
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
ForecastFragment調用Utility.isMetric
boolean isMetric = Utility.isMetric(getActivity());
我有顯示此行為的logcat,如果您希望看到該消息,請告訴我。
由於沒有限制。 字符集,可以在https://github.com/gosaliajigar/CSC519/tree/master/CSC519-HW6訪問完整代碼
ForecastFragment
public class ForecastFragment extends Fragment implements LoaderCallbacks<Cursor> {
private SimpleCursorAdapter mForecastAdapter;
private static final int FORECAST_LOADER = 0;
private String mLocation;
// For the forecast view we're showing only a small subset of the stored data.
// Specify the columns we need.
private static final String[] FORECAST_COLUMNS = {
// In this case the id needs to be fully qualified with a table name, since
// the content provider joins the location & weather tables in the background
// (both have an _id column)
// On the one hand, that's annoying. On the other, you can search the weather table
// using the location set by the user, which is only in the Location table.
// So the convenience is worth it.
WeatherEntry.TABLE_NAME + "." + WeatherEntry._ID,
WeatherEntry.COLUMN_DATETEXT,
WeatherEntry.COLUMN_SHORT_DESC,
WeatherEntry.COLUMN_MAX_TEMP,
WeatherEntry.COLUMN_MIN_TEMP,
LocationEntry.COLUMN_LOCATION_SETTING
};
// These indices are tied to FORECAST_COLUMNS. If FORECAST_COLUMNS changes, these
// must change.
public static final int COL_WEATHER_ID = 0;
public static final int COL_WEATHER_DATE = 1;
public static final int COL_WEATHER_DESC = 2;
public static final int COL_WEATHER_MAX_TEMP = 3;
public static final int COL_WEATHER_MIN_TEMP = 4;
public static final int COL_LOCATION_SETTING = 5;
public ForecastFragment() {
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// Add this line in order for this fragment to handle menu events.
setHasOptionsMenu(true);
}
@Override
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
inflater.inflate(R.menu.forecastfragment, menu);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int id = item.getItemId();
if (id == R.id.action_refresh) {
updateWeather();
return true;
}
return super.onOptionsItemSelected(item);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
final String[] columns = {WeatherEntry.COLUMN_DATETEXT,
WeatherEntry.COLUMN_SHORT_DESC,
WeatherEntry.COLUMN_MAX_TEMP,
WeatherEntry.COLUMN_MIN_TEMP
};
final int[] viewIDs = {R.id.list_item_date_textview,
R.id.list_item_forecast_textview,
R.id.list_item_high_textview,
R.id.list_item_low_textview
};
// The SimpleCursorAdapter will take data from the database through the
// Loader and use it to populate the ListView it's attached to.
mForecastAdapter = new SimpleCursorAdapter(
getActivity(),
R.layout.list_item_forecast,
null,
columns,
viewIDs,
0);
mForecastAdapter.setViewBinder(new SimpleCursorAdapter.ViewBinder() {
@Override
public boolean setViewValue(View view, Cursor cursor, int columnIndex) {
boolean isMetric = Utility.isMetric(getActivity());
switch (columnIndex) {
case COL_WEATHER_MAX_TEMP: {
String high = Utility.formatTemperature(
cursor.getDouble(cursor.getColumnIndex(WeatherEntry.COLUMN_MAX_TEMP)), isMetric);
((TextView) view).setText(high);
return true;
}
case COL_WEATHER_MIN_TEMP: {
// we have to do some formatting and possibly a conversion
String low = Utility.formatTemperature(
cursor.getDouble(cursor.getColumnIndex(WeatherEntry.COLUMN_MIN_TEMP)), isMetric);
((TextView) view).setText(low);
return true;
}
case COL_WEATHER_DATE: {
String dateString = Utility.formatDate(
cursor.getString(cursor.getColumnIndex(WeatherEntry.COLUMN_DATETEXT)));
((TextView) view).setText(dateString);
return true;
}
}
return false;
}
});
View rootView = inflater.inflate(R.layout.fragment_main, container, false);
// Get a reference to the ListView, and attach this adapter to it.
ListView listView = (ListView) rootView.findViewById(R.id.listview_forecast);
listView.setAdapter(mForecastAdapter);
listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> adapterView, View view, int position, long l) {
Cursor cursor = mForecastAdapter.getCursor();
if (cursor != null && cursor.moveToPosition(position)) {
Intent intent = new Intent(getActivity(), DetailActivity.class)
.putExtra(DetailActivity.DATE_KEY, cursor.getString(COL_WEATHER_DATE));
startActivity(intent);
}
}
});
return rootView;
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
getLoaderManager().initLoader(FORECAST_LOADER, null, this);
super.onActivityCreated(savedInstanceState);
}
private void updateWeather() {
String location = Utility.getPreferredLocation(getActivity());
// update weather only if location is not an EMPTY string
if (location != null
&& location.length() > 0) {
new FetchWeatherTask(getActivity()).execute(location, SettingsActivity.FORECAST_DAYS);
}
}
@Override
public void onResume() {
super.onResume();
if (mLocation != null && !mLocation.equals(Utility.getPreferredLocation(getActivity()))) {
getLoaderManager().restartLoader(FORECAST_LOADER, null, this);
}
}
@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
// This is called when a new Loader needs to be created. This
// fragment only uses one loader, so we don't care about checking the id.
// To only show current and future dates, get the String representation for today,
// and filter the query to return weather only for dates after or including today.
// Only return data after today.
String startDate = WeatherContract.getDbDateString(new Date());
// Sort order: Ascending, by date.
String sortOrder = WeatherEntry.COLUMN_DATETEXT + " ASC";
mLocation = Utility.getPreferredLocation(getActivity());
if (mLocation == null || mLocation.length() == 0) {
mLocation = "00000";
}
Uri weatherForLocationUri = WeatherContract.WeatherEntry.buildWeatherLocationWithStartDate(mLocation, startDate);
Log.d(getString(R.string.app_name), weatherForLocationUri.toString());
// Now create and return a CursorLoader that will take care of
// creating a Cursor for the data being displayed.
return new CursorLoader(
getActivity(),
weatherForLocationUri,
FORECAST_COLUMNS,
null,
null,
sortOrder
);
}
@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
mForecastAdapter.swapCursor(data);
}
@Override
public void onLoaderReset(Loader<Cursor> loader) {
mForecastAdapter.swapCursor(null);
}
}
效用
public class Utility {
public static String getPreferredLocation(Context context) {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
return prefs.getString(context.getString(R.string.pref_location_key),
context.getString(R.string.pref_location_default));
}
public static boolean isMetric(Context context) {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
return prefs.getString(context.getString(R.string.pref_units_key),
context.getString(R.string.pref_units_metric))
.equals(context.getString(R.string.pref_units_metric));
}
static String formatTemperature(double temperature, boolean isMetric) {
double temp;
if ( !isMetric ) {
temp = 9*temperature/5+32;
} else {
temp = temperature;
}
return String.format("%.0f", temp);
}
static String formatDate(String dateString) {
Date date = WeatherContract.getDateFromDb(dateString);
return DateFormat.getDateInstance().format(date);
}
}
我已將logcat放在兩個文件(實用程序和ForecastFragment)中以演示該問題。 這是在首次安裝ForecastFragment onCreateView時安裝了應用程序時的日志貓。
07-05 17:35:57.562 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:35:57.563 21604-21604/com.example.android.weather.app D/JIGAR: isMetric true
這是在按下“刷新”按鈕后加載ForecastFragment onCreateView時的日志貓。
07-05 17:36:44.433 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.435 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.435 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.436 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.445 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.449 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.450 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.452 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.458 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.458 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.458 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.458 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.459 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.460 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.460 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.460 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.461 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.461 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.461 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.461 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.462 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.462 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.462 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.463 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.463 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.464 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.464 21604-21604/com.example.android.weather.app D/JIGAR: {}
07-05 17:36:44.464 21604-21604/com.example.android.weather.app D/JIGAR: {}
我現在已經從上述兩個位置刪除了Log.d語句,但是非常歡迎您放置日志語句。
注意:這是一個家庭作業應用程序,現在在文件及其完整的應用程序中顯示了TO DO,但是在使用該應用程序並理解代碼時,我在應用程序中發現了此錯誤並試圖了解其發生的原因。 這NOT
家庭作業或任何提交的一部分。
感謝Dennis和Kaze,我的Android應用程序正在addPreferencesFromResource(R.xml.pref_general)
幫助下從xml文件加載默認的共享首選項,該extends PreferenceActivity
,這是一個單獨的活動,當有人單擊“設置”按鈕時啟動它,因此在按“設置”按鈕之前,不會填充共享的首選項。
您關於填充默認首選項的問題使我研究了如何填充默認首選項,應該使用PreferenceManager.setDefaultValues
將其填充在ForecastFragment onCreate
中,如下所示...
PreferenceManager.setDefaultValues(context, R.xml.pref_general, false);
這解決了我的問題。 我會投票支持您的答案! 謝謝
您的sharedPreferences為空的原因是因為您首先沒有成功輸入任何內容。
我建議以這種方式使用sharedPreferences,因為這對我有用:
import android.content.SharedPreferences;
在您的職能:
SharedPreferences preferences = getApplicationContext().getSharedPreferences("AppPref", MODE_PRIVATE);
SharedPreferences.Editor editor = preferences.edit();
editor.putString("**Input your data here**", **tag your data/give it a name*);//example editor.putString("userPassword", password);
editor.commit();//this one places your string into the sharedpreferences instance you created
為了在onCreate上訪問您的共享首選項:
private String data; //this receives your data from sharedpref
@Override
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
SharedPreferences conn = getSharedPreferences("AppPref", MODE_PRIVATE);
data = conn.getString("**your data tag**","");
}
希望這可以幫助。 如果要存儲會話中的值,則需要為運行時輸入的數據專門創建一個共享首選項實例。 我不太擅長技術說明,因此請澄清一下。
您應該使用PreferenceManager中的以下方法從xml加載默認值(如注釋中所述,默認值是從xml加載的)。
static void setDefaultValues(Context context, int resId, boolean readAgain)
Sets the default values from an XML preference file by reading the values defined by each Preference item's android:defaultValue attribute.
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.