Java多线程——线程同步

学习线程第二天-_-,上课的时候感觉全程高能:眼睛盯着ppt,手里敲着代码,脑子里还想着各种状态之前的切换和改变,整个人都不好了。其实好像也不是很难,只是中间的过程比较复杂,内容比较多。要多注意理解和分析。

题目:

利用多线程实现男孩存钱,女孩取钱的功能:

1,有一个银行账户Account类,此类中有余额属性private int balance;并且可以进行存、取钱功能

2,有Boy和Girl类都实现了Runnable接口,这两个类在主体方法中可以操作Account类中的存取钱方法

3,测试类进行检验男孩存钱,女孩取钱

提示:1,男孩女孩肯定是操作同一个银行账户,

2,可以使用Math.Random()方法完成存取钱的数目

3,同时,女孩取钱太多时,应该不成功,需要等到男孩再存钱,之后再取

拿到题目除了分析题之外觉得出题老师真是丧(gan)心(de)病(piao)狂(liang),好吧不能yy自己有这样的男朋友,还是要自食其力!努力敲代码!我爱学习!学校是我快乐!好吧,让我们来快乐的分析题目:

Account类:这个类的逻辑还是比较容易的。只有一个成员变量balance。对于成员方法来说稍作分析就可以得出除了对balance的get()、set()放方法外还需实现存钱、取钱功能即可。

具体实现:

public class Account {
    //题目中要求的属性,至于为什么用static下面再解释★
    private static int balance;

    //balance的get()方法,因为balance是private修饰的属性
    public int getBalance() {
        return balance;
    }
    //balance的set()方法,因为balance是private修饰的属性
    public void setBalance(int money) {
        balance = money;
        System.out.println("设置余额为:" + money);
    }
    //取钱方法传入要取的钱的数目并且判断是否可以取钱
    public boolean getMoney(int money) {
        if (balance == 0) {
            System.out.println("一分钱也没有啦,不能取钱");
            return false;
        } else if (balance < money) {
            System.out.println("余额不足,不能取钱");
            return false;
        } else {
            balance = balance - money;
            System.out.println("取钱后余额为:" + balance);
            return true;
        }
    }
    //存钱方法,不需要判断,直接存入即可
    public void saveMoney(int money) {
        balance = balance + money;
        System.out.println("存钱后余额为:" + balance);
    }
}

接下来写题目中要求的Girl和Boy类,稍作分析,Boy和Girl两个类都要实现Runnable接口,而且都要操作Account类,这两个类的写法应该是相似的我们先写好Boy类,之后的Girl类自然也就能写出来了。

//创建Boy类,实现Runnable接口
//题目要求而已,这里我试过直接继承Thread类也是可以的
class Boy implements Runnable {
    //声明一个Account类的变量用来操作这个类,并且将其初始化
    public Account account = new Account();
    //重写run方法
    @Override
    public void run() {
        //获取当前线程名字
        String name = Thread.currentThread().getName();
        //使用synchronized将线程中有竞争的资源锁起来。
        //拿到锁对象的线程才可以执行这段代码
        //因为两个线程都要操作Account类所以使用Account.class作为锁对象
        synchronized (Account.class) {
            //为了让线程一直运行观察结果将其放入死循环中
            while (true) {
                System.out.println("获取余额" + account.getBalance());
                //随机存入一个金额
                account.saveMoney((int) (Math.random() * 10));
                System.out.println(name+"存钱中...");
                //存钱过程中不能被打断,存完钱就可以释放锁对象
                Account.class.notify();
                try {
                //释放锁对象后进入阻塞状态等待其他线程将其唤醒
                //其实就是等待女孩没钱了叫他去存钱~
                    Account.class.wait();
                } catch (InterruptedException e1) {
                    // TODO Auto-generated catch block
                    e1.printStackTrace();
                }
                try {
                    Thread.sleep(100);

                } catch (InterruptedException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }
        }
    }
}

到这里Boy类就写完了,同样的我们继续写Girl类,大体逻辑差不多,多了一步就是要判断账户里的余额是否够用,直接上代码:

class Girl implements Runnable {
    public Account account = new Account();

    @Override
    public void run() {
        String name = Thread.currentThread().getName();
        //同样的,竞争资源使用synchronized把代码块锁起来
        synchronized (Account.class) {
            while (true) {
                System.out.println("获取余额" + account.getBalance());
                //我这里直接把getMoney()方法写成boolean了
                //所以一个if就可以直接判断,如果为true,可以直接取钱
                if (account.getMoney((int) (Math.random() * 10))) {
                    System.out.println(name + "取钱中................");
                    //如果为false,就释放资源,唤醒男孩去存钱,并且自己进入阻塞状态
                } else {
                    try {
                        System.out.println("-------------");
                        //就释放资源,唤醒男孩
                        Account.class.notify();
                        //进入阻塞状态
                        Account.class.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }
    }

}

接下来是测试类,很简单,调用写好的各种方法即可

public class Test {
    public static void main(String[] args) {
        Girl g = new Girl();
        Thread girl = new Thread(g);
        girl.setName("小红");
        girl.start();

        Boy b = new Boy();
        Thread boy = new Thread(b);
        boy.setName("小明");
        boy.start();
    }
}

运行结果:
这里写图片描述


这里写图片描述

结果就是小红一直取钱直到账户余额为0,让小明去存钱,存完后接着取如此反复,小红简直就是人生赢家呀!!!

★这里解释为什么balance要使用static修饰,因为我们是两个类中的去操作Account,new Account();会产生两个不同的实例对象,各自有各自的内存空间,如果不用static修饰balance就会产生两个balance,使用static后balance会放在常量池中,虽然是两个不同的Account对象,但是使用的balance是共有的~~~

好困好困好困 去睡觉-_-
我爱学习,学习使我快乐

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值