重学java 39.多线程 — 线程安全

逐渐成为一个情绪稳定且安静成长的人

------24.5.24

线程安全

什么时候发生?

当多个线程访问同一个资源时,导致了数据有问题,出现并发 问题,数据不能及时更新,导致数据发生错误,出现线程安全问题

多线程安全问题示例

java 复制代码
package S71ThreadSafe;

public class MyTicket implements Runnable{
    // 定义100章票
    int ticket = 100;
    @Override
    public void run(){
        while(true){
            if(ticket>0){
                System.out.println(Thread.currentThread().getName()+"买了第"+ticket+"张票");
                ticket--;
            }
        }
    }
}
java 复制代码
package S71ThreadSafe;

public class Demo210Test {
    public static void main(String[] args) {
        // new一次,分别传递到三个线程对象中去
        MyTicket myTicket = new MyTicket();

        Thread th1= new Thread(myTicket, "赵四");
        Thread th2= new Thread(myTicket, "刘能");
        Thread th3= new Thread(myTicket, "广坤");

        th1.start();
        th2.start();
        th3.start();
    }
}

解决线程安全问题方式一:同步代码块

synchronized代码块 == 上锁 可以解决线程安全问题,不保证线程顺序

1.格式:

synchronized(任意接收对象){

线程要执行的语句;

}

2.任意对象:

就是我们的锁对象

3.执行:

一个线程拿到锁之后,会进入到同步代码块中执行,在此期间,其他线程拿不到锁,就进不去同步代码块。需要在同步代码块外面等待排队,需要等待执行的线程执行完毕,出了同步代码块,相当于释放锁了,等待的线程才能抢到锁,才能进入到同步代码块中执行,上的锁必须是同一个锁对象,同一把锁

java 复制代码
package S71ThreadSafe;

import java.util.Scanner;

public class MyTicket implements Runnable{
    // 定义100章票
    int ticket = 100;

    // 任意new一个对象
    Object obj = new Object();
    Scanner sc = new Scanner(System.in);

    @Override
    public void run(){
        while(true){
            synchronized (sc) {
                if (ticket > 0) {
                    System.out.println(Thread.currentThread().getName() + "买了第" + ticket + "张票");
                    ticket--;
                }
            }
        }
    }
}
java 复制代码
package S71ThreadSafe;

public class Demo210Test {
    public static void main(String[] args) {
        // new一次,分别传递到三个线程对象中去
        MyTicket myTicket = new MyTicket();

        Thread th1= new Thread(myTicket, "赵四");
        Thread th2= new Thread(myTicket, "刘能");
        Thread th3= new Thread(myTicket, "广坤");

        th1.start();
        th2.start();
        th3.start();
    }
}

解决线程安全问题方式二:同步方法

1.普通同步方法_非静态

在普通方法内用一个同步代码块

① 格式:

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

方法体

return 结果

}

② 默认锁:

this

java 复制代码
package S73ThreadSafeSolve2;

public class Demo212Test {
    public static void main(String[] args) {
        // new一次,分别传递到三个线程对象中去
        MyTicket myTicket = new MyTicket();
        System.out.println(myTicket);

        Thread th1= new Thread(myTicket, "赵四");
        Thread th2= new Thread(myTicket, "刘能");
        Thread th3= new Thread(myTicket, "广坤");

        th1.start();
        th2.start();
        th3.start();
    }
}
java 复制代码
package S73ThreadSafeSolve2;

import java.util.Scanner;

public class MyTicket implements Runnable{
    // 定义100章票
    int ticket = 100;

    Scanner sc = new Scanner(System.in);

    @Override
    public void run(){
        while(true){
            try{
                Thread.sleep(100L);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            // 调用同步方法
            // method01();
            method02();
        }
    }

    // 定义一个同步方法
    public synchronized void method01(){
        if (ticket > 0) {
            System.out.println(Thread.currentThread().getName() + "买了第" + ticket + "张票");
            ticket--;
        }
    }

    // 在普通方法内用一个同步代码块
    public void method02(){
        synchronized (this){
            System.out.println(this+"------------------------------------------------------------");
            if (ticket>0){
                System.out.println(Thread.currentThread().getName() + "买了第" + ticket + "张票");
                ticket--;
            }
        }
    }

}

只new了一次对象

2.静态同步方法

① 格式

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

方法体

return 结果

}

② 默认锁

class对象

java 复制代码
package S74ThreadSafeSolve3;

import java.util.Scanner;

public class MyTicket implements Runnable{
    // 定义100章票
    static int ticket = 100;

    Scanner sc = new Scanner(System.in);

    @Override
    public void run(){
        while(true){
            try{
                Thread.sleep(100L);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            // 调用同步方法
            method02();
        }
    }

    // 在普通方法内用一个同步代码块
    public static void method02(){
        synchronized (MyTicket.class){
            if (ticket>0){
                System.out.println(Thread.currentThread().getName() + "买了第" + ticket + "张票");
                ticket--;
            }
        }
    }

}
java 复制代码
package S74ThreadSafeSolve3;

public class Demo213Test {
    public static void main(String[] args) {
        // new一次,分别传递到三个线程对象中去
        MyTicket myTicket = new MyTicket();
        System.out.println(myTicket);

        Thread th1= new Thread(myTicket, "赵四");
        Thread th2= new Thread(myTicket, "刘能");
        Thread th3= new Thread(myTicket, "广坤");

        th1.start();
        th2.start();
        th3.start();
    }
}

StringBuider是线程不安全的,StringBuffer加了死锁是线程安全,线程同步的

相关推荐
gjh12083 分钟前
Easy-excel监听器中对批量上传的工单做错误收集
java·spring boot
红衣女妖仙6 分钟前
JXLS 库导出复杂 Excel
java·excel·jxls·java 导出 excel
Hellyc13 分钟前
JAVA八股文:异常有哪些种类,可以举几个例子吗?Throwable类有哪些常见方法?
java·开发语言
西岭千秋雪_22 分钟前
Redis缓存架构实战
java·redis·笔记·学习·缓存·架构
2301_8035545229 分钟前
c++中的绑定器
开发语言·c++·算法
海棠蚀omo37 分钟前
C++笔记-位图和布隆过滤器
开发语言·c++·笔记
杰哥技术分享1 小时前
Yii2 安装-yii2-imagine
开发语言·yii
五岳1 小时前
深入研究使用DozerMapper复制List<Ojbect>前后元素类型不一致的问题
java·爬坑
人生在勤,不索何获-白大侠1 小时前
day15——Java常用API(二):常见算法、正则表达式与异常处理详解
java·算法·正则表达式
Bug退退退1231 小时前
RabbitMQ 高级特性之消息确认
java·分布式·rabbitmq