334 同步方法解决数据安全问题

14 篇文章 0 订阅
本文详细探讨了如何使用synchronized关键字实现数据同步,特别是通过对象锁定(this、类名.class)确保卖票过程中的数据一致性。通过实例展示了静态与非静态同步方法的区别,并揭示了正确设置锁对象的重要性。
摘要由CSDN通过智能技术生成

334 同步方法解决数据安全问题

(myThread)

【同步方法】

同步方法,就是把synchronized关键字加到方法声明上。

格式

修饰符 synchronized 返回值类型 方法名(参数){...}

synchronized (obj)

【同步方法的锁对象】

this

【static静态同步方法的锁对象】

类名.class,例如:synchronized (SellTicket.class)

> 非静态同步方法有什么存在的意义?

【代码思路】

定义一个int变量x,初始值0,运行一次后+1

把线程代码复制2份放在在if...else循环里,第偶数次运行进if循环,第奇数次运行进else循环。

分析:这样,同一时间允许2个线程同时运行

写一个sellTicket方法,把本来在else里的线程代码放进这个方法,else里清空,只写一行:调用sellTicket方法

现在的代码是这样的

private void sellTicket() {

        synchronized (obj) {

            if (ticketN > 0) {

                ...

            }

        }

    }

在以上代码中,锁位于方法体内部,这就涉及到同步方法。

代码改成如下

private synchronized void sellTicket() {

        if (ticketN > 0) {

            ...

        }

    }

运行,出问题了,又出了重复的票,这是因为 sellTicket没有设定obj对象。

但写成private synchronized (obj) void sellTicket() 会抛出编译时异常。这就涉及到同步方法的锁对象:this。要在同步方法上设定对象,写this就好了,this指代当前的对象。

sellTicket()改成private static void sellTicket(),随之把int变量ticketN改成静态的private static int ticketN = 20;

运行,还是有重复的票。

正确做法是把对象写成类名.class(该类的字节码对象,后面在反射里讲一下),如下

if (x%2==0){

  synchronized (SellTicket.class) {...}

  else{...}

}

---------------------------------------------

package e334;

public class SellTicket implements Runnable {

    private static int ticketN = 100;

    private Object obj = new Object();

    private int x = 0;

    @Override

    public void run() {

        while (true) {

            if (x % 2 == 0) {

                synchronized (SellTicket.class) {

                    if (ticketN > 0) {

                        try {

                            Thread.sleep(100);

                        } catch (InterruptedException e) {

                            e.printStackTrace();

                        }

                        System.out.println(Thread.currentThread().getName() + "the NO. " + ticketN + " ticket is selling.");

                        ticketN--;

                    }

                }

            } else {

                sellTicket();

            }

            x++;

        }

    }

    private static synchronized(SellTicket.class) void sellTicket() {

        if (ticketN > 0) {

            try {

                Thread.sleep(100);

            } catch (InterruptedException e) {

                e.printStackTrace();

            }

            System.out.println(Thread.currentThread().getName() + "the NO. " + ticketN + " ticket is selling.");

            ticketN--;

        }

    }

}

---------------------------------------------

package e334;

public class SellTicketDemo {

  //略,与前节同

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值