目录

一、多线程概述

二、Java中线程的创建

第一种实现方式:继承Thread类

第二种实现方式:实现Runnable接口

三、线程的生命周期及状态转换

四、线程的调度

五、线程的优先级

六、线程休眠

案例:龟兔赛跑

七、线程让步

八、线程插队

案例:Svip优先办理服务

九、线程安全问题

为什么会产生线程安全问题

怎么解决线程安全问题

十、同步代码块

十一、同步方法

十二、死锁问题

综合案例

一、模拟银行存取钱

二、工人搬砖

三、小朋友就餐


一、多线程概述

什么是进程:每个运行的程序都是一个进程,在一个进程中还可以有多个执行单元,这些执行单元可以看作执行程序的一条条线索,被称为线程。操作系统中的每个进程中都至少存在一个线程。

java file 多线程文件锁 java 多线程写文件_ide

什么是多线程:多线程指的是一个进程在执行过程中可以产生多个单线程,这些单线程程序在运行时是相互独立的,但他们可以并发执行

:进程是没有一起进行的,但是由于CPU在多个进程间高效的切换,所以看起来像是同时进行的。一条线程就可以看成是一个执行路径。 

二、Java中线程的创建

Java中两种多线程的实现方式:

1.继承java.lang包下的Thread类,覆写Thread类的run方法,在run()方法中实现运行进程上的代码

2.实现java.lang.Runnable接口,同样是在run()方法中实现运行在线程上的代码

第一种实现方式:继承Thread类

单线程

目前只有一个线程,在这个程序中就只有一条执行路径,只有MyThread中的run方法执行完毕后,才会执行main方法中的while,但是下面是while(true)是一个死循环,所以就没有办法执行上面的 

java file 多线程文件锁 java 多线程写文件_System_02

多线程


* 继承Thread类的实现步骤: * 1、需要定义一个类,然后这个类区继承Thread类;Thread类就是Java中的线程类 * 2、重写Thread类中的run方法, * 3、创建我们定义的类的对象 * 4、启动线程


public class Example02 {
    public static void main(String[] args) {
        //创建MyThread类的对象  -->这是一个线程类创建一个对象就相当于开启了一个线程
        MyThread myThread = new MyThread();
        //开启线程
        myThread.start();

        //编写while循环
        while (true) {
            System.out.println("Main方法执行了");
        }
    }
}


//定义一个线程类
class MyThread extends Thread {
    @Override
    public void run() { //封装的就是要被线程执行的代码
        while (true) {
            System.out.println("MyThread中的run方法执行了");
        }
    }
}

这个程序就有两条执行路径,两条交替执行,此时运行的时候两个就都会运行

java file 多线程文件锁 java 多线程写文件_优先级_03

关于单线程和多线程可以这样理解

java file 多线程文件锁 java 多线程写文件_优先级_04

通过继承Thread类可以实现多线程,但是这种方式有一定的局限性。因为Java只支持单继承,一个类一旦继承了某个父类就无法再继承Therad类了

第二种实现方式:实现Runnable接口


* 第二种实现方式的开发步骤 * 1、定义一个类,让这个类实现Runnable接口。Runnable接口就是一个任务接口,在该接口中定义了一个方法就是run,这个方法就是用来封装要被线程所执行的代码。 * 2、重写run方法 * 3、创建Thread对象,在创建这个对象之前需要先创建任务类对象,然后把任务另外对象作为Thread的构造方法参数传递过去 * 4、启动线程


public class Example03 {

    public static void main(String[] args) {
        //创建MyThread对象
        MyThread mythread = new MyThread();
        //创建Thread对象
        Thread thread = new Thread(mythread);
        //启动线程
        thread.start();

        while (true) {
            System.out.println("Main方法执行了");
        }
    }
}


//定义一个类
class MyThread implements Runnable {

    //重写run方法
    @Override
    public void run() {
        while (true) {
            System.out.println("MyThread中的run方法执行了");
        }
    }
}

这里同样是两个方法都执行了

java file 多线程文件锁 java 多线程写文件_java file 多线程文件锁_05

案例假设售票厅有4个窗口可发售某日某次列车的100整车票,这时,100张车票可以看作共享资源,四个售票窗口需要创建4个线程。为了更直观的显示窗口的售票情况,可以通过Thread的currentThread()方法得到当前的线程的实例对象,然后调用getName()方法获取到线程的名称。

第一种方法:

public class Example04 {

    public static void main(String[] args) {

        //创建4个线程对象
        new TicketWindow("窗口一").start();
        new TicketWindow("窗口二").start();
        new TicketWindow("窗口三").start();
        new TicketWindow("窗口四").start();

    }
}


//创建线程类
class TicketWindow extends Thread {
    //定义一个成员变量,这个成员变量记录的就是要出售的票的总数
    private int tickets = 100;

    public TicketWindow(String name) {
        super(name);
    }

    @Override
    public void run() {
        //为了模拟一直有票
        while (true) {
            if (tickets > 0) {
                //获取当前正在执行的线程对象
                Thread th = Thread.currentThread();
                String th_name = th.getName();

                System.out.println(th_name + "正在发售" + tickets-- + "张票");
            }
        }
    }
}

这个时候就会发现虽然程序可以运行但是票数是不对的,这里的4个窗口每个窗口都有100张票 

java file 多线程文件锁 java 多线程写文件_java_06

 需要实现4个窗口共同卖这100张票

java file 多线程文件锁 java 多线程写文件_System_07

加上static就可以实现

java file 多线程文件锁 java 多线程写文件_java_08

第二种方法:(这里不需要static)

public class example05 {
    public static void main(String[] args) {
        //创建TicketWindow对象
        TicketWindow tw = new TicketWindow();

        //创建线程对象
        new Thread(tw, "窗口一").start();
        new Thread(tw, "窗口二").start();
        new Thread(tw, "窗口三").start();
        new Thread(tw, "窗口四").start();
    }
}

class TicketWindow implements Runnable {

    //定义一个成员变量,这个成员变量记录的就是要出售的票的总数
    private int tickets = 100;

    @Override
    public void run() {
        //为了模拟一直有票
        while (true) {
            if (tickets > 0) {
                //获取当前正在执行的线程对象
                Thread th = Thread.currentThread();
                String th_name = th.getName();

                System.out.println(th_name + "正在发售" + tickets-- + "张票");
            }
        }
    }
}

使用实现Runnable接口相对于继承Thread的优势:

1.适合多个相同程序代码的线程区处理同一个资源的情况,吧线程和任务代码、有效的分离,很好的体现了面向对象的设计思想。

2.可以避免由于Java的单继承带来的局限性。在开发中经常碰到这样一种情况,就是使用一个已经继承了某一个类的子类创建线程,由于一个类不能同时有两个父类。因此不能使用继承Thread类的方式,只能采用实现Runnable接口的方式。

三、线程的生命周期及状态转换

什么叫线程的生命周期:线程从创建到销毁的整个过程。

当run()方法中代码正常执行完毕或者线程抛出一个未捕获的异常或者错误时,线程的生命周期就会结束。线程的整个生命周期可以分为五个阶段:新建状态(New)、就绪状态(Runable)、运行状态(Running)、阻塞状态(Blocked)、死亡状态(Terminated)。

java file 多线程文件锁 java 多线程写文件_java file 多线程文件锁_09

新建状态:创建一个线程对象后,该线程就处于新建状态,此时不能运行

就绪状态:当线程调用了start()方法后,该线程就进入了就绪状态。此时只是具备了运行的条件,能否获得CPU的使用权并开始运行还要看系统的调度。

运行状态:如果处于就绪状态的线程获得了CPU的使用权,并开始执行run()方法中的线程执行体,则该线程处于运行状态。【一个线程启动后可能不会一直处于运行状态,当运行中的线程用完系统分配的时间后,系统就会剥夺该线程占用的CPU资源,让其他线程获得执行的机会。只有处于就绪状态的线程才可能转换到运行状态】

阻塞状态:不具有CPU的执行资格和执行权。一个正在执行的线程在某些特殊情况下,如被人为挂起或执行人为的输入\输出操作时,会让CPU的使用权暂时中止自己的执行,进入阻塞状态。【线程进入阻塞状态后,就不能进入排队队列。只有当一起阻塞的原因被消除后,线程才可以进入就绪状态。】

死亡状态:如果程序调用stop()方法或者run()方法正常执行完毕,或者线程抛出一个未捕获的异常、错误,线程就进入死亡状态。【一旦进入死亡状态线程将不在拥有运行的资格,也不能再转换到其他状态。】

四、线程的调度

线程的调度:Java虚拟机按照特定的机制为程序中的每个线程分配CPU的使用权 

线程的调度模型:分时调度模型抢占式调度模型

分时调度模型:让所有线程轮流获得CPU的使用权,并且平均分配每个线程占用CPU的时间片。

抢占式调度模型:让可运行池中优先级高的线程优先占用CPU,而对于优先级相同的线程,随机选一个线程使其占用CPU,当它失去了CPU的使用权之后,再随机选用其他线程获取CPU使用权。

【注:Java虚拟机默认采用抢占式调度模型】

五、线程的优先级

线程的优先级用1~10之间的整数表示,数字越大优先级越高。

除了直接用数字表示线程的优先级,还可以使用Thread类中提供的三个静态常量表示线程的优先级


static int MAX_PRIORITY -->表示线程的最高优先级,值为10 static int MIN_PRIORITY -->表示线程的最低优先级,值为1 static int NORM_PRIORITY -->表示线程的普通优先级,值为5【线程的默认优先级】


线程的优先级可以通过Thread中的setPriority(int newPriority)方法进行设置,setPriority()方法中的参数newPriority接收的是1~10之间的整数或者Thread类的三个静态常量。

public class Example06 {
    public static void main(String[] args) {
        //创建两个线程对象
        Thread minPriority = new Thread(new MinPriority(), "优先级较低的线程");
        Thread maxPriority = new Thread(new MaxPriority(), "优先级较高的线程");

        //设置优先级
        minPriority.setPriority(Thread.MIN_PRIORITY);  //这个设置为最低优先级
        maxPriority.setPriority(Thread.MAX_PRIORITY);  //这个设置为最高优先级

        //启动两个线程
        maxPriority.start();
        minPriority.start();
    }
}


//创建一个任务类
class MaxPriority implements Runnable {

    @Override
    public void run() {
        for (int x = 0; x < 10; x++) {
            System.out.println(Thread.currentThread().getName() + "正在输出" + x);
        }
    }
}

//创建一个任务类
class MinPriority implements Runnable {

    @Override
    public void run() {
        for (int x = 0; x < 10; x++) {
            System.out.println(Thread.currentThread().getName() + "正在输出" + x);
        }
    }
}

这时输出的效果为

java file 多线程文件锁 java 多线程写文件_java_10

这里我们设置了优先级但是输出的结果并没有按照我们理想中的状态进行

【注:线程的优先级仅仅代表的是线程抢占到CPU执行权的概率增大了。但是CPU不一定CPU会执行】

六、线程休眠

想要人为的控制线程,使正在运行的线程暂停,将CPU让给别的线程,这里可以使用静态方法sleep(),该方法可以让当前正在执行的线程暂停一段时间,进入休眠等待状态。

public class Example07 {

    public static void main(String[] args) throws InterruptedException {
        //创建一个线程对象并启动
        new Thread(new SleepThread()).start();

        //另一个for循环
        for (int i = 1; i <= 10; i++) {

            if (i == 5) {
                Thread.sleep(2000);  // 当i等于5时让线程休眠2000毫秒
            }
            System.out.println("主线程正在输出" + i);
            Thread.sleep(500);
        }
    }
}

//创建一个任务类
class SleepThread implements Runnable {

    @Override
    public void run() {
        for (int i = 1; i <= 10; i++) {

            if (i == 3) {  //当i等于3时让线程休眠2000毫秒
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            System.out.println("SleepThread线程正在输出:" + i);
            //每次输出了都想让线程进行休眠
            try {
                Thread.sleep(500);  //让当前线程每次输出之后都休眠500毫秒
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

java file 多线程文件锁 java 多线程写文件_优先级_11

java file 多线程文件锁 java 多线程写文件_System_12

这里每个人运行的不一样,而且设置了时间线程也不是一定严格按照你所设置的时间进行

  • 比如Thread.sleep(1000),1000ms后是否立即执行?

不一定,在未来的1000毫秒内,线程不想再参与到CPU竞争。那么1000毫秒过去之后,这时候也许另外一个线程正在使用CPU,那么这时候操作系统是不会重新分配CPU的,直到那个线程挂起或结束;况且,即使这个时候恰巧轮到操作系统进行CPU 分配,那么当前线程也不一定就是总优先级最高的那个,CPU还是可能被其他线程抢占去

案例:龟兔赛跑

众所周知的“龟兔赛跑”故事,兔子因为太过自信,比赛中途休息而导致乌龟赢得了比赛。本案例要求编写一个程序模拟龟兔赛跑,乌龟的速度为1米/100毫秒,兔子的速度为2米/100毫秒,等兔子跑到第600米时选择休息60000毫秒结果乌龟赢得了比赛。

【根据题意思考:乌龟速度为1米/100毫秒,兔子的速度为2米/100毫秒,也就是1米/50毫秒。】

注:这里设置的就100毫秒和50毫秒,如果设置成100和200的话逻辑上是没有问题的,但是兔子休息的时间也就应该相应的增长。按照原来的休息时间算的话乌龟是依旧跑不过兔子的。

public class Rabbit {
    public static void main(String[] args) throws InterruptedException {
        //创建线程对象并启动
        new Thread(new Turtles()).start();

        //创建另一个循环
        for (int i = 1; i <= 1000; i++) {
            System.out.println( "兔子正在跑第" + i + "米");
            if (i == 600) {
                Thread.sleep(60000);
            } else if (i == 1000) {
                System.out.println("兔子跑完了!");
            }
            Thread.sleep(50);
        }
    }
}


class Turtles implements Runnable {
    @Override
    public void run() {
        for (int i = 1; i <= 1000; i++) {
            System.out.println("乌龟正在跑第" + i + "米");
            try {
                Thread.sleep(100);  // 每1米/100毫秒
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            if (i == 1000) {
                System.out.println("乌龟跑完了!");
            }
        }
    }
}

运行结果:可以看到在开始是两个线程交替进行,但是在兔子跑到第600米的时候就开始休眠了,这时乌龟奋起直追,到兔子的60000毫秒休眠完成之后再开始跑,这时就追不上乌龟了,最终乌龟先跑完乌龟获胜

【这里由于录屏有点大无法完成上传我就分别截几张图】

1.开始的两线程交替

java file 多线程文件锁 java 多线程写文件_ide_13

2.第600米兔子开始休眠

java file 多线程文件锁 java 多线程写文件_优先级_14

 3.兔子完成60000毫秒的休眠

java file 多线程文件锁 java 多线程写文件_优先级_15

 4.乌龟跑完1000米

java file 多线程文件锁 java 多线程写文件_System_16

七、线程让步

线程让步:正在执行的线程,在某些情况下将CPU资源让给其他线程进行

public class Example08 {

    public static void main(String[] args) {
        //创建两个线程对象
        Thread t1 = new YieldThread("线程A");
        Thread t2 = new YieldThread("线程B");

        //启动线程
        t1.start();
        t2.start();
    }
}

//定义一个类,让其继承Thread类
class YieldThread extends Thread {
    //提供一个构造方法,让别人在创建这个线程对象的时候,传递一个名称
    public YieldThread(String name) {
        super(name);
    }

    @Override
    public void run() {
        for (int i = 0; i < 6; i++) {
            System.out.println(Thread.currentThread().getName() + "---" + i);

            if (i == 3) {
                System.out.println("线程让步");
                Thread.yield();  // 调用yield()静态方法之后就会出让CPU的执行权
            }
        }
    }
}

这是理想状态 

java file 多线程文件锁 java 多线程写文件_ide_17

【注意:就算调用了yield方法让出了CPU的使用权之后马上它也可以那会CPU的使用权。这种情况并不是错误】 

java file 多线程文件锁 java 多线程写文件_优先级_18

八、线程插队

在Thread类种提供join()方法实现插队的功能。

当在某个线程中调用其他线程的join()方法时,当前调用的线程就会被阻塞,直到被join()方法加入的线程执行完成后它才会继续执行。

public class Example09 {

    public static void main(String[] args) throws InterruptedException {
        //创建一个线程对象
        Thread t = new Thread(new JoinThread(), "线程一");
        //启动线程
        t.start();


        for (int i = 1; i < 6; i++) {
            System.out.println(Thread.currentThread().getName() + "输入:" + i);

            if (i==2){
                t.join();  //调用join()方法,将t线程插队加入到main线程中
            }

            //线程每执行一次就休眠1000毫秒
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

//创建一个任务类
class JoinThread implements Runnable {
    @Override
    public void run() {
        for (int i = 1; i < 6; i++) {
            System.out.println(Thread.currentThread().getName() + "输入:" + i);

            //线程每执行一次就休眠1000毫秒
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

java file 多线程文件锁 java 多线程写文件_System_19

java file 多线程文件锁 java 多线程写文件_System_20

案例:Svip优先办理服务

在日常工作生活中,无论哪个行业都会设置一些Svip用户,Svip用户具有超级优先权,在办理业务时,Svip用户具有最大的优先级。本案例要求编写一个模拟Svip优先办理业务的程序,在正常的业务办理中,插入一个Svip用户,优先为Svip用户办理业务。本案例在实现时,可以通过多线程实现。

【注“我这里设置的是10个Svip用户和20个普通用户,在第5个普通用户办理完成后,Svip用户进行了插队】

public class Transact {

    public static void main(String[] args) throws InterruptedException {

        //创建线程对象并启动
        Thread t = new Thread(new Svip());
        //启动线程
        t.start();

        for (int i = 1; i <= 20; i++) {
            System.out.println("普通用户正在办理" + i);
            if (i == 5) {  //在第15个普通用户办理的时候这是Svip用户进行了插队
                t.join();
            }
            Thread.sleep(100);
        }
    }
}

//创建Svip的类
class Svip implements Runnable {
    @Override
    public void run() {
        for (int i = 1; i <= 10; i++) {
            System.out.println("Svip正在办理" + i);
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

java file 多线程文件锁 java 多线程写文件_优先级_21

九、线程安全问题

案例:

前面讲解的售票案例,极有可能碰到“意外”情况,如一张票被打印多次,或者打印出的票号为0甚至负数。这些“意外”都是由多线程操作共享资源ticket所导致的线程安全问题。接下来对售票案例进行修改,模拟四个窗口出售10张票,并在售票的代码中使用sleep()方法,令每次售票时线程休眠10毫秒。

public class Example10 {
    public static void main(String[] args) {
        //创建一个任务类的对象
        SaleThread saleThread = new SaleThread();

        //创建4个线程对象并指定线程名称并启动
        new Thread(saleThread, "窗口一").start();
        new Thread(saleThread, "窗口二").start();
        new Thread(saleThread, "窗口三").start();
        new Thread(saleThread, "窗口四").start();
    }
}

//创建一个任务类
class SaleThread implements Runnable {
    //定义一个成员变量,来记录票的总数量【这里设置少一点能见到效果即可】
    private int tickets = 10;

    @Override
    public void run() {
        while (tickets > 0) {
            try {
                Thread.sleep(100);  // 通过sleep来模拟网络的延迟
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "---卖出的票" + tickets--);
        }
    }
}

这里在访问共享数据时,存在数据错乱的问题导致出现了0号票和-1号票 

java file 多线程文件锁 java 多线程写文件_java_22

为什么会产生线程安全问题

在运行到tickets=1时首先窗口1进行抢占这张票,抢着之后进入while循环开始休眠,这时剩下的3个窗口开始抢占CPU的执行权,比如这时窗口2抢占到了CPU的执行权。上面窗口1抢占到之后就进行了休眠,这里的票并没有发生--,所以在窗口2抢占到CPU的执行权的时候票的数量仍旧是1,然后在这时窗口2也进入休眠,所以剩下的两个窗口又开始抢占CPU的执行权,以此类推。当最后一个窗口抢占到CPU的执行权后,线程开始进行下一步,这是窗口1售出第1张票,然后后面的窗口2在执行就是第0张票,然后就是第-1张,-2张...

本质上是由多个线程同时处理共享资源导致的

怎么解决线程安全问题

保证在任何时刻只有一个线程访问共享资源就可以解决掉线程安全问题。

在这个问题中保证只有一个线程执行下面这段代码就可以解决问题

java file 多线程文件锁 java 多线程写文件_System_23

这里用到同步代码块的相关知识来解决

十、同步代码块

同步代码块:将处理共享资源的代码放在一个使用synchronized关键字修饰的代码块中

java file 多线程文件锁 java 多线程写文件_ide_24

注: 这里的lock锁对象要被这几个线程所共享,是同一把锁,不能每个线程分别有一把锁

public class Example11 {

    public static void main(String[] args) {
        //创建一个任务类对象
        Ticket1 ticket = new Ticket1();

        //创建四个线程对象指定线程名称并启动
        new Thread(ticket, "窗口一").start();
        new Thread(ticket, "窗口二").start();
        new Thread(ticket, "窗口三").start();
        new Thread(ticket, "窗口四").start();
    }
}


//创建一个任务类
class Ticket1 implements Runnable {
    //定义一个成员变量,来记录票的总数量【这里设置少一点能见到效果即可】
    private int tickets = 10;

    //同步代码块的锁需要一个对象,那么就在这里定义一个对象,作为同步代码块中的锁。
    private Object lock = new Object();

    @Override
    public void run() {
        //模拟一直有票的状态
        while (true) {

            synchronized (lock) {
                try {
                    Thread.sleep(10);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

                if (tickets > 0) {
                    System.out.println(Thread.currentThread().getName() + "---卖出的票" + tickets--);
                } else {
                    break; //没票结束循环
                }
            }
        }
    }
}

 这时不管重新运行多少次都不会出现重复票和0号票以及负数票的情况

java file 多线程文件锁 java 多线程写文件_优先级_25

注:这里我们看到的只有窗口一在运行但实际上这应该时CPU执行的问题,当把票数增加到足够大的时候你就可以看到其他窗口执行

原因分析:当tickets=1时四个线程又开始抢占CPU的执行权,这时窗口一先抢到了CPU的执行权,窗口一就进入到同步代码块中,但是想要进入同步代码块首先就要获取同步代码块中定义的锁,这里的锁还并未被其他线程所获取,这里的窗口一就拿到了这把锁,窗口一就将进入到了同步代码块中,这时窗口一开始休眠,但是当窗口一进行休眠时并不会释放锁,所以窗口一仍旧持有这把锁,这时后面的三个线程抢占CPU的执行权,假如当窗口二抢到执行权之后,窗口二也需要进入到同步代码块中,但是不要忘记进入同步代码块中的前提是能够获取到那把锁,这个时候这把锁还在窗口一的手中,所以窗口二获取不到锁,得不到锁也就无法进入同步代码块,窗口二就处于等待的状态。同样对于剩下的两个线程都是拿不到锁的,都处于等待状态。这时窗口一的休眠时间到了,就继续向下执行这时tickets是1,再进行--,这时tickets就为0.这时窗口一就将同步代码块中的相关代码执行完了 ,窗口一就可以跳出同步代码块了,这时这把锁就会发生释放。假如这时窗口二抢占到CPU的执行权了,这时窗口二就进入同步代码块中,这时这把锁没有被其他线程所持有这时窗口二就拿到了这把锁,就进入到同步代码块中开始执行,开始执行时在这里会休眠10毫秒,这时CPU的执行权就会被释放出去其他线程就可以抢占CPU的执行权但是这时这把锁还在窗口二的手中,所以无论哪个线程抢到CPU的执行权都无法进入同步代码块。当窗口二休眠结束时,开始继续执行,由于这时tickets已经为0所以不满足代码块继续执行的条件,就会走到else中进行break中进行返回,后面的其他线程就都是类似的了。

【总结一句话就是同步代码块中的代码再任一时刻都只能由一个线程进行执行,从而保证了线程的安全问题】

十一、同步方法

当把共享资源的操作放在synchronized定义的区域内时,便为这些操作加了同步锁。在方法前面同样可以使synchronized关键字来修饰,被修饰的方法为同步方法,它能实现和同步代码块同样的功能。


语法格式:synchronized返回值类型方法名([参数1 , … ]){}


同样被synchronized修饰的方法在某一时刻只允许一个线程访问,访问该方法的其他线程都会发生阻塞,直到当前线程访问完毕后,其他线程才有机会执行该方法。

public class Example12 {

    public static void main(String[] args) {
        //创建一个任务类对象
        Ticket1 ticket = new Ticket1();

        //创建四个线程对象指定线程名称并启动
        new Thread(ticket, "窗口一").start();
        new Thread(ticket, "窗口二").start();
        new Thread(ticket, "窗口三").start();
        new Thread(ticket, "窗口四").start();
    }
}

//创建一个任务类
class Ticket1 implements Runnable {
    //定义一个成员变量,记录票的总数量
    private int tickets = 10;

    //同步代码块的锁需要一个对象,那么就在这里定义一个对象,作为同步代码块中的锁。
    private Object lock = new Object();


    @Override
    public void run() {
        while (true) {
            saleTicket();
            if (tickets <= 0) {
                break;
            }
        }
    }

    //创建一个方法
    private synchronized void saleTicket() {  //同步方法

        if (tickets > 0) {
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "---卖出的票" + tickets--);
        }
    }
}

如果需要定义一个方法而这个方法内部所有的代码都需要去保证安全性,这个时候的话,就需要把synchronized关键字直接定义在方法上,让这个方法成为一个同步方法,这时就可以提高开发效率

java file 多线程文件锁 java 多线程写文件_System_26

同步代码块能够保证线程安全问题的原因是有一把琐,只有获取到锁才能进入同步代码块执行同步代码块的方法。如果拿不到锁,就会处于等待状态。

同步方法其实也是有锁对象的,同步方法的锁对象就是当前这个类的一个对象,这个对象使用的是this来表示。我们这个时候定义的方法没有通过static关键字进行修饰,所以这个方法是一个非静态同步方法,这里我们在前面加上一个static关键字,就变成了静态同步方法了。静态中是不能直接去访问非静态成员的,所以下面会报错。 

java file 多线程文件锁 java 多线程写文件_java_27

想要解决就需要将上面的tickets也用static修饰

java file 多线程文件锁 java 多线程写文件_ide_28

【静态同步方法的锁对象:当前类名.class--->表示当前类所对应的字节码文件对象】

public class Example12 {

    public static void main(String[] args) {
        //创建一个任务类对象
        Ticket1 ticket = new Ticket1();

        //创建四个线程对象指定线程名称并启动
        new Thread(ticket, "窗口一").start();
        new Thread(ticket, "窗口二").start();
        new Thread(ticket, "窗口三").start();
        new Thread(ticket, "窗口四").start();
    }
}

//创建一个任务类
class Ticket1 implements Runnable {
    //定义一个成员变量,记录票的总数量
    private static int tickets = 10;

    //同步代码块的锁需要一个对象,那么就在这里定义一个对象,作为同步代码块中的锁。
    private Object lock = new Object();


    @Override
    public void run() {
        while (true) {
            saleTicket();
            if (tickets <= 0) {
                break;
            }
        }
    }

    //创建一个方法
    private static synchronized void saleTicket() {  //同步方法

        if (tickets > 0) {
            try {
                Thread.sleep(10);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "---卖出的票" + tickets--);
        }
    }
}

【注:当给程序加入同步代码块或者同步方法以及静态同步方法以后,程序的执行效率就会变低。原因是:每一次在进入到同步代码块之前都需要来判断一下这个锁有没有被其他线程获取,多了这一步,所以效率就比较低】

十二、死锁问题

有这样一个场景:

一个中国人和一个美国人在一起吃饭,美国人拿了中国人的筷子,中国人拿了美国人的刀叉,两个人开始争执不休:

中国人:“你先给我筷子,我再给你刀叉!”

美国人:“你先给我刀叉,我再给你筷子!”

结果可想而知,两个人都吃不到饭。

这个例子中的中国人和美国人相当于不同的线程,筷子和刀叉就相当于锁。两个线程在运行时都在等待对方的锁,这样便造成了程序的停滞,这种现象称为死锁。

【死锁现象的出现长长是由于写了同步代码块的嵌套】

public class Example13 {

    public static void main(String[] args) {

        //创建两个任务类的对象
        DeadLockThread d1 = new DeadLockThread(true);
        DeadLockThread d2 = new DeadLockThread(true);

        //创建两个线程对象并启动
        new Thread(d1, "Chinese").start();
        new Thread(d2, "American").start();
    }
}

//创建一个任务类
class DeadLockThread implements Runnable {

    //定义两个锁
    static Object chopsticks = new Object();
    static Object knifeAndFork = new Object();

    //定义一个boolean类型变量
    private Boolean flag;

    //提供一个有参构造方法
    DeadLockThread(boolean flag) {
        this.flag = flag;
    }


    @Override
    public void run() {
        if (flag) {
            while (true) {
                //进行同步代码块的嵌套
                synchronized (chopsticks) {
                    System.out.println(Thread.currentThread().getName() + "---if---chopsticks");

                    synchronized (knifeAndFork) {
                        System.out.println(Thread.currentThread().getName() + "---if---knifeAndFork");
                    }
                }
            }
        } else {
            while (true) {
                synchronized (knifeAndFork) {
                    System.out.println(Thread.currentThread().getName() + "---else---knifeAndFork");
                    synchronized (chopsticks) {
                        System.out.println(Thread.currentThread().getName() + "---else---chopsticks");
                    }
                }
            }

        }
    }
}

java file 多线程文件锁 java 多线程写文件_System_29

这个时候运行可以发现并没有出现死锁,原因是

这里的两个都是true

java file 多线程文件锁 java 多线程写文件_System_30

 始终只能执行下面的这个代码,就不会出现死锁现象

java file 多线程文件锁 java 多线程写文件_ide_31

将上面这个改为false

java file 多线程文件锁 java 多线程写文件_ide_32

此时运行:就卡在这了,美国人拿到了筷子,中国人拿到了刀叉,也就是出现了死锁问题 。

java file 多线程文件锁 java 多线程写文件_ide_33

 我的理解就是:总共只有两把锁,Chinese执行false拿了一把knifeAndFork的锁,这是两个线程,在Chinese执行的同时American也同时在执行true拿了一把chopsticks的锁。这时,Chinese想拿到chopsticks的锁但是锁只有一把已经被American拿走了。同时American也想拿knifeAndFork的锁,但是一样的knifeAndFork的锁已经被Chinese拿走了。所以这时中国和美国陷入了僵持。就出现了死锁问题。

综合案例

这里一定要再自己思考一遍进行巩固,加深对前面的理解。

一、模拟银行存取钱

在银行办理业务时,通常银行会开多个窗口,客户排队等候,窗口办理完业务,会呼叫下一个用户办理业务。本案例要求编写一个程序模拟银行存取钱业务办理。

假如有两个用户在存取钱,两个用户分别操作各自的账户,并在控制台打印存取钱的数量以及账户的余额。

【思考:在这里有两个操作也就是两个线程,一个是存钱一个是取钱。这时的两个用户在同时操作但是在不同的窗口,操作各自的账户】

用户类

//假如有两个用户在存取钱,两个用户分别操作各自的账户,并在控制台打印存取钱的数量以及账户的余额。

/**
 * 这里是用户类
 */

import java.util.Date;

public class User {

    //定义成员变量
    private String u_name;           //用户名
    private String u_login_name;     //登录名--银行卡卡号
    private String u_login_pwd;      //登陆密码
    private String u_wallet;         //钱包余额
    private Date draw_money_time;    //取钱时间
    private Date save_money_time;    //存钱时间

    //无参构造方法
    public User() {
    }

    //有参数构造方法

    public User(String u_name, String u_login_name, String u_login_pwd, String u_wallet) {
        this.u_name = u_name;
        this.u_login_name = u_login_name;
        this.u_login_pwd = u_login_pwd;
        this.u_wallet = u_wallet;
    }

    public String getU_name() {
        return u_name;
    }

    public void setU_name(String u_name) {
        this.u_name = u_name;
    }

    public String getU_login_name() {
        return u_login_name;
    }

    public void setU_login_name(String u_login_name) {
        this.u_login_name = u_login_name;
    }

    public String getU_login_pwd() {
        return u_login_pwd;
    }

    public void setU_login_pwd(String u_login_pwd) {
        this.u_login_pwd = u_login_pwd;
    }

    public String getU_wallet() {
        return u_wallet;
    }

    public void setU_wallet(String u_wallet) {
        this.u_wallet = u_wallet;
    }

    public Date getDraw_money_time() {
        return draw_money_time;
    }

    public void setDraw_money_time(Date draw_money_time) {
        this.draw_money_time = draw_money_time;
    }

    public Date getSave_money_time() {
        return save_money_time;
    }

    public void setSave_money_time(Date save_money_time) {
        this.save_money_time = save_money_time;
    }
}

银行类

import java.math.BigDecimal;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;

/**
 * 银行类中至少应该有两个方法:
 * 1.存钱
 * 2.取钱
 */

public class Bank {

    //定义一个变量,来初始化银行的用户信息
    private List<User> userList = new ArrayList<>();

    //提供一个构造方法,需要接受一个参数类型为List
    public Bank(List<User> userList) {
        this.userList = userList;
    }

    public List<User> getUserList() {
        return userList;
    }

    public void setUserList(List<User> userList) {
        this.userList = userList;
    }

    //存钱方法
    public Boolean saveMoney(String card, String pwd, String moneyNum) {//需要知道卡号,密码以及存钱数量
        //根据卡号在UesrList中查找具体的用户
        //只需要给某一块代码加上同步代码块就不需要在方法上加上synchronized
        User u = getUserByCard(card);

        synchronized (Bank.class) {
            //判断卡号和密码是否相同
            if (u.getU_login_name().equals(card) && u.getU_login_pwd().equals(pwd)) {
                //获取余额---这里的u.getU_wallet()是一个字符串后面会不方便进行运算
                BigDecimal oldDate = new BigDecimal(u.getU_wallet());
                //要存入的钱与上面要进行相同的操作
                BigDecimal money = new BigDecimal(moneyNum);
                //将上面两者进行相加,再转换成字符串
                u.setU_wallet(oldDate.add(money).toString());
                //记录存钱的时间
                u.setSave_money_time(new Date());

                //提示信息 【这里格式化时间用SimpleDateFormat】
                System.out.println(Thread.currentThread().getName() + "存钱--->" + u.getU_name() +
                        "在" + new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(u.getSave_money_time()) +
                        "存了[" + moneyNum + "]元钱,余额" + u.getU_wallet());

                //返回
                return true;

            }
        }
        //存钱失败
        System.out.println(u.getU_name() + "存钱失败了");
        return false;
    }

    //取钱方法
    public Boolean getMoney(String card, String pwd, String moneyNum) {//需要知道卡号,密码以及取钱数量
        //根据卡号在UesrList中查找具体的用户
        //只需要给某一块代码加上同步代码块就不需要在方法上加上synchronized
        User u = getUserByCard(card);
        synchronized (Bank.class) {
            //判断是否满足条件银行有该用户,卡号和密码正确
            if (u != null && u.getU_login_name().equals(card) && u.getU_login_pwd().equals(pwd)) {
                //获取余额
                BigDecimal oldDate = new BigDecimal(u.getU_wallet());
                //取钱数额
                BigDecimal money = new BigDecimal(moneyNum);
                //判断要取得钱的数量是否在余额内
                if (oldDate.compareTo(money) > 0) {
                    //这里就可以取钱
                    u.setU_wallet(oldDate.subtract(money).toString());
                    u.setDraw_money_time(new Date());

                    //提示信息 【这里格式化时间用SimpleDateFormat】
                    System.out.println(Thread.currentThread().getName() + "取钱--->" + u.getU_name() +
                            "在" + new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(u.getDraw_money_time()) +
                            "取了[" + moneyNum + "]元钱,余额" + u.getU_wallet());

                    return true;
                } else {
                    //取钱失败,余额不足
                    System.out.println(u.getU_name() + "要取[" + moneyNum + "]元钱,余额" + u.getU_wallet() + "不足");
                    return false;
                }
            }
        }
        //返回
        System.out.println(card + "取钱失败");
        return false;
    }

    //定义一个方法获取当前用户
    //后期是会通过多线程进行操作的,所以需要考虑线程的安全问题
    public synchronized User getUserByCard(String card) {
        for (User u : userList) {
            //判断登录名(卡号)是否与列表中的一样,一样就返回u
            if (u.getU_login_name().equals(card)) {
                return u;
            }
        }
        //如果列表中没有就返回空
        return null;
    }

    //让线程进行休眠
    public void delayTime(Integer nim){
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

 测试类

package DuoXianCheng.Example14;

import java.util.ArrayList;
import java.util.List;

public class BankText {
    public static void main(String[] args) {

        //创建两个用户
        User u1 = new User("张三", "123456", "123", "100");
        User u2 = new User("鲁正婷", "258741", "0924", "0");

        //创建集合并将用户放在集合中
        List<User> list = new ArrayList<>();
        list.add(u1);
        list.add(u2);

        //创建银行对象
        Bank bank = new Bank(list);

        //创建两个线程,进行两个用户的存取
        Thread t1 = new Thread("线程一") {
            @Override
            public void run() {
                //存取10次
                for (int i = 0; i <= 10; i++) {
                    //存钱
                    bank.saveMoney("123456", "123", "80");
                    bank.delayTime(100);
                    bank.getMoney("258741", "0924", "150");
                    bank.delayTime(100);
                }
            }
        };

        Thread t2 = new Thread("线程二") {
            @Override
            public void run() {
                //存取10次
                for (int i = 0; i <= 10; i++) {
                    //存钱
                    bank.getMoney("123456", "123", "100");
                    bank.delayTime(100);
                    bank.saveMoney("258741", "0924", "50");
                    bank.delayTime(100);
                }
            }
        };

        //启动线程
        t1.start();
        t2.start();
    }
}

最终结果:(我这里是进行了10次的存取)

java file 多线程文件锁 java 多线程写文件_java_34

 

java file 多线程文件锁 java 多线程写文件_System_35

二、工人搬砖

在某个工地,需要把100块砖搬运到二楼,现在有工人张三和李四,张三每次搬运3块砖,每趟需要10分钟,李四每次搬运5块砖,每趟需要12分钟。本案例要求编写程序分别计算两位工人搬完100块砖需要多长时间。本案例要求使用多线程的方式实现。

 【分析:现在需要创建两个线程一个是只能是张三一个是李四。现在总共有100块砖。

张三的效率:3块/10min  李四的效率:5块/12min   

首先计算每个工人搬完需要多少趟张三需要34趟就是340分钟,李四需要20趟就是240分钟】

public class BanZhuan {
    public static void main(String[] args) throws InterruptedException {
        //定义变量统计时间
        int y = 0;
        //创建一个线程对象并启动
        new Thread(new ZhangSan()).start();

        //另一个for循环
        for (int i = 1; i <= 100; i++) {
            if (i % 3 == 0) {
                try {
                    Thread.sleep(100);
                    System.out.println("李四正在搬" + i);
                    y += 10;
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            } else if (i == 100) {
                System.out.println("李四正在搬" + i + "李四搬完了");
                y += 10;
            }
        }
        System.out.println("李四总共用时" + y + "min");
    }
}

//创建任务类
class ZhangSan implements Runnable {
    //定义变量统计时间
    private int y = 0;

    @Override
    public void run() {
        for (int i = 1; i <= 100; i++) {
            if (i % 5 == 0 && i < 100) {
                try {
                    Thread.sleep(120);
                    System.out.println("张三正在搬" + i);
                    y += 12;
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            } else if (i == 100) {
                System.out.println("张三正在搬" + i + "张三搬完了");
                y += 12;
            }
        }
        System.out.println("张三总共用时" + y + "min");
    }
}

java file 多线程文件锁 java 多线程写文件_java file 多线程文件锁_36

三、小朋友就餐

java file 多线程文件锁 java 多线程写文件_java_37

一圆桌前坐着5位小朋友,两个人中间有一只筷子,桌子中央有面条。小朋友边吃边玩,当饿了的时候拿起左右两只筷子吃饭,必须拿到两只筷子才能吃饭。但是,小朋友在吃饭过程中,可能会发生5个小朋友都拿起自己右手边的筷子,这样每个小朋友都因缺少左手边的筷子而没有办法吃饭。本案例要求编写一个程序解决小朋友就餐问题,使每个小朋友都能成功就餐。

 

【本质:这里有5个小朋友就相当于5个线程,然后这里总共也就只有5只筷子。结合下图分析,每个小朋友都需要一左一右两只筷子才吃到饭】

java file 多线程文件锁 java 多线程写文件_System_38

在这里利用索引可以来解决这个问题 

 

java file 多线程文件锁 java 多线程写文件_System_39

 5个小朋友只给5只筷子!!这饭不吃也罢 !!

//测试类
public class Test {
    public static void main(String[] args) {

        //创建一个Fork对象
        Fork fork = new Fork();

        //创建5个线程并启动
        new Child("0", fork).start();
        new Child("1", fork).start();
        new Child("2", fork).start();
        new Child("3", fork).start();
        new Child("4", fork).start();

    }
}

//筷子类
class Fork {
    //定义一个boolean类型的数组,初始化5只筷子的使用状态
    private boolean[] used = {false, false, false, false, false};

    //获取筷子
    public synchronized void takeFork() {
        //获取执行该方法的线程名称
        String name = Thread.currentThread().getName();
        //将名称转化为int型,方便后面得到索引
        int i = Integer.parseInt(name);

        //左手边的筷子
        boolean leftFork = used[i];
        //右手边的筷子
        boolean rightFork;
        if (i == 0) {
            rightFork = used[4];
        } else {
            rightFork = used[i - 1];
        }

        //判断筷子是否处于空闲状态
        //当任意一只筷子处于使用状态时,其他的小朋友就处于等待状态
        while (leftFork || rightFork) {
            try {
                wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        //使用筷子,设置筷子的状态
        //左手边
        used[i] = true;
        //右手边
        if (i == 0) {
            used[4] = true;
        } else {
            used[i - 1] = true;
        }
    }

    //吃完了设置放下筷子的方法
    public synchronized void putFork() {
        //获取执行该方法的线程名称
        String name = Thread.currentThread().getName();
        //将名称转化为int型,方便后面得到索引
        int i = Integer.parseInt(name);

        //放左手筷子
        used[i] = false;
        //放右手边的筷子
        if (i == 0) {
            used[4] = false;
        } else {
            used[i - 1] = false;
        }
        //放完了之后唤醒其他的线程
        notifyAll();
    }
}

//小朋友类
class Child extends Thread {
    //定义两个成员变量
    private String name;
    private Fork fork;

    //定义构造方法进行初始化
    public Child(String name, Fork fork) {
        super(name);
        this.name = name;
        this.fork = fork;
    }

    @Override
    public void run() {
        while (true) {
            //等待的小朋友
            looking();

            //获取筷子
            fork.takeFork();
            eat();

            //放下筷子
            fork.putFork();
        }
    }

    private void eat() {
        System.out.println("小朋友"+name+"正在吃饭");
        try {
            sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    private void looking() {
        System.out.println("小朋友" + name + "在等待");
        try {  //
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}