线程相关知识
1.创建线程的两种方式
- 继承Thread类。
- 实现Runnable接口。(这种方式较为常用)
2.实现Runnable接口的好处
- 将线程的任务从线程的子类中分离出来,进行了单独的封装。按照面向对象的思想将任务的封装成对象。
- 避免了java单继承的局限性。
多线程并发安全之卖票
- 代码
/**
* Created by yuandl on 2016-09-30.
*/
public class RunnableTest implements Runnable {
private int tick = 60;
@Override
public void run() {
while (true) {
if (tick == 0) {
break;
}
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "=========" + tick--);
}
}
public static void main(String[] args) {
RunnableTest runnableTest=new RunnableTest();
new Thread(runnableTest).start();
new Thread(runnableTest).start();
}
}
- 打印结果
Thread-1=========11
Thread-1=========10
Thread-0=========9
Thread-1=========8
Thread-0=========7
Thread-0=========6
Thread-1=========5
Thread-0=========4
Thread-1=========3
Thread-0=========2
Thread-1=========1
Thread-0=========0
Thread-0=========-1
Thread-0=========-2
Thread-0=========-3
- 发现问题,卖票竟然出现了负数,这肯定是有问题的
- 线程安全问题产生的原因:
- 多个线程在操作共享的数据。
- 操作共享数据的线程代码有多条。
- 当一个线程在执行操作共享数据的多条代码过程中,其他线程参与了运算。就会导致线程安全问题的产生。
- 解决思路:
- 就是将多条操作共享数据的线程代码封装起来,当有线程在执行这些代码的时候,其他线程时不可以参与运算的。必须要当前线程把这些代码都执行完毕后,其他线程才可以参与运算。在java中,用同步代码块就可以解决这个问题。
- 同步代码块的格式
synchronized(对象)
{
需要被同步的代码 ;
}
- 同步的好处:解决了线程的安全问题。
- 同步的弊端:相对降低了效率,因为同步外的线程的都会判断同步锁。
- 同步的前提:同步中必须有多个线程并使用同一个锁。
最终线程安全同步的代码
/**
* Created by yuandl on 2016-09-30.
*/
public class RunnableTest implements Runnable {
private int tick = 60;
@Override
public void run() {
while (true) {
synchronized (this) {
if (tick == 0) {
break;
}
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "=========" + tick--);
}
}
}
public static void main(String[] args) {
RunnableTest runnableTest=new RunnableTest();
new Thread(runnableTest).start();
new Thread(runnableTest).start();
}
}
- 执行结果
Thread-1=========10
Thread-1=========9
Thread-1=========8
Thread-1=========7
Thread-1=========6
Thread-1=========5
Thread-1=========4
Thread-1=========3
Thread-1=========2
Thread-1=========1
Process finished with exit code 0
完美解决以上问题
网友评论
第一个冲突的例子中,其实只要把判断条件改成if (tick < 1)就很难出现负数了,要很好的运气才可以发生一次期望的异常结果。
其实方法对的,就是例子的代码不太友好,所以出现了极端结果。