[英]Android Activity takes too long to show content
我在onCreate()
Activity執行需要一些時間的長計算。
在同一個onCreate()
我調用setContentView()
來設置活動的外觀。
關鍵在於,由於執行上述計算需要一段時間,因此活動的屏幕僅在很長時間后加載。
請問有關如何避免這種情況的任何建議?
我試圖在onCreate()
調用setContentView()
並在onResume()
啟動計算,但同樣只在最后加載Activity屏幕。
除了使用例如AsyncTask之外別無他法。 原因是實際渲染不是異步發生的; 換句話說,setContentView只會設置一些數據,但在那個時間點什么都不會顯示。
然而,AsyncTask不一定意味着“長”計算。 但是,如果您的應用程序依賴於結果,並且沒有其他計算並行發生,那么它可能仍然是您實現所需內容的最簡單方法。 如果沒有,您可能必須使用線程甚至。
更新由於每個人都在更多地使用各種質量的AsyncTask答案來轟炸原始海報,我想再次強調AsyncTask用於短操作(引用參考: 最多幾秒 )而OP有沒有說明他的計算真正需要多長時間。 此外,AsyncTask是一個只能運行一次的一次性對象。
另一個需要考慮的重點是以下幾點。 Android為AsyncTask分配后台任務優先級。 這意味着,除了較低的調度優先級之外,AsyncTask中的計算將花費十倍於在前台執行的計算 ,因為Android運行具有后台優先級且具有10%CPU周期的人為限制的所有任務。 但是,AsyncTasks可以通過提高“優先級”來提升其優先級。 對於AsyncTask,它將像這樣完成:
public R doInBackground(I... is) {
Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND +
Process.THREAD_PRIORITY_MORE_FAVORABLE);
...
}
如果所有計算都需要很長時間才能執行,那么您的UI將被“鎖定”並且不會更新。
您需要在AsyncTask中完成所有長時間的工作
AsyncTask可以正確,方便地使用UI線程。 此類允許執行后台操作並在UI線程上發布結果,而無需操作線程和/或處理程序。
在開始初始化視圖之前調用onCreate方法中的setContentView(layoutID),在onCreate方法中調用setContentView之后創建AsyncTask並啟動AsyncTask線程。 下面給出的東西
onCreate(....){
--
--
setContentView(layoutID);
---
--
new asynchTask(); // load your ui in AsyncTask by creating an inner class in your activity by extending AsyncTask class
}
這是一個如何實現AsyncTask的教程
你必須實現AsyncTask
public class AsyncTaskActivity extends Activity implements OnClickListener {
Button btn;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
btn = (Button) findViewById(R.id.button1);.
//because we implement OnClickListener we only have to pass "this" (much easier)
btn.setOnClickListener(this);
}
public void onClick(View view){
//detect the view that was "clicked"
switch(view.getId())
{
case R.id.button1:
new LongOperation().execute("");
break;
}
}
private class LongOperation extends AsyncTask<String, Void, String> {
@Override
protected String doInBackground(String... params) {
for(int i=0;i<5;i++) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return "Executed";
}
@Override
protected void onPostExecute(String result) {
TextView txt = (TextView) findViewById(R.id.output);
txt.setText("Executed"); // txt.setText(result);
//might want to change "executed" for the returned string passed into onPostExecute() but that is upto you
}
@Override
protected void onPreExecute() {
}
@Override
protected void onProgressUpdate(Void... values) {
}
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.