由于android本身是单线程模型,UI主线程是不安全的,所以不能在非UI线程中操作UI。
如果在非UI线程中直接操作UI线程,会抛出android.view.ViewRoot$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views。
由于UI主线程只负责界面绘制和事件监听,所以应该必须保证它能随时响应用户操作,其它(耗时)操作应该另起线程或服务。
由上,弹出用户终止应用程序提醒对话框是因为在UI线程中进行了耗时操作,时间超过5s。
下面的这篇文章对我非常有用
做web开发也有一段时间,但发现现在有定时器的功能应用得是越来越多,有时我们总是需要在深夜风站访问量低的时候做一些工作,比如说数据转移,有些对账日志的生成,向一些接口平台上传或下载文件,以前在做PHP开发的时候这些任务都是要依懒于服务器的计划任务或者自己写一些应用放到服务器上运行。但最近在做Java开发的时候却发现这一点非常让人满意,我们直接可以用Servlet侦听器结合Java定时器就可以实现这一过程了,简单讲一下这个过程:首先我们需要实现javax.servlet.ServletContextListener接口,同时实现它的contextInitialized (ServletContextEvent event)和contextDestroyed(ServletContextEvent event)两个接口函数。考虑定时器有个建立和销毁的过程,看了前面两个接口函数,就不容置疑的把建立的过程置入 contextInitialized,把销毁的过程置入contextDestroyed了。
我把ServletContextListener的实现类取名为SysContextListener,在其内添加一个定时器,示例代码如下所示:
以上代码中, timer.schedule(new TimerAction(event.getServletContext()), 0, 5*1000)这一行为定时器调度语句,其中TimerAction是自定义需要被调度的执行任务,默认的情况下是调用run()方法,下面会重点讲述,第三个参数表示执行周期(单位为毫秒),中间参数0表示无延迟。
下面是执行的任务代码:
到这儿,ServletContextListener和TimerAction的代码都已完整了,也就是程序执行都已经完成了,到了这儿并没有完成的。我们还需要的j把ServletContextListener部署到您的Web工程中去,在您工程的web.xml配置文件中web-app节点中加入如下三行:
这个是从网上抄的: 原链接为: http://www.joyphper.net/article/201107/131.html
直接看这里吧,我只是搬运工。
定时执行在一段时候后停止的倒计时,在倒计时执行过程中会在固定间隔时间得到通知(译者:触发onTick方法),下面的例子显示在一个文本框中显示一个30s倒计时:
new CountdownTimer(30000, 1000) {
public void onTick(long millisUntilFinished) {
mTextField.setText("seconds remaining: " + millisUntilFinished / 1000);
}
public void onFinish() {
mTextField.setText("done!");
}
}.start();
onTick的调用是同步的,保证这次调用不会在之前调用完成前发生。这里的同步机制主要是用来:onTick的实现需要很多时间执行比倒计时间隔更重要的事情。
构造函数
public CountDownTimer (long millisInFuture, long countDownInterval)
参数
millisInFuture 从开始调用start()到倒计时完成并onFinish()方法被调用的毫秒数。(译者注:倒计时时间,单位毫秒)
countDownInterval 接收onTick(long)回调的间隔时间。(译者注:单位毫秒)
公共方法
public final void cancel ()
取消倒计时(译者:取消后,再次启动会重新开始倒计时)
public abstract void onFinish ()
倒计时完成时被调用
public abstract void onTick (long millisUntilFinished)
固定间隔被调用
参数
millisUntilFinished 倒计时剩余时间。
public synchronized final CountDownTimer start ()
启动倒计时
import android.app.Activity;
import android.os.Bundle;
import android.os.CountDownTimer;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
import com.test.R;
public class CountDownTimeActivity extends Activity implements OnClickListener {
TextView mTextView;
Button mButton1;
Button mButton2;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.countdown);
mTextView = (TextView)findViewById(R.id.textView1);
mButton1 = (Button)findViewById(R.id.button1);
mButton2 = (Button)findViewById(R.id.button2);
mButton1.setOnClickListener(this);
mButton2.setOnClickListener(this);
}
CountDownTimer timer = new CountDownTimer(40000,1000) {
@Override
public void onTick(long millisUntilFinished) {
mTextView.setText("seconds remaining: " + millisUntilFinished / 1000);
try {
Thread.sleep(1200);
} catch (InterruptedException e) {
e.printStackTrace();
}
Log.e("CountDown",millisUntilFinished+"");
}
@Override
public void onFinish() {
mTextView.setText("done!");
}
};
@Override
public void onClick(View v) {
switch(v.getId()){
case R.id.button1:
timer.start();
break;
case R.id.button2:
timer.cancel();
break;
}
}
}