• Android多线程研究(4)——从一道面试题说起


    有一道这样的面试题:开启一个子线程和主线程同时运行,子线程输出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()方法的作用由此可见,它会让其他线程等待该线程执行完毕后再执行。






  • 相关阅读:
    海量数据框架变迁——阿里巴巴上市背后的技术力量
    redis集群配置
    【等待优化】sql server CXPACKET 等待 导致 CPU飙高、CPU100%
    (4.39)sql server如何配置分布式事务(MSDTC)
    mysql断电,mysql ibdata 文件损坏(批量利用mysql表空间导入导出)
    mysqlfrm使用
    (1.1)zabbix 基础概念及工作原理整理【转】
    (5.17)mysql集群技术概述(LVS、Keepalived、HAproxy)
    事务日志备份失败错误:Backup detected log corruption in database
    sql server事务日志解析工具(开源,类似apexsql log)
  • 原文地址:https://www.cnblogs.com/lanzhi/p/6468937.html
Copyright © 2020-2023  润新知