单线程执行设计模式(有一个门始终只有一个通过)

在我们的业务开发中,在多线程的情况下始终会有资源的公用,就是共享资源的使用,这个时候我们怎么保证线程的安全性呢?

我们模拟一下业务场景:

        三个角色:Gute 相当于一个资源 调用一个方法pass 方法 

                        User 是使用资源的地方

                当User跑起来的时候不断的使用这个资源 

public class Gute {
    private int counter = 0; //默认第一个0
    private String name = "Nobody";
    private String address = "Nowhere";

    //临界值
    //门里面的一个方法
    public  void pass(String name, String address) {
        this.counter++;
        /**
         * 竞争锁
         */
        this.name = name;
        this.address = address;
        verify();
    }

    //通过的方法
    private void verify() {
        //判断如果首字母不相等就BROKE
        if (this.name.charAt(0) != this.address.charAt(0)) {
            System.out.println("********BROKE" + toString());
        }
    }
    //线程不安全
    public String toString() {
        return "No." + counter + ":" + name + "," + address;
    }
}
 **/
public class User extends Thread{
    private final String myname;
    private final String myAddress;
    private final Gute gate;

    public User(String myname, String myAddress, Gute gate) {
        this.myname = myname;
        this.myAddress = myAddress;
        this.gate = gate;
    }

    @Override
    public void run() {
        System.out.println(myname+"BEGEN***");
        while (true){
            this.gate.pass(myname,myAddress);
        }
    }
}
//去使用它
public static void main(String[] args) {
    Gute gute = new Gute();
    User bj =new User("Baobao","Beijing",gute);
    User sh =new User("ShangLao","Shanghai",gute);
    User sx =new User("GuanLao","Guanzhou",gute);
    bj.start();
    sh.start();
    sx.start();
}

我们模拟一下,大概的意思就只Guanzhou,对应着GuanLao,ShangLao对应着Shanghai

然后我们在通过的时候根据首字母来判断如果相等就通过,不相等就Block 


我们看到明明相等但是它却输出出来了,这是为什么?

原因分析:


也就是说它明明判断相等了,但是在toString的时候领外一个线程又把他修改了,所以输出结果出现了问题,

这就是在多线程中出现了不安全的操作,

多线程编程的三个问题,1工共享资源,也就是Gute, 有操作(pass)就会发生竞争 ,有临界值(有临界值就会有资源竞争)

我们怎么解决呢?

public synchronized void pass(String name, String address) {
    this.counter++;
    /**
     * 竞争锁
     */
    this.name = name;
    this.address = address;
    verify();
}

//通过的方法
private void verify() {
    //判断如果首字母不相等就BROKE
    if (this.name.charAt(0) != this.address.charAt(0)) {
        System.out.println("********BROKE" + toString());
    }
}
//线程不安全
public synchronized String toString() {
    return "No." + counter + ":" + name + "," + address;
}

在里面加了this锁,来保证资源共享的问题


这样他就会一直通过,也就是一直循环而不阻塞.......也就是说它资源每次访问只能有一个线程通过....

(如有错误请多指教)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值