一:什么是多线程:
多线程是这样一种机制,它允许在程序中并发执行多个指令流,每个指令流都称为一个线程,彼此间互相独立。
线程又称为轻量级进程,它和进程一样拥有独立的执行控制,由操作系统负责调度,区别在于线程没有独立的存储空间,
而是和所属进程中的其它线程共享一个存储空间,这使得线程间的通信远较进程简单。
多个线程的执行是并发的,也就是在逻辑上“同时”,而不管是否是物理上的“同时”。如果系统只有一个CPU,那么真正的“同时”是不可能的,但是由于
CPU的速度非常快,用户感觉不到其中的区别,因此我们也不用关心它,只需要设想各个线程是同时执行即可。 多线程和传统的单线程在程序设计上
最大的区别在于,由于各个线程的控制流彼此独立,使得各个线程之间的代码是乱序执行的,由此带来的线程调度,同步等问题。
二.线程的状态
1.新建(new) 用new语句创建的线程对处于新建状态,此时它和其它Java对象一样,仅仅在Heap中被分配了内存。当一个线程处于新建状态时,它仅仅是一个空的线程对象,系统不为它分配资源。 Thread t = new Thread(new Runner());start
2. 就绪(Runnable) 程序通过线程对象调用启动方法start()后,系统会为这个线程分配它运行时所需的除处理器之外的所有系统资源。这时,它处在随时可以运行的状态,在随后的任意时刻,只要它获得处理器即会进入运行状态。
3.运行状态:处于这个状态的线程占用CPU,执行程序的代码
4.阻塞状态:当线程处于阻塞状态时,java虚拟机不会给线程分配CPU,直到线程重新进入就绪状态,它才有机会转到运行状态。
三.创建线程的两种方法
1.继承Thread类来创建多线程。
public class ThreadTest2 extends Thread{ public static void main(String[] args) { Thread thread1 = new ThreadTest2(); Thread thread2 = new ThreadTest2(); thread1.setName("a"); thread2.setName("b"); thread1.start(); thread2.start(); } public void run() { System.out.println(Thread.currentThread().getName()+" say Hello!!"); } }
2.通过实现Runnable接口线程创建。
public class ThreadTest1 implements Runnable{ public static void main(String[] args) { ThreadTest1 test = new ThreadTest1(); Thread thread1 = new Thread(test); Thread thread2 = new Thread(test); thread1.setName("a"); thread2.setName("b"); thread1.start(); thread2.start(); } @Override public void run() { // TODO Auto-generated method stub System.out.println(Thread.currentThread().getName()+" say Hello!!"); } }
采用继承Thread类方式:
(1)优点:编写简单,如果需要访问当前线程,无需使用Thread.currentThread()方法,直接使用this,即可获得当前线程。
(2)缺点:因为线程类已经继承了Thread类,所以不能再继承其他的父类。
采用实现Runnable接口方式:
(1)优点:线程类只是实现了Runable接口,还可以继承其他的类。在这种方式下,可以多个线程共享同一个目标对象,所以非常适合多个相同线程来处理同一份资源的情况,从而可以将CPU代码和数据分开,形成清晰的模型,较好地体现了面向对象的思想。
(2)缺点:编程稍微复杂,如果需要访问当前线程,必须使用Thread.currentThread()方法。
四.使用多线程实现一个售票窗口
实现一个简单的售票系统。
public class TickWindow implements Runnable{ int ticknum =20; @Override public void run() { while(ticknum>0) { selltick(); } } public synchronized void selltick() { if(ticknum>0) { ticknum=ticknum-1; System.out.println(Thread.currentThread().getName()+" 售出一张,还剩下"+ticknum+"张票"); } try { Thread.sleep(500); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } } public static void main(String[] args) { TickWindow tickWindow = new TickWindow(); Thread wid1 = new Thread(tickWindow); Thread wid2 = new Thread(tickWindow); Thread wid3 = new Thread(tickWindow); wid1.setName("窗口1"); wid2.setName("窗口2"); wid3.setName("窗口3"); wid1.start(); wid2.start(); wid3.start(); } }