Java多线程的测试

一、通过继承Thread类方式创建线程

步骤:创建一个Thread类,或者一个Thread子类的对象,即通过继承Thread类的方式创建线程类,重写run()方法。

例子:线程1和线程2竞争获取cpu资源,两者获得cpu资源具有不确定性,是随机获取资源的

package com.imooc.thread;

class MyThread extends Thread{
    
    MyThread(String name){
        super(name);
    }
    public void run() {
        for(int i = 0; i<10;i++) {
            System.out.println(getName() + "正在执行" + i);
        }
    }
}
/**
  * 测试线程1和线程2竞争获取cpu资源,两者获得cpu资源具有不确定性,是随机获取资源的
 */
public class ThreadTest1 {
    
    public static void main(String[] args) {
        MyThread mt1 = new MyThread("线程1");
        MyThread mt2 = new MyThread("线程2");
        mt1.start();
        mt2.start();
    }
}

二、测试Thread类的join() 和 join(x ms)方法

带参数的join方法:public final void join(long millis)

作用:调用join(x)方法的线程,优先执行x毫秒,x时间一过,丧失优先级,其他线程可以竞争cpu资源执行。

package com.imooc.thread;

class MyThread2 extends Thread{
	public void run() {
		for(int i =1;i<=300;i++) {
			System.out.println(getName()+"线程真正执行" + i +"次");
		}
	}
}

/**
 * 带参数的join方法:public final void join(long millis)
 * 调用join(x)方法的线程,优先执行x毫秒,x时间一过,丧失优先级,其他线程可以竞争cpu资源执行。
*/

public class JoinDemo {
	public static void main(String[] args) {
		MyThread2 mt = new MyThread2();
		mt.start();
		
		try {
			//mt线程执行1ms后,丧失优先级,主线程开始竟争资源,然后执行
			mt.join(1);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		for(int i=1;i<=20;i++) {
			System.out.println("主线程执行第" + i +"次");
		}
		
		int main = Thread.currentThread().getPriority();
		System.out.println(main);
		mt.setPriority(Thread.MAX_PRIORITY);
	}
}

三、通过实现Runnable接口创建线程

步骤(三步):类实例化,根据实例化的对象创建线程类,线程子类调用start方法

package com.imooc.runnable;

public class PrintRunnable implements Runnable {

	//成员变量,共享资源 
	int i = 1;
	//run方法被多个线程共享,这个例子中被t1和t2线程共享,t1和t2共运行10次
	//适合多个线程处理同一个资源
	public void run() {
		while(i<=10) {
			//注意,实现runnable接口的类中,不能直接使用getName方法,因为getName方法是Thread类的方法
			System.out.println(Thread.currentThread().getName() + "正在运行" + (i++));
		}
		
	}

/**
 *  通过实现Runnable接口创建线程有3步,类实例化,根据实例化的对象创建线程类,线程子类调用start方法
 *  
 */
	public static void main(String[] args) {
		//类实例化,分配类的实例,在创建Thread时作为参数传递
		PrintRunnable pr = new PrintRunnable();
		//根据实例化的对象创建线程类
		Thread t1 = new Thread(pr);
		//线程子类调用start方法
		t1.start();
		
		//PrintRunnable pr1 = new PrintRunnable();
		Thread t2 = new Thread(pr);
		t2.start();
	}

}

四、线程间的通信和同步

package com.imooc.queue;
//消费者类,不断消耗n值
public class Consumer implements Runnable{
	//消费者和生产者共享queue
	Queue queue;
	//构造方法
	Consumer(Queue queue){
		this.queue=queue;
	}

	@Override
	public void run() {
		while(true){
			//不断获取queue的n值
			queue.get();
			try {
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}	
}
package com.imooc.queue;

//生产者类,不断生产n值
public class Producer implements Runnable{
	//生产者和消费者共享queue 
	Queue queue;
	//构造方法
	Producer(Queue queue){
		this.queue=queue;
	}
	@Override
	public void run() {
		int i=0;
		//这里用了一个死循环,一直输出
		while(true){
			//修改共享queue的n值
			queue.set(i++);
			try {
				//休眠1s
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}	
	}
}
package com.imooc.queue;

public class Queue {
	
	private int n;
	boolean flag=false;
	//flag为false的时候,容器queue中没有数据,需要先生产后获取
	//为true时,有数据,需要先获取再生产
	
	//get方法,获得n的值
	//要加上同步关键字,保证共享资源queue同一时间只有一个线程访问
	public synchronized int get() {
		//flag为false时,没有数据,需要执行等待
		if(!flag){
			try {
				//等待set生成资源
				wait();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
		System.out.println("消费:"+n);
		flag=false;//消费完毕,容器中没有数据
		notifyAll();//唤醒其他线程
		//一般使用这个,因为实际中可能不知道该唤醒哪一个,那把它们都唤醒。
		return n;
	}
	//set方法,设置n的值
	public synchronized void set(int n) {
		//当flag为true,即容器中有数据时,不要再生产了,等待消费完再生产
		if(flag){
			try {
				wait();
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}
		System.out.println("生产:"+n);
		this.n = n;
		flag=true;//生产完毕,容器中已经有数据
		notifyAll();//唤醒其他线程,否则会引起死锁
	}
}
package com.imooc.queue;

public class Test {

	public static void main(String[] args) {
		Queue queue=new Queue();
		//实现runnable接口,创建线程三步走,连起来写
		new Thread(new Producer(queue)).start();
		new Thread(new Consumer(queue)).start();
	}

}