美文网首页
多线程基础2:同步方法与同步块

多线程基础2:同步方法与同步块

作者: 赶路人_3864 | 来源:发表于2019-04-23 21:29 被阅读0次

并发原因

存在临界资源(临界值)
存在多条线程共同操作共享数据
话不多说直接看例子(这是小菜上一篇博文中的例子 )
模仿抢票:

package thread;

public class Ticket_garbbing implements Runnable{
    private int tickrtSize=100;
    @Override
    public void run() {
        while(true) {
            if (tickrtSize < 0) {
                break;
            }
            try {
                Thread.sleep(200);//模仿网络延时此时出现负数出现并发问题
            } catch (InterruptedException e) {//并发需要保证线程安全
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"/"+tickrtSize--);
        }
    }

    public static void main(String[] args) {
        Ticket_garbbing tg1=new Ticket_garbbing();
        new Thread(tg1,"io1").start();
        new Thread(tg1,"io2").start();
        new Thread(tg1,"io3").start();
        new Thread(tg1,"io4").start();
        new Thread(tg1,"io5").start();
        new Thread(tg1,"io6").start();
    }
}

运行结果:


image.png

为什么这里会出现负数呢?按道理程序运行到0就应该结束了。其实这是因为此处发生了并发。我们都知道每一个线程都有一个独立的线程空间。当前有六个线程,要从主内存中拿到票,假如他们都拿到票此时票数为1,此时io5将票抢到,此时票为0,那剩下拿到票的线程只好继续操作。如下图:


image.png

解决方法

同步方法(加锁锁定当前对象也就是this)
同步块(加锁锁定某对象通常在块上加锁)
代码如下:


package thread2;
/*
* 线程不安全
* 相同或者负数都不安全
* synchrohized同步方法
* 同步块
* */
public class webtickets implements Runnable{
    private int tickrtSize=100;
    private boolean flag=true;
    //线程安全同步
    public synchronized void stop(){
        if(tickrtSize<=0){
            flag=false;
            return;
        }
        try {
            Thread.sleep(200);//模仿网络延时此时出现负数出现并发问题
        } catch (InterruptedException e) {//并发需要保证线程安全
            e.printStackTrace();
        }
        System.out.println(Thread.currentThread().getName()+"/"+tickrtSize--);
    }
    @Override
    public void run() {
        while (flag){
            stop();
        }

    }

    public static void main(String[] args) {
        webtickets tg1=new webtickets();
        new Thread(tg1,"io1").start();
        new Thread(tg1,"io2").start();
        new Thread(tg1,"io3").start();
       /* new Thread(tg1,"io4").start();
        new Thread(tg1,"io5").start();
        new Thread(tg1,"io6").start();*/
    }
}

运行结果:


image.png

加锁原理:
举个简单的例子吧,比如多个人此时在排队上厕所,如果厕所可以随便可以出入,那可能就会造成两个同时争抢厕所的情况。那如果厕所可以上锁,一人为用完其他人也不能使用,必须等厕所解锁以后才能允许下一个人使用。其实加锁就是在其资源上加上一个限制,一次只能一个线程使用资源。

我们再来看看同步块的例子:

未加锁:


package thread2;

public class GetMoney {
    public static void main(String[] args) {
        Accout accout=new Accout(100,"tom");
        Drawing d1=new Drawing(accout,10,accout.money);
        Drawing d2=new Drawing(accout,20,accout.money);
        new Thread(d1).start();
        new Thread(d2).start();
    }
}

class Accout{
    int money;
    String name;
    public Accout(int money, String name) {
        this.money = money;
        this.name = name;
    }
}

class Drawing implements Runnable{
    Accout accout;
    int drawingMoney;
    int drawtotal;

    public Drawing(Accout accout, int drawingMoney, int drawtotal) {
        this.accout = accout;
        this.drawingMoney = drawingMoney;
        this.drawtotal = drawtotal;
    }

    @Override
    public void run() {
        test();
    }
    public  void test(){
      
           accout.money -= drawingMoney;
           System.out.println(accout.money + "--" + accout.name + "-" + drawtotal);
      
    }
}


加锁:


package thread2;

public class GetMoney {
    public static void main(String[] args) {
        Accout accout=new Accout(100,"tom");
        Drawing d1=new Drawing(accout,10,accout.money);
        Drawing d2=new Drawing(accout,20,accout.money);
        new Thread(d1).start();
        new Thread(d2).start();
    }
}

class Accout{
    int money;
    String name;
    public Accout(int money, String name) {
        this.money = money;
        this.name = name;
    }
}

class Drawing implements Runnable{
    Accout accout;
    int drawingMoney;
    int drawtotal;

    public Drawing(Accout accout, int drawingMoney, int drawtotal) {
        this.accout = accout;
        this.drawingMoney = drawingMoney;
        this.drawtotal = drawtotal;
    }

    @Override
    public void run() {
        test();
    }
    public  void test(){
    //此时针对某个对象如果将锁加在test并发非同步
       synchronized (accout) {//更加的明确
           accout.money -= drawingMoney;
           System.out.println(accout.money + "--" + accout.name + "-" + drawtotal);
       }
    }
}

第三个例子


package thread2;

import java.util.ArrayList;
import java.util.List;

public class Test3 {
    public static void main(String[] args) {
        List<String> list=new ArrayList<>();
        for(int i=0;i<10000;i++){
            new Thread(()->{
                synchronized (list) {
                    list.add("1");
                }
            }).start();
        }
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(list.size());
    }
}


未加锁结果:
8592
加锁结果:
10000

小结
并发同步是要牺牲性能的。


作者:欲指_Object
来源:CSDN
原文:https://blog.csdn.net/apple596529/article/details/89481592
版权声明:本文为博主原创文章,转载请附上博文链接!

相关文章

  • 多线程基础2:同步方法与同步块

    并发原因 存在临界资源(临界值)存在多条线程共同操作共享数据话不多说直接看例子(这是小菜上一篇博文中的例子 )模仿...

  • 释放锁的三种情况

    1.执行完同步方法或同步代码块 2.在同步方法或同步代码块中调用wait方法 3.在同步方法或同步代码块中发生异常...

  • 【synchronized】同步方法与同步块

    这里我们分三块来解读学习: 同步方法思想同步块思想同步方法与同步块的区别 1. 同步方法思想 上面是我画的同步方法...

  • Java并发之synchronized

    Java多线程同步关键词是常用的多线程同步手段。它可以修饰静态类方法,实例方法,或代码块。修饰static静态方法...

  • JUC学习笔记三

    JUC学习笔记三 用于解决多线程同步问题的方式 隐式锁(synchronized) 同步代码块 同步方法 显式锁(...

  • 同步锁Lock

    用于解决多线程安全问题的方式:synchronized 隐式锁 同步代码块 同步方法 jdk1.5后新增 同步锁...

  • 多线程"锁重入"概念

    Java多线程锁重入是指: 在已经获得锁的同步方法或同步代码块内部可以调用锁定对象的其他同步方法, 不需要重新获取...

  • Java之同步代码块

    Java多线程的同步代码块 synchronized(对象){ 需要同步的代码 } 同步代码块可以解决安全...

  • 多线程之_释放锁

    以下操作将会释放锁 1、当前线程的同步方法、同步代码块执行完毕2、当前线程在同步代码块、同步方法中遇到break、...

  • Java多线程同步2——同步方法

    java多线程同步除了上文说到的同步代码块,还可以使用同步方法,还是银行取钱的那个问题,代码如下 public c...

网友评论

      本文标题:多线程基础2:同步方法与同步块

      本文链接:https://www.haomeiwen.com/subject/vwtigqtx.html