JAVA多线程

【TX】前言:JAVA是少数的几种支持多线程的语言之一,它可以让不同的程序块同时运行,从而使程序更为顺畅,性能也更高,同时也达到了多任务处理的目的。

一:进程和线程概念

1:【进程】是程序的一次动态执行过程,它需要经历从代码加载,代码执行到执行完毕的一个完整过程,这个过程也是进程本身送产生,发展到最终消亡的过程。多进程系统能同时运行过个进程【程序】。由于CPU具备分时机制,所以每个进程都能循环获取到自己的CPU时间片。由于CPU的运行速度很快,所以使得所有程序好像是在“同时”运行一样。

2:【线程】是比进程更小的执行单位,是在进程基础上的进一步划分。多线程是指一个进程在执行的过程中可以产生多个更小的程序单元,这些更小的程序单位称为线程,这些线程可以同时存在,同时运行,多线程是实现并发机制的一种有效手段

二:多线程的实现:继承Thread类;实现Runnable接口

例1:继承Thread类

package com.tmx.Threads;

public class ThreadTest1 extends Thread{
	private String name;

	public ThreadTest1(String name) {
		super();
		this.name = name;
	}
	@Override
	public void run(){         //------------------run()方法用于定义线程的主体
		for (int i = 0; i < 3; i++) {
			System.out.println(name+"执行,i="+i);
		}
	}
	public static void main(String[] args) {
		ThreadTest1 thr1=new ThreadTest1("线程A");
		ThreadTest1 thr2=new ThreadTest1("线程B");
		thr1.start();//------------------start()方法用于启动线程
		thr2.start();
	}
}

例1程序的执行结果:

线程B执行,i=0
线程A执行,i=0
线程A执行,i=1
线程A执行,i=2
线程B执行,i=1
线程B执行,i=2

这里需要注意的是,run()方法只是定义线程的主体,而启动线程则需要调用start()方法

例2:实现Runnable接口

需要注意的是Runnable接口中只定义了一个方法,就是run()方法,所以启动线程就需要依靠Thread类的start()方法。

public class RunnableTest1 implements Runnable{
	private String name;
	public RunnableTest1(String name) {
		super();
		this.name = name;
	}
	@Override
	public void run() {
		for (int i = 0; i < 3; i++) {
			System.out.println(name+"执行,i="+i);
		}
	}
	public static void main(String[] args) {
		RunnableTest1 rnb1=new RunnableTest1("线程A");
		RunnableTest1 rnb2=new RunnableTest1("线程B");
		Thread thread1 = new Thread(rnb1);
		Thread thread2 = new Thread(rnb2);
		thread1.start();
		thread2.start();
	}
}

例2执行结果:

线程A执行,i=0
线程A执行,i=1
线程A执行,i=2
线程B执行,i=0
线程B执行,i=1
线程B执行,i=2

看见这个结果,可能我们会以为这并不是多线程,因为看起来是按顺序执行的。而实际上,这仍然是多线程执行,因为两个线程对象是交错运行的,谁抢到了CPU资源,谁就先运行,所以每次运行的结果也是不同的。

三:Thread类和Runnable接口的区别和联系

1:两者的联系

public class Thread implements Runnable

从Thread类的定义我们可以清楚地发现,Thread类也是Runnanble接口的子类。我们继续看Thread类的部分定义:

public Thread(Runnable target) {
        init(null, target, "Thread-" + nextThreadNum(), 0);
    }

private void init(ThreadGroup g, Runnable target, String name,
                      long stackSize) {
        init(g, target, name, stackSize, null);
    }

@Override
    public void run() {
        if (target != null) {
            target.run();
        }
    }

从Thread类的定义中可见,run方法调用的是Runnable接口中的run方法,所以如果要通过继承Thread类实现多线程,则必须复写run方法

2:两者的区别

实现Runnable接口相对于继承Thread类的优势

(a) 适合多个相同程序代码的线程去处理同一资源的情况

(b)可以避免由于JAVA单继承特性带来的局限

(c)增强了程序的健壮性,代码能够被多个线程共享,代码与数据是独立的

Thread类并不适合多个线程共享资源:

例3:继承Thread类

package com.tmx.Threads;

public class ThreadTest1 extends Thread{
	private String name;
	private int ticket=3;
	
	public ThreadTest1(String name) {
		super();
		this.name = name;
	}
	@Override
	public void run(){
		for (int i = 0; i < 10; i++) {
			if(ticket>0){
				System.out.println(name+"卖票:ticket="+ticket--);
			}
		}
	}
	public static void main(String[] args) {
		ThreadTest1 thr1=new ThreadTest1("售票员A");
		ThreadTest1 thr2=new ThreadTest1("售票员B");
		thr1.start();
		thr2.start();
	}
}

例3运行结果:

售票员B卖票:ticket=3
售票员A卖票:ticket=3
售票员B卖票:ticket=2
售票员A卖票:ticket=2
售票员B卖票:ticket=1
售票员A卖票:ticket=1

可见,继承Thread类,只起到了多线程执行的作用,并没有起到资源共享的效果。

例4:实现Runnable接口

package com.tmx.Threads;

public class RunnableTest1 implements Runnable{
	private String name;
	private int ticket=3;
	
	public RunnableTest1(String name) {
		super();
		this.name = name;
	}
	@Override
	public void run(){
		for (int i = 0; i < 10; i++) {
			if(ticket>0){
				System.out.println(name+"卖票:ticket="+ticket--);
			}
		}
	}
	public static void main(String[] args) {
		RunnableTest1 rnb1=new RunnableTest1("售票员");
		Thread thread1 = new Thread(rnb1);
		Thread thread2 = new Thread(rnb1);
		Thread thread3 = new Thread(rnb1);
		thread1.start();
		thread2.start();
		thread3.start();
	}
}

例4运行结果:

售票员卖票:ticket=3
售票员卖票:ticket=1
售票员卖票:ticket=2

此处一定要注意这两者启动多个线程的方式是不一样的,而且要观察例4的运行结果顺序。所以在实际开发中应尽量实现Runnable接口。

四:线程的状态

任何线程一般都5中状态,即:创建,就绪,运行,阻塞,死亡。


1:创建状态

线程的创建通过Thread类的构造方法来实现:

Thread thread = new Thread();

Thread thread = new Thread(Runnable target);

线程创建后它即有了相应的内存空间和其他资源,但此时它是不可运行的

2:就绪状态

调用start()方法可以启动线程,此时线程处于就绪状态,等待CPU服务,表名它已经具备了运行的条件

3:当就绪状态获得处理器资源是,线程被调用进入了运行状态,此时,自动调用改线程对象的run()方法。run方法定义了线程的操作和功能。

4:堵塞状态

在可执行状态下,调用sleep(),suspend(),wait()方法都会时线程进入阻塞状态,堵塞时,线程不能进入排队队列,只有当引起阻塞的原因被消除后,线程才可以转入就绪状态。

5:死亡状态

线程调用stop()方法或者run()方法执行结束后,即处于死亡状态,死亡状态的线程不具有继续运行的能力。

从线程状态可以看出,run()方法只是定义线程的主体功能,而start()方法才能启动线程。

五:线程的常用方法

Runnable接口中只有一个run()方法,所以线程的操作方法基本都在Thread类中。

1:线程的构造方法

Thread thread = new Thread(String name);
Thread thread = new Thread(Runnable target);
Thread thread = new Thread(String name,Runnable target);

线程主要有三个常见构造方法,其中String参数用于设置线程名称

例1:设置及获取当前线程的名称

public class RunnableTest1 implements Runnable{
	@Override
	public void run(){
		for (int i = 0; i < 2; i++) {
			String name2 = Thread.currentThread().getName();
			System.out.println(name2);
		}
	}
	public static void main(String[] args) {
		RunnableTest1 rnb1=new RunnableTest1();
		Thread thread1 = new Thread(rnb1);
		Thread thread2 = new Thread(rnb1);
		Thread thread3 = new Thread(rnb1,"线程A");
		thread1.start();
		thread3.start();
		thread2.start();
	}
}

例1运行结果:

Thread-0
Thread-0
线程A
线程A
Thread-1
Thread-1

可见当没有为一个线程指定一个明确的名称时,系统在使用时会为线程分配一个名称,默认格式为Thread-Xx

例1_1:主方法线程

public class RunnableTest1 implements Runnable{
	@Override
	public void run(){
		for (int i = 0; i < 2; i++) {
			String name2 = Thread.currentThread().getName();
			System.out.println("当前线程名称为:"+name2);
		}
	}
	public static void main(String[] args) {
		RunnableTest1 rnb1=new RunnableTest1();
		rnb1.run();
		
		Thread thread1 = new Thread(rnb1,"线程A");
		thread1.start();
	}
}

例1_1运行结果:

当前线程名称为:main
当前线程名称为:main
当前线程名称为:线程A
当前线程名称为:线程A

由例2可见主方法也是一个线程。在JAVA中所有线程都是同时启动的,哪个线程先抢到了CPU资源,那个线程就先运行。

java在运行时至少会启动两个线程,一个是main线程,另一个是垃圾收集线程

例1_2:

public class RunnableTest1 implements Runnable{
	@Override
	public void run(){
		for (int i = 1; i <= 2; i++) {
			String name1 = Thread.currentThread().getName();
			System.out.println(name1+"线程运行:"+name1);
			System.out.println(name1+"线程运行:"+name1);
		}
	}
	public static void main(String[] args) throws InterruptedException {
		RunnableTest1 rnb1=new RunnableTest1();
		Thread thread1 = new Thread(rnb1,"线程A");
		Thread thread2 = new Thread(rnb1,"线程B");
		thread1.start();
		thread2.start();
	}
}

例1_2运行结果:

线程A线程运行:线程A
线程B线程运行:线程B
线程A线程运行:线程A
线程A线程运行:线程A
线程B线程运行:线程B
线程B线程运行:线程B
线程B线程运行:线程B
线程A线程运行:线程A
再次观察线程的无序。

例2:isAlive()判断线程是否启动

public class RunnableTest1 implements Runnable{
	@Override
	public void run(){
		for (int i = 0; i < 2; i++) {
			String name2 = Thread.currentThread().getName();
			System.out.println("当前线程名称为:"+name2);
		}
	}
	public static void main(String[] args) {
		RunnableTest1 rnb1=new RunnableTest1();
		Thread thread1 = new Thread(rnb1,"线程A");
		System.out.println("未执行start方法前,线程是否启动:"+thread1.isAlive());
		thread1.start();
		System.out.println("执行start方法后,线程是否启动:"+thread1.isAlive());
		for (int i = 0; i < 2; i++) {
			System.out.println("main方法执行:"+i);
		}
		System.out.println("执行main方法后,线程是否启动:"+thread1.isAlive());
	}
}

例2运行结果1:

未执行start方法前,线程是否启动:false
执行start方法后,线程是否启动:true
main方法执行:0
当前线程名称为:线程A
main方法执行:1
当前线程名称为:线程A
执行main方法后,线程是否启动:true

例2运行结果2:

未执行start方法前,线程是否启动:false
执行start方法后,线程是否启动:true
main方法执行:0
当前线程名称为:线程A
当前线程名称为:线程A
main方法执行:1
执行main方法后,线程是否启动:false

从这两个结果我们可以发现:

main方法是一个线程;

线程是否存活,要看谁先执行结束。

例3:线程的强制执行 join()方法

使用join()方法可以让一个线程强制执行,在执行期间,其他线程无法运行,必须等待此线程结束后才可以继续执行

public class RunnableTest1 implements Runnable{
	@Override
	public void run(){
		for (int i = 1; i <= 5; i++) {
			String name2 = Thread.currentThread().getName();
			System.out.println(name2+"线程运行:"+i);
		}
	}
	public static void main(String[] args) throws InterruptedException {
		RunnableTest1 rnb1=new RunnableTest1();
		Thread thread1 = new Thread(rnb1,"线程A");
		thread1.start();
		for (int i = 1; i <= 5; i++) {
			if(i>3){
				thread1.join();
			}
			System.out.println("main线程运行:"+i);
		}
	}
}

例3执行结果:

main线程运行:1
main线程运行:2
线程A线程运行:1
main线程运行:3
线程A线程运行:2
线程A线程运行:3
线程A线程运行:4
线程A线程运行:5
main线程运行:4
main线程运行:5
当main线程执行3次后,就会挂起,等候线程A运行结束后才会继续执行main线程

例4:线程的休眠:sleep()

线程休眠直接调用sleep()方法即可,这是一个Static关键字修饰的方法

public class RunnableTest1 implements Runnable{
	@Override
	public void run(){
		for (int i = 1; i <= 2; i++) {
			try {
				Thread.sleep(2000L);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			String name1 = Thread.currentThread().getName();
			SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
			String format = sdf.format(new Date());
			System.out.println(name1+"线程运行时间:"+format);
		}
	}
	public static void main(String[] args) throws InterruptedException {
		RunnableTest1 rnb1=new RunnableTest1();
		Thread thread1 = new Thread(rnb1,"线程A");
		thread1.start();
	}
}

例4运行结果:

线程A线程运行时间:2018-03-11 16:12:31
线程A线程运行时间:2018-03-11 16:12:33

例5:线程的终止:interrupt()方法

public class RunnableTest1 implements Runnable{
	@Override
	public void run(){
		System.out.println("1:进入run方法");
		try {
			Thread.sleep(2000L);
			System.out.println("2:休眠2s已结束");
		} catch (InterruptedException e) {
			System.out.println("3:休眠被终止,抛出异常:"+e);
			return;
		}
		System.out.println("4:run方法结束");
	}
	public static void main(String[] args) throws InterruptedException {
		RunnableTest1 rnb1=new RunnableTest1();
		Thread thread1 = new Thread(rnb1,"线程A");
		thread1.start();
		Thread.sleep(1000L);
		thread1.interrupt();
	}
}

例5运行结果:

1:进入run方法
3:休眠被终止,抛出异常:java.lang.InterruptedException: sleep interrupted

由于线程休眠期间,主方法将线程中断,休眠一旦中断之后将执行catch中的内容,由于使用了return,所以"4:run方法结束"语句并没有执行。

例6:后台线程

在JAVA中,只要前台有一个线程在运行,则整个JAVA进程都不会消失,所以此时可以设置一个后台线程,这样即使java进程结束了,此后台线程依然会继续执行。此时就需要使用setDaemon()方法

public class RunnableTest1 implements Runnable{
	@Override
	public void run(){
		while(true){
			String name2 = Thread.currentThread().getName();
			System.out.println(name2+"线程运行");
		}
	}
	public static void main(String[] args) throws InterruptedException {
		RunnableTest1 rnb1=new RunnableTest1();
		Thread thread1 = new Thread(rnb1,"线程A");
		thread1.setDaemon(true);
		thread1.start();
//		System.out.println("main线程运行");
	}
}

例6运行结果:

线程A线程运行

需要注意的是,程序也可能什么也不打印,但不会出现死循环,因为死循环线程操作已经被设置成后台运行了。

例7:线程的优先级

Java线程优先级有三种,他们都是常量

public static final int MIN_PRIORITY;//------最低优先级,常量为1
public static final int NORM_PRIORITY;//-----默认中等优先级,常量为5
public static final int MAX_PRIORITY;//------最高优先级,常量为10
public class RunnableTest1 implements Runnable{
	@Override
	public void run(){
		for (int i = 0; i < 3; i++) {
			String name = Thread.currentThread().getName();
			System.out.println(name+"线程运行");
		}
	}
	public static void main(String[] args) throws InterruptedException {
		RunnableTest1 rnb1=new RunnableTest1();
		Thread thread1 = new Thread(rnb1,"线程A");
		Thread thread2 = new Thread(rnb1,"线程B");
		Thread thread3 = new Thread(rnb1,"线程C");
		thread1.setPriority(Thread.MAX_PRIORITY);//thread1.setPriority(10)
		thread2.setPriority(Thread.NORM_PRIORITY);//thread1.setPriority(5)
		thread3.setPriority(Thread.MIN_PRIORITY);//thread1.setPriority(1)
		thread1.start();
		thread2.start();
		thread3.start();
	}
}

例7运行结果1:

线程A线程运行
线程A线程运行
线程A线程运行
线程B线程运行
线程B线程运行
线程B线程运行
线程C线程运行
线程C线程运行
线程C线程运行

例7运行结果2:

线程B线程运行
线程C线程运行
线程C线程运行
线程C线程运行
线程A线程运行
线程A线程运行
线程B线程运行
线程A线程运行
线程B线程运行

由结果可见,并非线程的优先级越高,此线程就一定会优先执行,哪个线程先执行将有CPU的调度决定。其特点为:

(a),线程优先级具有继承的特性,比如线程1启动线程2,那么线程2的优先级就和线程1的优先级是一样的
(b),线程的优先级只能确保CPU尽量将执行的资源让给优先级高的线程用,但不保证定义的高优先级的线程的大部分都能先于低优先级的线程执行完。
(c),线程的优先级具有随机性,也就是高优先级的线程不一定每一次都先执行完

主方法main的优先级为5

线程基础就先到这里。




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值