• 201521123055 《Java程序设计》第11周学习总结


    1. 本章学习总结

    2. 书面作业

    Q1.互斥访问与同步访问

    ** 完成题集4-4(互斥访问)与4-5(同步访问) **

    1.1 除了使用synchronized修饰方法实现互斥同步访问,还有什么办法实现互斥同步访问(请出现相关代码)?

    1.2 同步代码块与同步方法有何区别?

    1.3 实现互斥访问的原理是什么?请使用对象锁概念并结合相应的代码块进行说明。当程序执行synchronized同步代码块或者同步方法时,线程的状态是怎么变化的?

    1.4 Java多线程中使用什么关键字实现线程之间的通信,进而实现线程的协同工作?为什么同步访问一般都要放到synchronized方法或者代码块中?

    1.1 在Java1.5之后,可使用java.util.concurrent.locks包提供的操作方法替代synchronized机制。

    测试代码:

    public class Main {
        static Lock lock=new ReentrantLock();
        public static void main(String[] args) {
    
            lock.lock();    //若锁处于空闲状态,则获得锁;若其他进程获得锁,则禁用当前进程
    
        }
    
        public static invoke()
        {
            lock.unlock();   //释放锁
        }
    }
    

    Lock的出现其实是同步机制的升级版,由于运行到synchronized修饰的代码块之外时,进程便自动解除block状态,因此通过Lock的使用使用户可以更自由控制锁的使用。

    1.2

    当一个进程获得了某个object的锁时,其他进程将无法访问该obj内的同步代码块,但可以访问非同步代码块。而同步方法则是限制了进程对于obj的访问。

    1.3

    为了实现互斥访问,在编程中引入了互斥锁的概念,每个对象都有一个互斥锁的标记,来保证在同一时刻只能有一个进程访问该对象。

    测试代码:

    class Account{
        private int balance;
    
        public Account(int balance) {
            this.balance = balance;
        }
    
        synchronized void deposit(int money)
        {
            this.balance+=money;
        }
    
        synchronized  void withdraw(int money)
        {
            this.balance-=money;
        }
    
        public int getBalance() {
            return balance;
        }
        
    }
    

    将实现类Account中的俩方法都上锁,就意味着当某个进程调用方法时,若其他进程也调用到该方法,那么便进入block状态,当方法锁释放时重新进入running状态。

    1.4

    使用wait()、notify()可以用来实现线程之间的通信协作。使用synchronized关键字的原因便是防止多个线程同时访问同一对象导致的冲突。

    Q2.交替执行###

    实验总结(不管有没有做出来)

    Q3.互斥访问###

    3.1 修改TestUnSynchronizedThread.java源代码使其可以同步访问。(关键代码截图,需出现学号)

    3.2 进一步使用执行器改进相应代码(关键代码截图,需出现学号)

    参考资料:Java多线程之Executor、ExecutorService、Executors、Callable、Future与FutureTask

    3.1

    //201521123055
    public static synchronized void addId() {
        id++;
    }
    
    public static synchronized void subtractId() {
        id--;
    }
    

    两个函数加上synchronized关键字即可

    3.2

        ExecutorService ec= Executors.newCachedThreadPool();
    
        for(int i=0;i<3;i++)
        {
            ec.execute(new Adder());
        }
        for(int i=0;i<3;i++)
        {
            ec.execute(new Subtracter());
        }
    
        ec.shutdown();
        ec.awaitTermination(1, TimeUnit.DAYS);
    

    关键是调用awaitTermination()来完成join效果。

    Q4.线程间的合作:生产者消费者问题###

    4.1 运行MyProducerConsumerTest.java。正常运行结果应该是仓库还剩0个货物。多运行几次,观察结果,并回答:结果正常吗?哪里不正常?为什么?

    4.2 使用synchronized, wait, notify解决该问题(关键代码截图,需出现学号)

    4.1
    有时候会出现还剩十个货物,因为存取的速度不同导致的。

    4.2

    //201521123055
    public synchronized void add(String t) {
        while (repo.size() == capacity) {
            try{
                wait();
            }catch (Exception e)
            {
               e.printStackTrace();
            }
        }
    
        repo.add(t);
        notifyAll();
    
    }
    public synchronized void remove() {
        while (repo.size() == 0) {
            try{
                wait();
            }catch (Exception e)
            {
                e.printStackTrace();
            }
        }
    
        repo.remove(0);
        notifyAll();
    }
    

    Q5.查询资料回答:什么是线程安全?(用自己的话与代码总结,写自己看的懂的作业)

    线程安全就是多线程访问时,采用了加锁机制,当一个线程访问该类的某个数据时,进行保护,其他线程不能进行访问直到该线程读取完,其他线程才可使用

    听起来是有点糊涂,打个比方就是,俩人都要去图书馆借同一本书,A先弄到手,那B去借肯定是借不到的,做到同一时间只有一个线程调用某一资源,就是线程安全。保证线程安全是很重要的,因为多线程中很容易对资源的调用产生问题,这时候做到唯一性是很必要的。

    3. PTA实验总结及码云上代码提交记录

    3.1

    3.2

  • 相关阅读:
    flask 安装及基础学习(url_for反转,静态文件引入)
    collections 模块之Counter
    collections 数据结构模块namedtuple
    docker 私有仓库之Harbor搭建与使用
    ansible Api 2.3-2.4
    Ansible Callback
    saltstack returners
    快速入门Http协议
    小兔JS教程(五) 简单易懂的JSON入门
    小兔JS教程(四)-- 彻底攻略JS数组
  • 原文地址:https://www.cnblogs.com/wkfg/p/6817870.html
Copyright © 2020-2023  润新知