java-study Thread线程、synchronized同步锁、await()/notify()

 demo1:

package com.TestThread;

public class Test {
    public static void main(String[] args) {
        Account a0 = new Account();
        Account a1 = new Account();
        User u0 = new User(a0, 2000);
        User u1 = new User(a1, 2000);

        Thread t0 = new Thread(u0);
        Thread t1 = new Thread(u1);
        t0.start();
        t1.start();
    }
}

class Account {
    private static int balance = 3000;

    /**
     *  synchronized加在成员方法上只能锁调用当前方法的对象,例如man方法中如果u0、u1接口构造中传入的不是同一个Account对象的话,无法使用实现同步锁。
     *  要想所有实例成员共用一个同步锁可以将该方法修饰为static
     */
    public synchronized void TakeBalance(int m) {
        String tname = Thread.currentThread().getName();
        System.out.println(tname+"当前余额"+balance);
        if(m < balance) {
            System.out.println(tname+"取款2000");
            System.out.println(tname+"取款中。。。");
            balance -= m;
            System.out.println(tname+"取款成功");
            System.out.println(tname+"余额剩余"+balance);
        }else{
            System.out.println(tname+"取款失败,余额不足");
            System.out.println(tname+"余额剩余"+balance);
        }
    }

    /**
     * 代码块同步锁,synchronized(instance),加在代码块前面,传入的参数为一个实例对象(或者this),表示对该对象使用同一个锁。
     *
     */
    public void TakeBalance2(int m) {
       synchronized (this) {
           String tname = Thread.currentThread().getName();
           System.out.println(tname+"当前余额"+balance);
           if(m < balance) {
               System.out.println(tname+"取款2000");
               System.out.println(tname+"取款中。。。");
               balance -= m;
               System.out.println(tname+"取款成功");
               System.out.println(tname+"余额剩余"+balance);
           }else{
               System.out.println(tname+"取款失败,余额不足");
               System.out.println(tname+"余额剩余"+balance);
           }
       }
    }
}

class User implements Runnable {
    public User(Account a, int m) {
        money = m;
        account = a;
    }
    private int money;
    private Account account;

    @Override
    public void run() {
//        account.TakeBalance(money);
        account.TakeBalance2(money);
    }
}

demo2:

package com.TestThread;

/**
 * 场景说明:生产者->供应商店,消费者->从商店消费
 * 需求说明:商店商品数量最多存放10个,数量少于4个时生产者供应商品;消费者在商品数量大于0时都可以消费,否则等待生产者供应;
 *
 * 疑惑点:在两个线程中,当此处while在代码块锁内的时候控制台输出规律是一致的,都是 生产满5个->消费至0->生产满5个->消费至0...,而目前这样打印出来是会出现随机的,即生产完一个就有可能会马上被消费。
 */
public class Test2 {
    public static void main(String[] args) {
        Shop shop = new Shop();
        new Thread(new Runnable() {//内部类直接实现接口
            @Override
            public void run() {
                while(true) {
                    synchronized (shop) {
                        if(shop.productionNum < 5) {
                            System.out.println("开始生产,当前剩余商品:"+shop.productionNum);
                            shop.productionNum++;
                            System.out.println("生产完成,当前剩余商品:"+shop.productionNum);

                            shop.notify();//生产完毕后,唤醒当前正在等候的线程(消费者)
                        }else{
                            System.out.println("商品数量充足 暂不生产,当前剩余商品:"+shop.productionNum);
                            try {
                                shop.wait();//当前线程挂起并放弃cpu、同步资源,使别的线程可访问并修改共享资源,而当前线程等候。
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }

            }
        },"生产者").start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                while(true) {
                    synchronized (shop) {
                        if(shop.productionNum > 0) {
                            System.out.println("开始消费,当前剩余商品:"+shop.productionNum);
                            shop.productionNum--;
                            System.out.println("消费完成,当前剩余商品:"+shop.productionNum);

                            shop.notify();//消费完毕后,唤醒当前正在等候的线程(生产者)
                        }else{
                            System.out.println("商品数量不足 不能消费,当前剩余商品:"+shop.productionNum);
                            try {
                                shop.wait();//当前线程挂起并放弃cpu、同步资源,使别的线程可访问并修改共享资源,而当前线程等候。
                            } catch (InterruptedException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
            }
        },"消费者").start();
    }
}

class Shop {
    public static int productionNum = 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 您好!如果我理解正确的话,您的问题是关于JMU Java 07多线同步访问的。以下是我的回答: 多线程编程是指在一个程序中同时执行多个线程的编程技术。多线程编程在提高程序性能、增强程序交互性等方面有很大的优势,但也会引入一些问题,比如线程安全问题。 当多个线程同时访问共享资源时,可能会出现线程不安全的问题,比如数据竞争、死等。为了避免这些问题,我们需要使用同步机制来确保线程安全。 Java中提供了多种同步机制,比如synchronized关键字、ReentrantLock类、Semaphore类等。这些同步机制都可以用来保证多线程访问共享资源的安全性。 其中,synchronized是最常用的同步机制之一。在Java中,synchronized关键字可以用来修饰方法或代码块,以实现同步访问共享资源。synchronized可以保证在同一时间只有一个线程可以访问共享资源,从而避免了数据竞争等线程安全问题。 除了synchronized之外,Java还提供了一些其他的同步机制,比如ReentrantLock类和Semaphore类。ReentrantLock类提供了比synchronized更灵活的机制,可以实现可重入、公平等功能。Semaphore类可以控制同时访问共享资源的线程数量,从而避免资源被过度占用的问题。 总的来说,多线程编程是一项复杂而重要的技术,需要仔细研究和实践。在实际开发中,我们需要根据具体的需求选择合适的同步机制来确保多线程访问共享资源的安全性。 ### 回答2: 多线程编程是在当前计算机领域中最为常见的技术之一,它可以利用计算机中的多核处理器来使程序运行更加高效。但是,多线程编程中可能会出现的最大问题就是线程安全,因为线程之间可能会访问相同的资源,从而导致竞态条件。 在Java中,可以通过使用synchronized关键字来实现同步访问,从而避免线程安全问题。synchronized关键字可以用于两种不同的情形:同步方法和同步块。在同步方法中,方法是同步的,即每个线程在执行该方法时都需要获取该对象的,如果该已经被其他线程获取,则需要等待直到此被释放。在同步块中,需要手动指定,即每个线程在执行同步块时需要获取该指定,其他线程如果需要访问该代码块中的共享资源也需要获取该指定,这样就保证了该代码块中的所有共享资源的同步访问。 除了synchronized关键字外,Java还提供了其他一些同步机制来实现线程安全,如ReentrantLock类和CountDownLatch类等。ReentrantLock类可以实现更为灵活的同步访问控制,但需要手动释放;而CountDownLatch类则用于同步一个或多个线程,使这些线程在某个条件满足之前一直处于等待状态。 在进行多线程编程时,应该尽量避免对同步访问造成瓶颈,应该通过减小同步代码块的范围等方式来提高程序的效率。此外,多线程编程时还应该进行线程安全性的测试,以确保程序能够正确地运行。 ### 回答3: 在Java中,多线程是一种非常常见的编程方式。由于多线程的特点,对共享资源的访问会出现竞争的情况,这种竞争可能会导致数据不一致或程序异常等问题。因此,在多线程编程中,我们需要采取一些措施来保证共享资源的访问能够正确、有序地进行,这就是同步机制。 同步机制包括两种方式:和信号量。是最基本的同步机制。有两种类型:互斥(Mutex)和读写(ReadWriteLock)。互斥用于保护共享资源,保证同一时间只有一个线程可以访问它,其他线程需要等待释放后才能继续访问。读写用于读写分离场景,提高了多线程访问共享资源的并发性。读写支持多个线程同时读取共享资源,但只允许一个线程写入共享资源。 信号量是一种更加高级的同步机制。信号量可以用来控制并发线程数和限制访问共享资源的最大数量。在Java中,Semaphore类提供了信号量的实现。Semaphore可以控制的线程数量可以是任意的,线程可以一起执行,也可以分批执行。 除了和信号量,Java还提供了一些其他同步机制,比如阻塞队列、Condition等。阻塞队列是一种特殊的队列,它支持线程在插入或者删除元素时阻塞等待。Condition是一种的增强,它可以让线程在某个特定条件下等待或者唤醒。 在多线程编程中,使用同步机制需要注意以下几点。首先,同步机制要尽可能的保证资源访问的公平性,避免因为某些线程执行时间过长导致其他线程等待时间过长。其次,同步机制要尽可能的避免死的发生,尤其要注意线程之间的依赖关系。最后,同步机制的实现要尽可能地简单,避免过于复杂的代码实现带来的维护成本。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值