java线程安全与死锁
线程 安全问题的解决方案:sun提供了线程同步机制让我们解决这类问题的。
出现线程安全问题的根本原因:
1. 存在两个或者两个以上 的线程对象,而且线程之间共享着一个资源。
2. 有多个语句操作了共享资源。
java线程同步机制的方式:
方式一:同步代码块
同步代码块的格式:
synchronized(锁对象){
需要被同步的代码...
}
同步代码块要注意事项:
1. 任意的一个对象都可以做为锁对象。
2. 在同步代码块中调用了sleep方法并不是释放锁对象的。
3. 只有真正存在线程安全问题的时候才使用同步代码块,否则会降低效率的。
4. 多线程操作的锁 对象必须 是唯一共享 的。否则无效。
方式二:同步函数 : 同步函数就是使用synchronized修饰一个函数。
同步函数要注意的事项 :
1. 如果是一个非静态的同步函数的锁 对象是this对象,如果是静态的同步函数的锁 对象是当前函数所属的类的字节码文件(class对象)。
2. 同步函数的锁对象是固定的,不能由你来指定 的。
推荐使用: 同步代码块。
原因:
1. 同步代码块的锁对象可以由我们随意指定,方便控制。同步函数的锁对象是固定 的,不能由我们来指定。
需求: 一个银行账户5000块,两夫妻一个拿着 存折,一个拿着卡,开始取钱比赛,每次只能取一千块,要求不准出现线程安全问题。
运行结果:
老公取走了1000块,还剩余4000元
老婆取走了1000块,还剩余3000元
老婆取走了1000块,还剩余2000元
老婆取走了1000块,还剩余1000元
老婆取走了1000块,还剩余0元
取光了...
取光了...
java中同步机制解决了线程安全问题,但是也同时引发死锁现象。
死锁现象出现 的根本原因:
1. 存在两个或者两个以上的线程。
2. 存在两个或者两个以上的共享资源。
运行结果:
张三拿到了遥控器,准备 去拿电池!!
狗娃拿到了电池,准备去拿遥控器!!
出现线程安全问题的根本原因:
1. 存在两个或者两个以上 的线程对象,而且线程之间共享着一个资源。
2. 有多个语句操作了共享资源。
java线程同步机制的方式:
方式一:同步代码块
同步代码块的格式:
synchronized(锁对象){
需要被同步的代码...
}
同步代码块要注意事项:
1. 任意的一个对象都可以做为锁对象。
2. 在同步代码块中调用了sleep方法并不是释放锁对象的。
3. 只有真正存在线程安全问题的时候才使用同步代码块,否则会降低效率的。
4. 多线程操作的锁 对象必须 是唯一共享 的。否则无效。
方式二:同步函数 : 同步函数就是使用synchronized修饰一个函数。
同步函数要注意的事项 :
1. 如果是一个非静态的同步函数的锁 对象是this对象,如果是静态的同步函数的锁 对象是当前函数所属的类的字节码文件(class对象)。
2. 同步函数的锁对象是固定的,不能由你来指定 的。
推荐使用: 同步代码块。
原因:
1. 同步代码块的锁对象可以由我们随意指定,方便控制。同步函数的锁对象是固定 的,不能由我们来指定。
2. 同步代码块可以很方便控制需要被同步代码的范围,同步函数必须是整个函数 的所有代码都被同步了。
- class SaleTicket extends Thread{
- static int num = 50;//票数 非静态的成员变量,非静态的成员变量数据是在每个对象中都会维护一份数据的。
- static Object o = new Object();
- public SaleTicket(String name) {
- super(name);
- }
- @Override
- public void run() {
- while(true){
- //同步代码块
- synchronized (o) {
- if(num>0){
- System.out.println(Thread.currentThread().getName()+"售出了第"+num+"号票");
- try {
- Thread.sleep(100);
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- num--;
- }else{
- System.out.println("售罄了..");
- break;
- }
- }
- }
- }
- }
- public class Demo4 {
- public static void main(String[] args) {
- //创建三个线程对象,模拟三个窗口
- SaleTicket thread1 = new SaleTicket("窗口1");
- SaleTicket thread2 = new SaleTicket("窗口2");
- SaleTicket thread3 = new SaleTicket("窗口3");
- //开启线程售票
- thread1.start();
- thread2.start();
- thread3.start();
- }
- }
需求: 一个银行账户5000块,两夫妻一个拿着 存折,一个拿着卡,开始取钱比赛,每次只能取一千块,要求不准出现线程安全问题。
- class BankThread extends Thread{
- static int count = 5000;
- public BankThread(String name){
- super(name);
- }
- @Override //
- public synchronized void run() {
- while(true){
- synchronized ("锁") {
- if(count>0){
- System.out.println(Thread.currentThread().getName()+"取走了1000块,还剩余"+(count-1000)+"元");
- count= count - 1000;
- }else{
- System.out.println("取光了...");
- break;
- }
- }
- }
- }
- //静态的函数---->函数所属 的类的字节码文件对象--->BankThread.class 唯一的。
- public static synchronized void getMoney(){
- }
- }
- public class Demo1 {
- public static void main(String[] args) {
- //创建两个线程对象
- BankThread thread1 = new BankThread("老公");
- BankThread thread2 = new BankThread("老婆");
- //调用start方法开启线程取钱
- thread1.start();
- thread2.start();
- }
- }
老公取走了1000块,还剩余4000元
老婆取走了1000块,还剩余3000元
老婆取走了1000块,还剩余2000元
老婆取走了1000块,还剩余1000元
老婆取走了1000块,还剩余0元
取光了...
取光了...
java中同步机制解决了线程安全问题,但是也同时引发死锁现象。
死锁现象出现 的根本原因:
1. 存在两个或者两个以上的线程。
2. 存在两个或者两个以上的共享资源。
死锁现象的解决方案: 没有方案。只能尽量避免发生而已。
- class DeadLock extends Thread{
- public DeadLock(String name){
- super(name);
- }
- public void run() {
- if("张三".equals(Thread.currentThread().getName())){
- synchronized ("遥控器") {
- System.out.println("张三拿到了遥控器,准备 去拿电池!!");
- synchronized ("电池") {
- System.out.println("张三拿到了遥控器与电池了,开着空调爽歪歪的吹着...");
- }
- }
- }else if("狗娃".equals(Thread.currentThread().getName())){
- synchronized ("电池") {
- System.out.println("狗娃拿到了电池,准备去拿遥控器!!");
- synchronized ("遥控器") {
- System.out.println("狗娃拿到了遥控器与电池了,开着空调爽歪歪的吹着...");
- }
- }
- }
- }
- }
运行结果:
张三拿到了遥控器,准备 去拿电池!!
狗娃拿到了电池,准备去拿遥控器!!