Java多线程

来源:互联网 发布:js getelementbyid 编辑:程序博客网 时间:2024/06/06 17:52

引文

在复习多线程之前,我们得先明白一下几个概念:
①进程与线程的区别:
http://blog.csdn.net/mingming_shiwo/article/details/76037935

②并行与并发的区别:
并行:两个或多个事件或活动在同一时刻发生;多个程序在同一时刻在多个CPU上同时执行;
并发:一个CPU或者多个CPU在若干程序之间多路复用;并发性是对有限的物理资源强制行使多用户共享以提高效率。
打个比方:一个母亲给两个孩子喂饭,你一口他一口,表面上看是两个孩子一块吃饭,这就是并发;而父亲和母亲同时一人给一个孩子喂饭,这就是并行。

③同步与异步:
同步:指发送一个请求,需要等待返回,然后才能够发送下一个请求,有个等待的过程;
异步:指发送一个请求,不需要等待,随时可以发送下一个请求。
打个比方:下班了,同事叫你去吃饭,你没听见,然后同时就一直叫你,直到你回答说是去或者不去,这就叫同步;而同事叫你吃饭,在叫完你之后就直接自己去吃了,并不理会你去不去,这就叫异步。

线程的状态及状态转移图

NEW:刚创建还没启动
RUNNABLE:就绪状态,随时可以执行
RUNNNING:正在执行
BLOCKED:阻塞状态,等待持有锁
WAITING:处理等待状态
TERMINATED:终止状态
TIMEWAITING:睡眠状态

线程状态转移图

多线程的实现的三种方式

一、继承Thread类,并重该类的run()方法:

class MyThread extends Thread{    @Override    public void run(){        System.out.println(Thread.currentThread().getName());    }}public class ThreadDemo {    public static void main(String[] args) {        // TODO Auto-generated method stub        System.out.println("Main线程开始启动:");        //创建一个新的线程myThread1,此线程进入新建状态        Thread myThread1 = new MyThread();        //创建一个新的线程myThread2,此线程进入新建状态        Thread myThread2 = new MyThread();        //调用start()方法使线程进入就绪状态        myThread1.start();        myThread2.start();        try {            //join()是为了等待主线程终止            myThread1.join();            myThread2.join();        } catch (InterruptedException e) {            // TODO Auto-generated catch block            e.printStackTrace();        }        System.out.println("Main线程结束");    }}

执行结果:

Main线程开始启动:Thread-0Thread-1Main线程结束

二、实现Runnable接口,并重写该接口的run()方法:

class MyThread implements Runnable{    @Override    public void run() {        // TODO Auto-generated method stub        System.out.println(Thread.currentThread().getName());    }}public class ThreadDemo {    public static void main(String[] args) {        // TODO Auto-generated method stub        System.out.println("Main线程开始启动:");        //创建一个Runnable接口的实现类的对象        Runnable myThread = new MyThread();        //将myThread作为Thread target来创建新的线程myThread1,此线程进入新建状态        Thread myThread1 = new Thread(myThread);        //将myThread作为Thread target来创建新的线程myThread2,此线程进入新建状态        Thread myThread2 = new Thread(myThread);        //调用start()方法使线程进入就绪状态        myThread1.start();        myThread2.start();        //join()是为了等待主线程终止        try {            myThread1.join();            myThread2.join();        } catch (InterruptedException e) {            // TODO Auto-generated catch block            e.printStackTrace();        }        System.out.println("Main线程结束");    }}

执行结果:

Main线程开始启动:Thread-0Thread-1Main线程结束

三、使用Callable和Future接口创建线程。具体是创建Callable接口的实现类,并实现clall()方法并使用FutureTask类来包装Callable实现类的对象,且以此FutureTask对象作为Thread对象的target来创建线程:

import java.util.concurrent.Callable;import java.util.concurrent.ExecutionException;import java.util.concurrent.FutureTask;class MyThread implements Callable<Integer>{    @Override    public Integer call() throws Exception {        // TODO Auto-generated method stub        int count = 0;        for(int i = 0; i < 5; i++){            System.out.println(Thread.currentThread().getName() + " " + i);            count += i;        }        return count;    }}public class ThreadDemo {    public static void main(String[] args) {        // TODO Auto-generated method stub        System.out.println("Main线程开始启动:");        //创建Callable接口的实现类MyThread的对象        Callable<Integer> myCallable = new MyThread();        //使用FutureTask类来包装myCallable对象        FutureTask<Integer> ft = new FutureTask<>(myCallable);        //FutureTask对象作为Thread对象的target创建新的线程MyThread1        Thread myThread1 = new Thread(ft);          myThread1.start();        try {            System.out.println("count = " + ft.get());        } catch (InterruptedException | ExecutionException e) {            // TODO Auto-generated catch block            e.printStackTrace();        }        System.out.println("Main线程结束");    }

执行结果:

Main线程开始启动:Thread-0 0Thread-0 1Thread-0 2Thread-0 3Thread-0 4count = 10Main线程结束
原创粉丝点击