[java进阶]——多线程Thread类,处理并发异常的方法(下)

简介: [java进阶]——多线程Thread类,处理并发异常的方法
public class MyCallable implements Callable<Integer> {
    @Override
    public Integer call() throws Exception {
        //线程要执行的方法
        return total;
    }
}
public static void main(String[] args) throws ExecutionException, InterruptedException {
        /*
        * 多线程的第三种实现方式
        * 1.定义一个类实现callable接口
        * 2.重写里面的call方法
        *
        * 3.创建mycallable对象 - 任务
        * 4.创建futuretask对象 - 管理线程结果
        * 5.创建thread对象 - 创建线程
        *
        * */
        MyCallable myCallable = new MyCallable();
        FutureTask<Integer> integerFutureTask = new FutureTask<Integer>(myCallable);
        Thread thread = new Thread(integerFutureTask);
    }

3.4应用场景总结

如果要用到线程的处理结果(返回值),就用第三种实现方法

如果你想继承其它子类,用第二种或第三种

如果是单纯的继承Thread类,用第一种

因为java只支持单继承,不支持多继承

四、多线程应用场景

🖌用多线程模拟三个窗口卖100张票的过程

思路:1.定义一个线程类  2.重写里面的run方法模拟买票  3.创建3个对象模拟3个窗口

public class Mythread extends Thread {
    static int ticket = 0;//静态修饰
    @Override
    public void run() {
        while (ticket++ < 100) {
                try {
                    //休眠1秒
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName() + "窗口正在卖第" + ticket + "张票");
            }
    }
}
public class Test1 {
    public static void main(String[] args) {
        //实现三个窗口卖100张票的
        Thread t1 = new Mythread("窗口1");
        Thread t2 = new Mythread("窗口2");
        Thread t3 = new Mythread("窗口3");
        t2.start();
        t1.start();
        t3.start();
}

我们把代码跑起来

这和现实生活中的完全不一样,这其实引发了并发异常,简单说一下,开启了3个进程,每个进程被cpu选中是随机的,当线程1被选中并且代码运行到while循环里,cpu又去执行线程2,还有可能去执行线程三等等,所以导致同时卖出了第三张票

五、解决并发问题的方法

5.1 synchronized()关键字 - 同步代码块

原理分析:

给可能发生异常的代码加上锁之后,其他线程只能等待该线程执行完释放该锁。

代码分析:

synchronized()

括号要求是一个对象,什么类型都可,但针对该类必须唯一,也就是这把锁是管理该类的一把锁,可以是静态修饰得对象,可以是Mythread得字节码文件。

还是以买票举例

public class Mythread extends Thread {
    static int ticket = 0;//静态修饰
    @Override
    public void run() {
        while (ticket++ < 100) {
            //synchronized()括号里写一个锁对象,必须针对该类唯一
            synchronized (Mythread.class) {
                try {
                    //休眠1秒
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName() + "窗口正在卖第" + ticket + "张票");
            }
        }
    }
}

把可能引发并发异常的代码抽象成方法 用synchronized修饰

这是的锁对象会默认创建,如果是静态的方法,该对象是ths 如果是非静态,该对象是该类的class文件

@Override
    public void run() {
        while (ticket++ < 100000) {
            //同步方法
            method();
        }
    }
    private synchronized void method() {
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(Thread.currentThread().getName() + "窗口正在卖第" + ticket + "张票");
    }
5.2使用lock锁

获取静态锁对象

调用lock方法把并发代码锁起来

并发代码用try-catch包裹

释放锁放在finally代码块里

public class MyThread extends Thread{
    static int ticket = 0;
    static Lock lock = new ReentrantLock();
    @Override
    public void run() {
        while(ticket++ <= 100){
            lock.lock();
            try {
                if(ticket <= 100){
                    System.out.println(Thread.currentThread().getName() + "窗口卖出第" + ticket + "张票");
                    Thread.sleep(10);
                }
            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                lock.unlock();
            }
        }
    }
}

这样修改后的买票系统就能正常实现啦

注意:不要使用锁嵌套,可能出现死锁。看下面代码

public class DeadlockExample {
    private static final Object lock1 = new Object();
    private static final Object lock2 = new Object();
    public static void main(String[] args) {
        Thread thread1 = new Thread(() -> {
            synchronized (lock1) {
                System.out.println("Thread 1 acquired lock1");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                synchronized (lock2) {
                    System.out.println("Thread 1 acquired lock2");
                }
            }
        });
        Thread thread2 = new Thread(() -> {
            synchronized (lock2) {
                System.out.println("Thread 2 acquired lock2");
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                synchronized (lock1) {
                    System.out.println("Thread 2 acquired lock1");
                }
            }
        });
        thread1.start();
        thread2.start();
    }
}

📕总结

为什么会有并发异常,并发异常会导致什么后果

如何解决并发异常

相关文章
|
5天前
|
存储 Java 索引
Java快速入门之数组、方法
### Java快速入门之数组与方法简介 #### 一、数组 数组是一种容器,用于存储同种数据类型的多个值。定义数组时需指定数据类型,如`int[]`只能存储整数。数组的初始化分为静态和动态两种: - **静态初始化**:直接指定元素,系统自动计算长度,如`int[] arr = {1, 2, 3};` - **动态初始化**:手动指定长度,系统给定默认值,如`int[] arr = new int[3];` 数组访问通过索引完成,索引从0开始,最大索引为`数组.length - 1`。遍历数组常用`for`循环。常见操作包括求和、找最值、统计特定条件元素等。
|
1天前
|
Java
Java快速入门之类、对象、方法
本文简要介绍了Java快速入门中的类、对象和方法。首先,解释了类和对象的概念,类是对象的抽象,对象是类的具体实例。接着,阐述了类的定义和组成,包括属性和行为,并展示了如何创建和使用对象。然后,讨论了成员变量与局部变量的区别,强调了封装的重要性,通过`private`关键字隐藏数据并提供`get/set`方法访问。最后,介绍了构造方法的定义和重载,以及标准类的制作规范,帮助初学者理解如何构建完整的Java类。
|
2天前
|
安全 Java 程序员
Java面试必问!run() 和 start() 方法到底有啥区别?
在多线程编程中,run和 start方法常常让开发者感到困惑。为什么调用 start 才能启动线程,而直接调用 run只是普通方法调用?这篇文章将通过一个简单的例子,详细解析这两者的区别,帮助你在面试中脱颖而出,理解多线程背后的机制和原理。
33 12
|
3天前
|
算法 Java API
Java 方法注释:规范、实用和高质量的写法
本文深入探讨了如何编写高质量的 Java 方法注释
25 11
|
3天前
|
SQL Java 数据库连接
【潜意识Java】Java中JDBC过时方法的替代方案以及JDBC为什么过时详细分析
本文介绍了JDBC中一些常见过时方法及其替代方案。
21 5
|
18天前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
80 17
|
28天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者
|
13天前
|
缓存 安全 算法
Java 多线程 面试题
Java 多线程 相关基础面试题
|
30天前
|
NoSQL Redis
单线程传奇Redis,为何引入多线程?
Redis 4.0 引入多线程支持,主要用于后台对象删除、处理阻塞命令和网络 I/O 等操作,以提高并发性和性能。尽管如此,Redis 仍保留单线程执行模型处理客户端请求,确保高效性和简单性。多线程仅用于优化后台任务,如异步删除过期对象和分担读写操作,从而提升整体性能。
65 1
|
3月前
|
存储 消息中间件 资源调度
C++ 多线程之初识多线程
这篇文章介绍了C++多线程的基本概念,包括进程和线程的定义、并发的实现方式,以及如何在C++中创建和管理线程,包括使用`std::thread`库、线程的join和detach方法,并通过示例代码展示了如何创建和使用多线程。
72 1

热门文章

最新文章