第10讲UI线程阻塞及其优化

、UI
阻塞demo

(首先在activity_main.xml中放置两个button,分别命名为button1,button2)

//首先设置一个button1用来进行动画演示

Button button1=(Button) findViewById(R.id.button1);

//位移方向移动的动画

TranslateAnimation animation=new TranslateAnimation(100,150, 50, 100);

animation.setDuration(2000);                 //设置持续的时间,单位为微秒

animation.setRepeatCount(10);             //设置重复的次数

button1.setAnimation(animation);

//设置一个button2进行耗时操作sleep

Button button2=(Button) findViewById(R.id.button2);

button2.setOnClickListener(new OnClickListener(){

public void onClick(View v) {

try {

Thread.sleep(5000);                                    //设置线程耗时操作,单位ms

} catch (InterruptedException e) {

e.printStackTrace();

}

}

});

点击button2之后,button1的移动发生卡死,发生UI线程阻塞。

当一个应用程序启动之后,Android系统会为这个应用程序创建一个主线程。这个线程非常重要,它负责渲染视图,分发事件到响应监听器并执行,对界面进行轮询监听。因此,一般也叫做“UI线程”。

Android系统不会给应用程序的多个元素组件,建立多个线程来执行。一个视图(activity)中的多个view组件运行在同一个UI线程当中。因此多个view组件的监听器的执行可能对相互影响(即是UI线程阻塞)

、UI阻塞解决方案:创建一个新线程

Button button2 = (Button)findViewById(R.id.button2);

button2.setOnClickListener(new OnClickListener(){

public void onClick(View v) {

new Thread(newRunnable() {                 //创建一个新线程

public void run() {

try {

Thread.sleep(5000);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

}).start();

}

});

将耗时操作放在一个新的线程里面进行操作,利用new Thread().start()打开新的线程。

此时因为没有更改view,所以不会报错。但是一旦我们在UI线程外修改了view组件的属性,则会报错。

安卓官方规定的两个规则

There are two simple rules to Android’ssingle thread model:

Do not block the UI thread. 不要阻塞UI线程

Do not access the Android UI toolkit from outside of the UIthread. 不要再UI线程之外的其他线程中,对视图当中的组件进行设置

违反rule2会报异常:Only the original thread that created aview hierarchy can touch its views.

、在其他的线程中修改UI线程中的组件

方法a:view.post

将需要进行view组件修改的程序部分放在v.post(new Runnable() { }中的run()方法进行即可。

Buttonbutton2 = (Button) findViewById(R.id.button2);

button2.setOnClickListener(newOnClickListener() {

publicvoid onClick(final View v) {

newThread(newRunnable() {

publicvoid run() {

try {

Thread.sleep(5000);      //设置线程耗时操作,单位ms。在新线程中进行耗时操作

} catch (InterruptedException e){

e.printStackTrace();

}

} .

}).start();

intsum=10;                         //假定通过耗时操作计算处理的得到的一个值

v.post(newRunnable() {

public void run() {

TextView textView=(TextView) v;         //在post中进行view修改操作

textView.setText(""+10);

}

});

}

});

post原理:通过view.post在任务队列中添加一个任务Runnable对象,UI thread在轮询过程中会检查任务队列中是否有任务,有则执行(如下图所示)。此时任务队列中的修改view组件的操作是在UI线程中执行的,则不违反上面的rule2.

方案a:post方法,可读性差,维护性差。

原理是在新线程中进行耗时操作,在post中进行view修改操作。

方法b:AsyncTask (异步任务)

AsyncTask允许你在你的用户接口上执行异步工作。他在工作者线程中执行阻塞操作,然后将结果发布给 UI线程,不需要你自己处理线程或者 handler。

为了使用他,你需要继承 AsyncTask并且实现他的 doInBackground()回调方法,他在一个后台线程池中运行。为了更新你的 UI,你需要实现 onPostExecute()方法,他传递来自 doInBackgroud()的结果且在 UI线程中运行。所以你可以安全的更新你的 UI。你可以在 UI线程通过 execute()来运行你的任务。

privateButton button2 = null;

protected voidonCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

//为button1添加一个动画操作

Buttonbutton1 = (Button) findViewById(R.id.button1);

//位移方向移动的动画

TranslateAnimationanimation = new TranslateAnimation(0,150,0,0);

animation.setRepeatCount(30);     //设置重复的次数

animation.setDuration(2000);                  //设置持续的时间,单位为微秒

button1.setAnimation(animation);

button2= (Button) findViewById(R.id.button2);

button2.setOnClickListener(newOnClickListener() {

publicvoid onClick(final View v){

newDownloadImageTask().execute();

}

});

}

privateclass DownloadImageTask extends AsyncTask<String, Void,Integer> {

protected IntegerdoInBackground(String... urls) {

try{

Thread.sleep(5000);

}catch (InterruptedException e) {

e.printStackTrace();

}

intsum=10;

returnsum;

}

protected voidonPostExecute(Integer sum) {

button2.setText(""+sum);

}

}

第10讲- UI线程阻塞及其优化的更多相关文章

  1. Android Studio学习随笔-UI线程阻塞以及优化

    我们在使用手机的时候,经常会遇到一个问题:先是卡死,然后跳出该程序无响应,是否关闭的提示(当然有可能是我们手机性能太差=.=)这是因为线程的阻塞引起的,在这里我讲述一下UI线程,一般处理程序会在UI线 ...

  2. jQuery同步Ajax带来的UI线程阻塞问题及解决办法

    俗话说不作死就不会死,今天作死了一回,写了一个比较二逼的函数,遇到了同步Ajax引起的UI线程阻塞问题,在此记录一下. 事情起因是这样的,因为页面上有多个相似的异步请求动作,本着提高代码可重用性的原则 ...

  3. jQuery同步Ajax带来的UI线程阻塞问题及解决方法

    遇到了同步Ajax引起的UI线程阻塞问题,在此记录一下. 事情起因是这样的,因为页面上有多个相似的异步请求动作,本着提高代码可重用性的原则,我封装了一个名为getData的函数,它接收不同参数,只负责 ...

  4. jQuery同步Ajax带来的UI线程阻塞问题

    一.需求 在调ajax接口的时候因为有时间延迟,想要做一个loading加载的效果,等数据返回时再把loading的效果去掉. 所以我在调ajax的代码块前面加了显示loading效果的代码,ajax ...

  5. Jquery ajax 同步阻塞引起的UI线程阻塞的坑(loading图片显示不出来,layer.load延迟)

    今天想做一个点击地市用ajax重新获取数据刷新页面功能,因为ajax属于耗时操作,想在获取数据且加载页面时显示加载遮罩层,结果发现了ajax的好多坑. 例如如上栗子,我想点击按钮让遮罩层显示,ajax ...

  6. Javascript之UI线程与性能优化

    在浏览器中,Javascript执行与UI更新是发生在同一个进程(浏览器UI线程)中的.UI线程的工作基于一个简单的队列系统,任务会被保存到队列中直到进程空闲时被提取出来执行.所以Javascript ...

  7. [ionic开源项目教程] - 第10讲 新闻详情页的用户体验优化

    目录 [ionic开源项目教程] 第1讲 前言,技术储备,环境搭建,常用命令 [ionic开源项目教程] 第2讲 新建项目,架构页面,配置app.js和controllers.js [ionic开源项 ...

  8. 通过使用Web Workers,Web应用程序可以在独立于主线程的后台线程中,运行一个脚本操作。这样做的好处是可以在独立线程中执行费时的处理任务,从而允许主线程(通常是UI线程)不会因此被阻塞/放慢。

    Web Workers API - Web API 接口参考 | MDNhttps://developer.mozilla.org/zh-CN/docs/Web/API/Web_Workers_API ...

  9. C#.NET使用Task,await,async,异步执行控件耗时事件(event),不阻塞UI线程和不跨线程执行UI更新,以及其他方式比较

    使用Task,await,async,异步执行事件(event),不阻塞UI线程和不跨线程执行UI更新 使用Task,await,async 的异步模式 去执行事件(event) 解决不阻塞UI线程和 ...

随机推荐

  1. pyqt搜索指定信息 github处找到,谢谢这位朋友的帮助了

    def tabunqi(self,text):    #第一遍添加之后,不提示,当第二次添加相同的数据时,就提示下    text1=str(text)    items = self.downwid ...

  2. eclispe远程调试tomcat

    在eclispe中新建web应用,名字叫webtest.里面只有一个HelloServlet.Web.xml配置如下. 修改tomcat的启动脚本startup.bat.复制startup.bat为s ...

  3. 区间DP(初步了解)

    区间动态规划问题一般都是考虑.对于每段区间,他们的最优值都 是由几段更小区间的最优值得到,是分治思想的一种应用,将一个区间 问题不断划分更小的区间直至一个元素组成的区间,枚举他们的组合  .求合并后的 ...

  4. C#获取时间戳的方法

    获取时间戳的方法 /// <summary>        /// 获取时间戳        /// </summary>        /// <param name= ...

  5. 只包含schema的dll生成和引用方法

    工作中,所有的tools里有一个project是只包含若干个schema的工程,研究了一下,发现创建这种只包含schema的dll其实非常简单. 首先,在visual studio-new proje ...

  6. Android异步请求

    class MyTask_SendMessage extends AsyncTask<String, Void, String> { @Override protected void on ...

  7. Jobs定时器 - Quartz

    Quartz是OpenSymphony开源组织在Job scheduling领域又一个开源项目,它可以与J2EE与J2SE应用程序相结合也可以单独使用.Quartz可以用来创建简单或为运行十个,百个, ...

  8. Shiro 权限框架使用总结

    我们首先了解下什么是shiro ,Shiro 是 JAVA 世界中新近出现的权限框架,较之 JAAS 和 Spring Security,Shiro 在保持强大功能的同时,还在简单性和灵活性方面拥有巨 ...

  9. 方法覆盖(override)”的要点

    方法覆盖要求子类与父类的方法一模一样,否则就是方法重载(overload)!请自行编写代码测试以下特性:在子类中,若要调用父类中被覆盖的方法,可以使用super关键字. 结论:          在“ ...

  10. poj1064 二分,注意精度!

    Cable master Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 35269   Accepted: 7513 Des ...