我刚好明白了,但我在事情发生的顺序上有一个小问题。 具体来说,在我的 thread() 中,我正在设置一个 Spinner 使用的数组。问题是 Spinner 基本上在我的 thread() 完成之前就已全部设置完毕,因此它会使用空数组自行设置。
如何将微调器 ArrayAdapter 与另一个线程正在加载的数组相关联?
我已将代码缩减到我认为理解问题所必需的部分,但如果需要更多,请告诉我。无论是否调用refreshData()都会出现问题。
同样,有时我想从菜单中调用 loadData()。直接跟随 loadData() 如果我尝试在下一行触发 toast ,这会导致强制关闭,这也是因为我如何实现 ProgressDialog。
感谢您的关注
public class CMSHome extends Activity {
private static List<String> pmList = new ArrayList<String>();
// Instantiate helpers
PMListHelper plh = new PMListHelper();
ProjectObjectHelper poc = new ProjectObjectHelper();
// These objects hold lists and methods for dealing with them
private Employees employees;
private Projects projects;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
// Loads data from filesystem, or webservice if necessary
loadData();
// Capture spinner and associate pmList with it through ArrayAdapter
spinner = (Spinner) findViewById(R.id.spinner);
ArrayAdapter<String> adapter = new ArrayAdapter<String>(
this, android.R.layout.simple_spinner_item,
pmList);
adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
spinner.setAdapter(adapter);
//---the button is wired to an event handler---
Button btn1 = (Button)findViewById(R.id.btnGetProjects);
btn1.setOnClickListener(btnListAllProjectsListener);
spinner.setOnItemSelectedListener(new MyOnItemSelectedListener());
}
private void loadData()
{
final ProgressDialog pd = ProgressDialog.show(this,
"Please wait", "Loading Data...", true, false);
new Thread(new Runnable(){
public void run(){
employees = plh.deserializeEmployeeData();
projects = poc.deserializeProjectData();
// Check to see if data actually loaded, if not then refresh
if ((employees == null) || (projects == null)) {
refreshData();
}
// Load up pmList for spinner control
pmList = employees.getPMList();
pd.dismiss();
}
}).start();
}
private void refreshData()
{
// Refresh data for Projects
projects = poc.refreshData();
poc.saveProjectData(mCtx, projects);
// Refresh data for PMList
employees = plh.refreshData();
plh.savePMData(mCtx, employees);
}
}
<---- 编辑 -----> 在 Jims 建议之后,我尝试将 onCreate() 更改为以下内容。不确定我这样做是否正确,但仍然不起作用:
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
mCtx = this;
// Loads data from filesystem, or webservice if necessary
// Would like to extend this to update if files are over x days old
final ProgressDialog pd = ProgressDialog.show(this,
"Please wait", "Loading Data...", true, false);
new Thread(new Runnable(){
public void run(){
employees = plh.deserializeEmployeeData();
projects = poc.deserializeProjectData();
// Check to see if data actually loaded, if not then refresh
if ((employees == null) || (projects == null)) {
refreshData();
}
pd.dismiss();
runOnUiThread(new Runnable() {
public void run(){
// Load up pmList for spinner control
pmList = employees.getPMList();
}
});
}
}).start();
spinner = (Spinner) findViewById(R.id.spinner);
ArrayAdapter<String> adapter = new ArrayAdapter<String>(
this, android.R.layout.simple_spinner_item, pmList);
adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
spinner.setAdapter(adapter);
//---the button is wired to an event handler---
Button btn1 = (Button)findViewById(R.id.btnGetProjects);
btn1.setOnClickListener(btnListAllProjectsListener);
spinner.setOnItemSelectedListener(new MyOnItemSelectedListener());
}
哇,这让我花了很长时间才找到解决方案,但我很高兴终于让它起作用了。
使用处理程序可以用后台线程更新 Spinner。处理程序在线程的主要工作完成后被调用。
mProgressDlg = ProgressDialog.show(this, "App_Name", "Loading data...",
true, false);
new Thread(new Runnable(){
public void run() {
/*Load Data, set pmList in my case*/
mProgressDlg.dismiss();
hRefresh.sendEmptyMessage(REFRESH);
}
}).start();
Handler hRefresh = new Handler(){
@Override
public void handleMessage(Message msg) {
switch(msg.what){
case REFRESH:
spinner = (Spinner) findViewById(R.id.spinner);
final ArrayAdapter<String> adapter = new ArrayAdapter<String>(
mCtx, android.R.layout.simple_spinner_item, pmList);
adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
spinner.setAdapter(adapter);
spinner.setOnItemSelectedListener(new MyOnItemSelectedListener());
break;
}
}
};
感谢 bhatt4982 和他在 this thread 上的回答
最佳答案
不知道你是否解决过这个问题,但这对我有用:
public class Start extends Activity {
private static final String TAG = "PriceList";
ArrayAdapter<ProductCategory> category_adapter;
ArrayAdapter<ProductGroup> group_adapter;
ArrayList<ProductCategory> categories;
ArrayList<ProductGroup> groups;
ArrayList<Price> prices;
Spinner group_spinner;
Spinner category_spinner;
ProgressDialog progressDialog;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
group_spinner = (Spinner) findViewById(R.id.group_spinner);
category_spinner = (Spinner) findViewById(R.id.category_spinner);
// product category spinner
categories = new ArrayList<ProductCategory>();
category_adapter = new CustomArrayAdapter<ProductCategory>(categories);
category_adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
// load category spinner from database
loadCategory(); // adapter MUST be setup before this is called
category_spinner.setAdapter(category_adapter);
category_spinner.setOnItemSelectedListener(new OnItemSelectedListener () {
..... other stuff ......
private final Handler handler = new Handler() {
@Override
public void handleMessage(final Message msg) {
Log.v(TAG, "worker thread done, setup adapter");
switch (msg.what) {
case Constants.CATEGORIES:
category_adapter.notifyDataSetChanged();
break;
case Constants.GROUPS:
group_adapter.notifyDataSetChanged();
break;
case Constants.PRICES:
startActivity(new Intent(Start.this, ShowPrices.class));
break;
default:
}
// dismiss dialog
progressDialog.dismiss();
}
};
// loadCategory() essentially the same....
private void loadGroup(final String cat) {
Log.v(TAG, "loadGroup");
progressDialog = new ProgressDialog(this);
progressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
progressDialog.setMessage("Retrieving Product Groups...");
progressDialog.setMax(100);
progressDialog.setProgress(0);
progressDialog.show();
new Thread() {
@Override
public void run() {
int count = 100;
int i = 0;
SQLiteDatabase db = DbUtils.getStaticDb();
Cursor c = db.rawQuery("select count(*) from productgroup where category = \'"
+ cat + "\';", null);
c.moveToFirst();
if (!c.isAfterLast()) {
count = c.getInt(0);
}
c.close();
progressDialog.setMax(count);
groups.clear();
groups.add(new ProductGroup("-1", "--- Select ---"));
StringBuilder sb = new StringBuilder("select _id,description from productgroup");
sb.append(" where category = \'");
sb.append(cat);
sb.append("\' order by description;");
Log.v(TAG, sb.toString());
c = db.rawQuery(sb.toString(), null);
c.moveToFirst();
while (!c.isAfterLast()) {
Log.v(TAG, c.getString(0));
groups.add(new ProductGroup(c.getString(0), c.getString(1)));
i++;
if (i % 5 == 0) {
progressDialog.setProgress(i);
}
c.moveToNext();
}
c.close();
// tell UI thread OK
handler.sendEmptyMessage(Constants.GROUPS);
}
}.start();
}
// custom ArrayAdapter allows us to have our own ArrayList<T>
class CustomArrayAdapter<T> extends ArrayAdapter<T> {
CustomArrayAdapter(ArrayList<T> list) {
super(Start.this, android.R.layout.simple_spinner_item, list);
}
}
关于Android ProgressDialog 进度条按正确的顺序做事,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2596390/