[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 API
java如何请求接口然后终止某个线程
通过本文的介绍,您应该能够理解如何在Java中请求接口并根据返回结果终止某个线程。合理使用标志位或 `interrupt`方法可以确保线程的安全终止,而处理好网络请求中的各种异常情况,可以提高程序的稳定性和可靠性。
35 6
|
14天前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
14天前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
38 3
|
15天前
|
监控 Java 开发者
深入理解Java中的线程池实现原理及其性能优化####
本文旨在揭示Java中线程池的核心工作机制,通过剖析其背后的设计思想与实现细节,为读者提供一份详尽的线程池性能优化指南。不同于传统的技术教程,本文将采用一种互动式探索的方式,带领大家从理论到实践,逐步揭开线程池高效管理线程资源的奥秘。无论你是Java并发编程的初学者,还是寻求性能调优技巧的资深开发者,都能在本文中找到有价值的内容。 ####
|
18天前
|
监控 Java 数据库连接
Java线程管理:守护线程与用户线程的区分与应用
在Java多线程编程中,线程可以分为守护线程(Daemon Thread)和用户线程(User Thread)。这两种线程在行为和用途上有着明显的区别,了解它们的差异对于编写高效、稳定的并发程序至关重要。
26 2
|
2月前
|
存储 消息中间件 资源调度
C++ 多线程之初识多线程
这篇文章介绍了C++多线程的基本概念,包括进程和线程的定义、并发的实现方式,以及如何在C++中创建和管理线程,包括使用`std::thread`库、线程的join和detach方法,并通过示例代码展示了如何创建和使用多线程。
58 1
C++ 多线程之初识多线程
|
2月前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
27 3
|
2月前
|
Java 开发者
在Java多线程编程中,选择合适的线程创建方法至关重要
【10月更文挑战第20天】在Java多线程编程中,选择合适的线程创建方法至关重要。本文通过案例分析,探讨了继承Thread类和实现Runnable接口两种方法的优缺点及适用场景,帮助开发者做出明智的选择。
23 2
|
2月前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
38 2
|
2月前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
44 1