一、在Java中有两种方法实现多线程:
继承Thread类和实现Runnable接口。
实现Runnable接口对于继承Thread接口有如下显著优势:
1、 适合多个相同程序代码的线程去处理同一资源,因为可以只建一个对象,然后new Thread()多次实现多线程,这样资源就是共享的
2、 可以避免Java的单继承带来的局限
3、 增强了程序的健壮性,代码能够被多个线程共享,代码与数据是独立的。
所以尽量使用Runnable
二、线程的状态
1、 创建状态:有了相应的内存空间和其他资源,但还处于不可运行状态
2、 就绪状态:调用start()方法就进入就绪状态,线程将进入线程队列排队,等待CPU服务
3、 运行状态:获得CPU资源时,自动调用run()方法
4、 堵塞状态:一个正在执行的线程在某些特殊的情况下,如被认为挂起或需要执行耗时的输入输出操作,将让出CPU并暂时中止自己的执行,进入堵塞状态。在可执行状态下,如果调用sleep()、suspend()、wait()等方法,线程将进入堵塞状态。堵塞时线程不能进入排队队列,只有引起堵塞的原因消除后,才能重新进入就绪状态。
5、 死亡状态:线程调用stop()方法或者run()方法执行结束后。
三、线程操作的相关方法:
在Java中多有的贤臣个都是同时启动的,主方法实际上也是一个线程。
1、取得和设置线程名字:getName() setName()
2、判断线程是否启动:isAlive()
3、线程的强制运行:join() 强制运行期间,其他线程无法运行,必须等待此线程完成之后才可以继续执行。
4、线程的休眠:Thread.sleep();休眠时不进入排队队列
5、中断线程:interrupt()
6、后台线程:在Java程序中只要前台有一个程序在运行,整个Java程序进程都不会消失,所以此时可以设置一个后台线程,这样即使Java进程结束了,此后太线程依然会继续执行。要想实现这样操作,直接使用setDaemon()方法即可。
如果出了这个后台线程别的线程都执行完了Java程序就会结束了。
6、 线程的优先级:setPriority()
不过并非线程的优先级越高就一定会先执行,哪个线程先执行将有CPU的调度来决定。
7、 线程的礼让:yield()方法将一个线程的操作暂时让给其他线程执行。
四、同步与死锁:
1、同步
问题的引出:
【 知识条:java的内存模型是对每一个进程有一个主内存, 每个线程有自己的内存, 他们从主内存中取数据, 然后计算, 再存入主内存中。
】
一个多线程的程序如果通过Runnable接口实现,则意味着类中的属性将被多个线程共享,那么这样一来就会造成一种问题,如果这多个线程要操作同一资源就有可能出现资源的同步问题。
问题的解决:引入同步的概念:就是指多个操作在同一时间段内只能有一个线程进行,其他线程要等待此线程完成之后才可以继续执行。
【知识条:JAVA中规定对非DOUBLE, LONG的原始类型的取和存操作为原子操作。 其实就是对一个字(32位)的取,存位原始操作, 因为DOUBLE, LONG为两个字节的长度, 所以其取, 存为非原子操作。 如果想把他们也变为原子操作, 可以用VOLATILE关键字来修饰
】
需要明确的几个问题:
1)synchronized关键字可以作为函数的修饰符,也可作为函数内的语句,也就是平时说的同步方法和同步语句块。如果再细的分类,synchronized可作用于instance变量、object reference(对象引用)、static函数和class literals(类名称字面常量)身上。
2)无论synchronized关键字加在方法上还是对象上,它取得的锁都是对象,而不是把一段代码或函数当作锁――而且同步方法很可能还会被其他线程的对象访问。
3)每个对象只有一个锁(lock)与之相关联。
4)实现同步是要很大的系统开销作为代价的,甚至可能造成死锁,所以尽量避免无谓的同步控制。
使用同步解决问题:解决资源共享的同步操作,可以使用同步代码块(synchronized(同步对象){需要同步的代码;})和金额同步方法(用关键字synchronized将一个方法声明为同步方法)两种方式完成。
2、死锁:
问题引出:过多的同步也会产生死锁问题,死锁就是指两个线程都在等待彼此先完成,造成了程序的停滞。
五、线程操作案例——生产者与消费者(咱略)
六、线程的生命周期
Stop() suspend() resume() 方法已不推荐使用,容易出现死锁。所以可以通过标签实现暂停重新开始和停止。
P:详细参考:《Java 开发实战经典》李兴华