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

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。

以下我们来看看线程的生命周期:

<span>Android多线程研究(4)——从一道面试题说起</span>

实现线程调度的方法例如以下:

1、sleep():该线程是让线程休眠一定的时间,须要捕获InterruptedException

2、yield():暂停当前线程,让同等级优先权的线程运行,假设没有同等级优先权线程则不会起作用。起作用后会让出CPU运行时间,进入就绪状态。

3、join():让一个线程等待调用join方法的线程运行完成后再继续运行。

<span>Android多线程研究(4)——从一道面试题说起</span>

看一段代码:

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()方法的作用由此可见,它会让其它线程等待该线程运行完成后再运行。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/118988.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Borderline-SMOTE算法介绍及Python实现【内附源代码】[通俗易懂]

    Borderline-SMOTE算法介绍及Python实现【内附源代码】[通俗易懂]机器学习之Borderline-SMOTE算法介绍及Python实现,内附源代码鸭!~

    2022年6月17日
    39
  • Ajaxpro组件

    Ajaxpro组件这一篇我们来看一个开源的组件:ajaxpro。虽然这是一个比较老的组件,不过实现思想和源码还是值得我们学习的。通过上一篇的介绍,我们知道要调用页面对象的方法,就是靠反射来实现的,关键是整个处理过程,包括反射调用方法、参数映射等。ajaxpro不仅在后台帮我们实现了这个过程,在前台也封装了请求调用的方法,例如ajax的相关方法,用ajaxpro的方法就可以发送异步请求了,不需要自己封装js或者使用j…

    2022年7月12日
    11
  • Oracle函数详解:regexp_replace[通俗易懂]

    Oracle函数详解:regexp_replace[通俗易懂]regexp_replace:官方内容:官方语法:官方说明:https://docs.oracle.com/cd/B19306_01/server.102/b14200/functions130.htmregexp_replace函数是replace函数的扩展函数,用于通过正则表达式来进行匹配替换,默认情况下,每次匹配到的正则,都替换为replace_string,返回的字符串与…

    2022年4月29日
    1.0K
  • js 字符串截取方法汇总

    js 字符串截取方法汇总一、使用slice()截取1,函数说明slice()方法可通过指定的开始和结束位置,提取字符串的某个部分,并以新的字符串返回被提取的部分。语法如下:string.slice(start,end);start(必需):规定从何处开始选取。如果是负数,那么它规定从字符串尾部开始算起的位置。也就是说,-1指最后一个字符,-2指倒数第二个字符,以此类推。参数说明:end(可选):规定从何处结束选取,即结束处的字符下标。如果没有指定该参数,那么截取的字符串包含从start到结束的所…

    2022年6月11日
    52
  • 【JMeter】参数Parameters和Body Data

    【JMeter】参数Parameters和Body Data在做接口并发测试的时候,才发现Jmeter中的Parameters和BodyData两种参数格式并不是简单的一个是xx=xx,另外一个是json格式的参数先看一个接口[post]/api/xx/xxxx/xxxx通知服务端文件上传完毕输入参数:httpcontenttype:application/json名称|类型|是否必须|参数限制|描述———|–

    2022年6月23日
    18

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号