java线程全面解析

java线程全面解析



线程的实现方式

 java实现线程的方式有五种,分别是继承Thread类、实现Runable接口、实现Callable接口、使用线程池接口ExecutorService。下面将对其做一一介绍。


一、继承Thread类

 继承Thread类,本质上还是实现Runable接口,并实现相应的run方法,相关代码如下所示:

public class MyThread extends Thread {  
   public void run() {  
       System.out.println("MyThread.run()");  
   }  
}
public class Main {
	public static void main(String[] args) {
		MyThread mt = new MyThread();
		Thread t1 = new Thread(mt);
		Thread t2 = new Thread(mt);
		t1.start();
		t2.start();
	} 
}

 注意:线程启动必须用start()方法,由于start方法涉及到系统的资源相关调度,而run方法就是普通执行。

二、实现Runable接口

 继承Thread类本质上也是实现Runable接口,相应方法如下所示。相应的实现代码如下所示:

public class MyThread implements Runnable{ // 实现Runnable接口
    @Override
    public void run() {
       System.out.println("线程执行");
    }
}
public class Test {
    public static void main(String[] args) {
    	MyThread mt = new MyThread();
    	Thread thread = new Thread(mt);        
		thread.start(); 
    }
}

三、实现Callable接口

 实现Callable接口通过FutureTask包装器来创建Thread线程,相应的实现代码如下所示:

public class CallableDemo implements Callable<String> {
    @Override
    public String call() throws Exception {
        for (int i = 0; i < 10; i++) {
            String s = "线程" + i + "被执行了";
            System.out.println(s);
        }
        return "线程执行完毕";
    }
}
public class CallableDemo implements Callable<String> {
    @Override
    public String call() throws Exception {
        for (int i = 0; i < 10; i++) {
            String s = "线程" + i + "被执行了";
            System.out.println(s);
        }
        return "线程执行完毕";
    }
}

 Callable与Runable不同的是,Callable有相应的返回值,且非实现run方法,而是call方法。

四、使用线程池接口ExecutorService

 使用线程池接口ExecutorService需要结合Callable、Future实现有返回结果的线程,相应实现代码如下所示:

public class MyCallable implements Callable<Object> {
    private String taskNum;
    public MyCallable(String taskNum) {
        this.taskNum = taskNum;
    }
    @Override
    public Object call() throws Exception {
        return taskNum + "任务启动";
    }
}

public class ExecutorServiceDemo {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        System.out.println("----程序开始运行----");
        // 创建线程池
        int taskSize = 5;
        ExecutorService pool = Executors.newFixedThreadPool(taskSize);
        // 创建多个有返回值的任务
        List<Future> list = new ArrayList<>();
        for (int i = 0; i < taskSize; i++) {
            MyCallable callable = new MyCallable(i + " ");
            // 执行任务,并获取future对象
            Future<Object> f = pool.submit(callable);
            list.add(f);
        }
        pool.shutdown();
        // 获取所有并发任务的执行结果
        for (Future future : list) {
            System.out.println(future.get());
        }
    }
}

 Future作为执行的记录接口。

线程的状态变化

 要想实现多线程,必须在主线程中创建新的线程对象。任何线程一般具有5种状态,即创建,就绪,运行,阻塞,终止。下面分别介绍一下这几种状态:

  • 创建状态
     在程序中用构造方法创建了一个线程对象后,新的线程对象便处于新建状态,此时它已经有了相应的内存空间和其他资源,但还处于不可运行状态。新建一个线程对象可采用Thread 类的构造方法来实现,例如 “Thread thread=new Thread()”。
  • 就绪状态
     新建线程对象后,调用该线程的 start() 方法就可以启动线程。当线程启动时,线程进入就绪状态。此时,线程将进入线程队列排队,等待 CPU 服务,这表明它已经具备了运行条件。
  • 运行状态
     当就绪状态被调用并获得处理器资源时,线程就进入了运行状态。此时,自动调用该线程对象的 run() 方法。run() 方法定义该线程的操作和功能。
  • 阻塞状态
     一个正在执行的线程在某些特殊情况下,如被人为挂起或需要执行耗时的输入/输出操作,会让 CPU 暂时中止自己的执行,进入阻塞状态。在可执行状态下,如果调用sleep(),suspend(),wait() 等方法,线程都将进入阻塞状态,发生阻塞时线程不能进入排队队列,只有当引起阻塞的原因被消除后,线程才可以转入就绪状态。
  • 死亡状态
     线程调用 stop() 方法时或 run() 方法执行结束后,即处于死亡状态。处于死亡状态的线程不具有继续运行的能力。

线程的操作方法和状态

 线程的操作方法包括线程强制运行、线程休眠、线程中断、守护线程(后台线程)、线程礼让(yield)、线程同步(synchronized)等,线程的主要状态为线程死锁(是有synchronized同步资源造成的)。
 相应的链接如下所示:https://www.cnblogs.com/java1024/archive/2019/11/28/11950129.html

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值