Android多线程研究(4)——从一道面试题说起
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了Android多线程研究(4)——从一道面试题说起,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含2855字,纯文字阅读大概需要5分钟。
内容图文
![Android多线程研究(4)——从一道面试题说起](/upload/InfoBanner/zyjiaocheng/1296/03b09fc879f64e1eb71c98dc8e0c94b3.jpg)
有一道这样的面试题:开启一个子线程和主线程同时运行,子线程输出10次后接着主线程输出100次,如此反复50次。先看下面代码:
package com.maso.test; /** * * @author Administrator * 两个线程,其中是一个主线程,第一个线程先运行输出10次,主线程接着运行输出100次,如此反复50次 */ public class ThreadTest3 implements Runnable{ private static Test test; @Override public void run() { for(int i=0; i<50; i++){ test.f1(i); } } public static void main(String[] args) { test = new Test(); new Thread(new ThreadTest3()).start(); for(int i=0; i<50; i++){ test.f2(i); } } /** * 将控制和逻辑及数据分类(该类就是数据) * @author Administrator * */ static class Test{ private boolean isf1 = true; /** * 输出10次 */ public synchronized void f1(int j){ if(!isf1){ try { this.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } for(int i=1; i<=10; i++){ System.out.println(Thread.currentThread().getName() + "第" + j + "次轮巡,输出" + i); } isf1 = false; notify(); } /** * 输出100次 */ public synchronized void f2(int j){ if(isf1){ try { this.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } for(int i=1; i<=100; i++){ System.out.println(Thread.currentThread().getName() + "第" + j + "次轮巡,输出" + i); } isf1 = true; notify(); } } }上面判断用的是if语句,这样做看似没有什么问题,实际上这样做是不安全的,因为线程在等待的过程中有可能被假唤醒,所以我们需要使用while语句。另外在使用wait和notify的时候需要注意一下几点:
1、调用object的wait方法和notity方法时,必须先获得object的对象锁(必须写在synchronized中)。
2、如果调用了object的wait方法,则该线程就放掉了对象锁。
3、如果A1、A2、A3都在object.wait(),则B调用object.notify()只能唤醒A1、A2、A3中的一个(具体哪一个由JVM决定)
4、object.notifyAll()能够唤醒全部。
5、B在唤醒A的时候,B如果还持有对象锁,则要等到B释放锁后,A才有机会执行。
Sleep和Wait有什么区别?
sleep()并不释放对象锁,wait()释放对象锁。但是wait()和sleep()都可以通过interrupt()方法打断线程的暂停状态,从而使线程立刻抛出InterruptedException。如果线程A希望立即结束线程B,则可以对线程B对应的Thread实例调用interrupt方法。如果此刻线程B正在wait/sleep/join,则线程B会立刻抛出InterruptedException,在catch() {} 中直接return即可安全地结束线程。需要注意的是,InterruptedException是线程自己从内部抛出的,并不是interrupt()方法抛出的。对某一线程调用interrupt()时,如果该线程正在执行普通的代码,那么该线程根本就不会抛出InterruptedException。但是,一旦该线程进入到wait()/sleep()/join()后,就会立刻抛出InterruptedException。
下面我们来看看线程的生命周期:
实现线程调度的方法如下:
1、sleep():该线程是让线程休眠一定的时间,需要捕获InterruptedException
2、yield():暂停当前线程,让同等级优先权的线程运行,如果没有同等级优先权线程则不会起作用。起作用后会让出CPU运行时间,进入就绪状态。
3、join():让一个线程等待调用join方法的线程执行完毕后再继续执行。
看一段代码:
public class ThreadTest4 implements Runnable{ private static int a = 0; @Override public void run() { for(int i=0; i<10; i++){ a++; } } public static void main(String[] args) { new Thread(new ThreadTest4()).start(); System.out.println(a); } }这段代码会输出10吗?答案是不会的,因为在启动子线程后,就立马输出了a的值,此时子线程对a还没有操作。修改如下:
public class ThreadTest4 implements Runnable{ private static int a = 0; @Override public void run() { for(int i=0; i<10; i++){ a++; } } public static void main(String[] args) throws InterruptedException { Thread t = new Thread(new ThreadTest4()); t.start(); t.join(); System.out.println(a); } }这回输出了10,join()方法的作用由此可见,它会让其他线程等待该线程执行完毕后再执行。
原文:http://blog.csdn.net/dawanganban/article/details/26565601
内容总结
以上是互联网集市为您收集整理的Android多线程研究(4)——从一道面试题说起全部内容,希望文章能够帮你解决Android多线程研究(4)——从一道面试题说起所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。