java面试题-并发基础

news2025/1/12 15:47:46

1.多线程的出现是要解决什么问题的? 本质什么?

  1. 提高程序性能:单线程程序只能按照固定的顺序依次执行每个任务,无法同时处理多个任务。多线程技术可以在同一时间内执行多个任务,从而提高程序的运行效率和响应速度。

  1. 提高程序的并发性:单线程程序在处理多个任务时,需要等待某个任务完成后才能进行下一个任务,导致程序无法同时处理多个请求。多线程技术可以同时处理多个请求,提高程序的并发性。

  1. 充分利用多核CPU:现代计算机大多都具备多核CPU,但是单线程程序只能利用其中的一个核心,无法充分利用CPU的性能。多线程技术可以同时使用多个CPU核心,充分利用计算机的性能。

  1. 程序设计的灵活性:多线程技术可以将程序拆分成多个独立的任务,每个任务可以独立进行设计和测试,从而提高程序的灵活性和可维护性。

多线程的本质是通过将程序拆分成多个线程并行执行,实现程序的并发执行,从而提高程序的效率和性能。同时,多线程也带来了一些问题,如线程安全性、死锁、竞争条件等,需要仔细考虑和处理。

2.Java是怎么解决并发问题的?

Java提供了多种机制来解决并发问题,包括锁、原子变量、线程池、并发集合等等。其中最常用的机制包括synchronized关键字、volatile关键字、以及并发集合类如ConcurrentHashMap、ConcurrentLinkedQueue等。

  1. synchronized关键字

synchronized关键字可以保证同一时刻只有一个线程可以访问被synchronized修饰的代码块或方法,从而避免多个线程同时访问同一个共享资源,导致数据不一致或者线程安全问题。synchronized关键字可以用于方法、代码块、静态方法等。

  1. volatile关键字

volatile关键字可以确保在多线程环境下变量的可见性和有序性,从而避免了由于指令重排等原因导致的线程安全问题。使用volatile关键字修饰的变量在每次被访问时,都会从内存中读取最新的值,而不是从缓存中读取,保证了变量的可见性。

  1. 原子变量

Java提供了原子变量类如AtomicInteger、AtomicLong等,这些类提供了一些线程安全的操作,如自增、自减、比较并交换等操作。使用原子变量可以避免多线程同时修改同一个变量导致的线程安全问题。

  1. 线程池

线程池是一种重用线程的机制,可以减少线程创建和销毁的开销。Java提供了Executor框架来支持线程池的使用。通过使用线程池,可以减少线程的创建和销毁次数,从而提高了程序的执行效率。

  1. 并发集合类

Java提供了一系列的并发集合类,如ConcurrentHashMap、ConcurrentLinkedQueue等,这些集合类可以在多线程环境下保证线程安全,从而避免了使用传统的集合类时可能出现的线程安全问题。

  1. JMM

Java内存模型(Java Memory Model,简称JMM)是Java用来描述多线程访问共享内存时,内存可见性、指令重排、原子性等特性的规范。Java通过JMM来保证多线程之间的可见性、有序性和原子性,从而避免了线程安全问题。

综上所述,Java通过提供锁、原子变量、线程池、并发集合等机制以及JMM规范来解决并发问题,从而保证了多线程环境下程序的正确性和安全性。

3.线程安全有哪些实现思路?

线程安全的实现思路主要有以下几种:

  1. 互斥同步

互斥同步是通过锁来保证多个线程之间的互斥访问,从而避免多个线程同时访问共享资源导致的数据不一致或者线程安全问题。常见的互斥同步机制包括synchronized关键字、Lock接口等。

  1. 非阻塞同步

非阻塞同步是一种无需线程阻塞等待的同步方式,通过CAS操作等原子性操作来保证线程之间的互斥访问,从而避免了线程阻塞等待的开销。Java中的Atomic类就是基于非阻塞同步的机制。

  1. 读写同步

读写同步是一种针对读写操作进行区分的同步方式,通过读写锁来实现多个线程之间的并发读取、独占写入,从而提高程序的执行效率。Java中的ReentrantReadWriteLock就是一个典型的读写锁。

  1. 信号量

信号量是一种多线程同步机制,通过信号量来控制同时访问共享资源的线程数,从而实现线程之间的同步。Java中的Semaphore类就是一种基于信号量的同步机制。

  1. ThreadLocal

ThreadLocal是一种线程本地变量,它可以为每个线程提供独立的变量副本,从而避免多个线程之间共享变量导致的线程安全问题。ThreadLocal的实现原理是为每个线程维护一个独立的变量副本,从而保证线程之间互不干扰。

  1. 并发容器

Java提供了一些并发容器类,如ConcurrentHashMap、ConcurrentLinkedQueue等,这些容器类可以在多线程环境下保证线程安全,从而避免了使用传统的容器类时可能出现的线程安全问题。

综上所述,实现线程安全的方式有很多种,可以根据具体的应用场景来选择最合适的实现方式。

4.如何理解并发和并行的区别?

并发是指一个处理器同时处理多个任务。

并行是指多个处理器或者是多核的处理器同时处理多个不同的任务。

5.线程有哪几种状态? 分别说明从一种状态到另一种状态转变有哪些方式?

  1. 新建(New):当一个线程对象被创建但还没有开始运行时,它的状态就是新建状态。

  1. 就绪(Runnable):当一个线程处于就绪状态时,表示它已经准备好了,等待系统的调度程序分配CPU时间片段,以便开始执行它的run()方法。

  1. 运行(Running):当一个线程正在执行run()方法时,它的状态就是运行状态。

  1. 阻塞(Blocked):当一个线程因为某些原因暂时停止执行,以等待系统资源或者等待某个条件得到满足,此时它的状态就是阻塞状态。

  1. 等待(Waiting):当一个线程因为某些条件没有得到满足而处于等待状态时,它的状态就是等待状态。在等待状态下,线程不会占用CPU资源,直到其它线程调用notify()或notifyAll()方法来唤醒它。

  1. 计时等待(Timed Waiting):当一个线程处于计时等待状态时,它会等待一定的时间,或者等待某个条件被满足,直到时间到期或者条件被满足时才会被唤醒。

  1. 终止(Terminated):当一个线程的run()方法执行完毕,或者因为出现了异常而结束时,它的状态就是终止状态。

一个线程在不同状态之间转换的方式有以下几种:

  1. 新建状态转变为就绪状态:通过调用start()方法来启动线程,使它进入就绪状态。

  1. 就绪状态转变为运行状态:当系统调度程序把CPU时间片段分配给了线程,使它开始执行run()方法时,线程就从就绪状态转变为运行状态。

  1. 运行状态转变为就绪状态:当一个线程的run()方法执行完毕或者调用了wait()方法,使它暂时停止执行时,它就从运行状态转变为就绪状态。

  1. 运行状态转变为阻塞状态:当一个线程因为某些原因而暂停执行,如等待某个条件得到满足时,它就从运行状态转变为阻塞状态。

  1. 阻塞状态转变为就绪状态:当等待的条件得到满足或者等待时间到期时,线程就从阻塞状态转变为就绪状态。

  1. 运行状态、就绪状态、阻塞状态、等待状态转变为终止状态:当一个线程的run()方法执行完毕,或者因为出现了异常而结束时,它的状态就会转变为终止状态。

6.通常线程有哪几种使用方式?

  1. 基于继承Thread类:通过继承Thread类,重写其run()方法来创建线程,并调用start()方法启动线程。例如:

public class MyThread extends Thread {
    @Override
    public void run() {
        // 线程执行的代码
    }
}

MyThread thread = new MyThread();
thread.start();
  1. 实现Runnable接口:通过实现Runnable接口,并将其实例作为Thread的构造函数参数来创建线程。例如:

public class MyRunnable implements Runnable {
    @Override
    public void run() {
        // 线程执行的代码
    }
}

Thread thread = new Thread(new MyRunnable());
thread.start();
  1. 实现Callable接口:与实现Runnable接口类似,但是Callable接口的call()方法可以返回执行结果,并且可以抛出异常。例如:

public class MyCallable implements Callable<String> {
    @Override
    public String call() throws Exception {
        // 线程执行的代码
        return "result";
    }
}

FutureTask<String> task = new FutureTask<>(new MyCallable());
Thread thread = new Thread(task);
thread.start();
  1. 使用线程池:通过线程池可以有效地管理线程的生命周期,避免频繁地创建和销毁线程。例如:

ExecutorService executor = Executors.newFixedThreadPool(5);
executor.submit(new MyRunnable());
executor.submit(new MyCallable());
executor.shutdown();

7.线程的中断方式有哪些?

线程的中断方式主要有以下三种:

  1. 通过调用interrupt()方法:这种方式是最常见的中断线程的方式。调用线程的interrupt()方法会将线程的中断状态设置为true,但并不会中断线程的执行。在线程的执行代码中,需要不断地检查线程的中断状态,如果中断状态为true,则可以结束线程的执行。

  1. 通过检查isInterrupted()方法:可以在线程的执行代码中使用isInterrupted()方法来检查线程的中断状态,如果中断状态为true,则可以结束线程的执行。与interrupt()方法不同,调用isInterrupted()方法不会改变线程的中断状态。

  1. 通过使用stop()方法:stop()方法可以立即结束线程的执行,但是这种方式已经被废弃,不建议使用。在使用stop()方法时,线程可能会被中断在执行过程中的任意位置,导致数据不一致和资源泄露等问题。

需要注意的是,使用interrupt()方法和isInterrupted()方法时,需要在线程的执行代码中不断地检查线程的中断状态,以便及时地结束线程的执行。同时,在使用这两种方式中断线程时,需要注意处理InterruptedException异常,以便正确地维护线程的中断状态。

8.线程之间有哪些协作方式?

  1. wait()和notify()/notifyAll()方法:可以使用wait()方法让一个线程等待,直到其他线程调用notify()或notifyAll()方法唤醒它。以下是一个简单的例子:

public class WaitNotifyExample {
    public static void main(String[] args) {
        Object lock = new Object();
        Thread t1 = new Thread(() -> {
            synchronized (lock) {
                try {
                    System.out.println("Thread 1 waiting");
                    lock.wait();
                    System.out.println("Thread 1 notified");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });

        Thread t2 = new Thread(() -> {
            synchronized (lock) {
                System.out.println("Thread 2 notifying");
                lock.notify();
            }
        });

        t1.start();
        t2.start();
    }
}

在上面的例子中,t1线程调用了lock的wait()方法,导致它被阻塞,直到t2线程调用了lock的notify()方法。在t2线程调用notify()方法之后,t1线程就会被唤醒,继续执行。

  1. join()方法:可以使用join()方法让一个线程等待另一个线程执行完成,再继续执行。以下是一个简单的例子:

public class JoinExample {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(() -> {
            try {
                Thread.sleep(1000);
                System.out.println("Thread 1 finished");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        Thread t2 = new Thread(() -> {
            try {
                Thread.sleep(2000);
                System.out.println("Thread 2 finished");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });

        t1.start();
        t2.start();

        t1.join();
        t2.join();

        System.out.println("All threads finished");
    }
}

在上面的例子中,主线程启动了t1和t2两个线程,然后调用了它们的join()方法,等待它们执行完成。当t1和t2两个线程都执行完成后,主线程才会继续执行。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/357018.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

前端借助Canvas实现压缩图片两种方法

一、具体代码 1、利用canvas压缩图片方法一 // 第一种压缩图片方法&#xff08;图片base64,图片类型,压缩比例,回调函数&#xff09;// 图片类型是指 image/png、image/jpeg、image/webp(仅Chrome支持)// 该方法对以上三种图片类型都适用 压缩结果的图片base64与原类型相同// …

Python自动化测试-使用Pandas来高效处理测试数据

Python自动化测试-使用Pandas来高效处理测试数据 目录&#xff1a;导读 一、思考 二、使用pandas来操作Excel文件 三、使用pandas来操作csv文件 四、总结 一、思考 1.Pandas是什么&#xff1f; 功能极其强大的数据分析库可以高效地操作各种数据集 csv格式的文件Excel文件H…

Python3-基本数据类型

Python3 基本数据类型 Python 中的变量不需要声明。每个变量在使用前都必须赋值&#xff0c;变量赋值以后该变量才会被创建。 在 Python 中&#xff0c;变量就是变量&#xff0c;它没有类型&#xff0c;我们所说的"类型"是变量所指的内存中对象的类型。 等号&…

注意力机制笔记——结合沐神和B站老弓up主

B站【大白话浅谈【注意力机制】】 聚类 是针对 样本, 注意力机制是针对样本相关性,来进行计算的 自注意力机制 指的是 query ,key,value都是同一个部分。 可以学到 类似的 短语 ,和 语义特征。如its 指代的对象。 评论区大佬 根据这篇论文《Effective Approaches to…

[ vulhub漏洞复现篇 ] Drupal 远程代码执行漏洞(CVE-2018-7602)

&#x1f36c; 博主介绍 &#x1f468;‍&#x1f393; 博主介绍&#xff1a;大家好&#xff0c;我是 _PowerShell &#xff0c;很高兴认识大家~ ✨主攻领域&#xff1a;【渗透领域】【数据通信】 【通讯安全】 【web安全】【面试分析】 &#x1f389;点赞➕评论➕收藏 养成习…

设计模式之各种设计模式总结与对比

目录1 目标2 定位3 一句话归纳设计原则4 G0F 23种设计模式简介5 设计模式使用频次总结6 —句话归纳设计模式7 设计模式之间的关联关系和对比1 目标 1、 简要分析GoF 23种设计模式和设计原则,做整体认知。 2、 剖析Spirng的编程思想&#xff0c;启发思维,为之后深入学习Spring…

利用React实现多个场景下的鼠标跟随框提示框

前言 鼠标跟随框的作用如下图所示&#xff0c;可以在前端页面上&#xff0c;为我们后续的鼠标操作进行提示说明&#xff0c;提升用户的体验。本文将通过多种方式去实现&#xff0c;从而满足不同场景下的需求。 实现原理 实现鼠标跟随框的原理很简单&#xff0c;就是监听鼠标在…

删除链表的倒数第N个节点

题目描述19. 删除链表的倒数第 N 个结点难度中等2410收藏分享切换为英文接收动态反馈给你一个链表&#xff0c;删除链表的倒数第 n 个结点&#xff0c;并且返回链表的头结点。示例 1&#xff1a;输入&#xff1a;head [1,2,3,4,5], n 2输出&#xff1a;[1,2,3,5]示例 2&#…

【ROS】Windows系统安装ROS体验

大家平时玩ROS都是在Ubuntu系统上&#xff0c;那Windows系统可以安装吗&#xff0c;答案是&#xff1a;可以的&#xff01;Windows为了发展自家的物联网生态&#xff0c;已经在Windows系统支持ROS了。 文章目录1.安装VS 20172.安装Chocolatey & Git3.安装ROS4.运行ROS例程1…

RabbitMQ核心内容:实战教程(java)

文章目录一、安装二、入门1.分类2.核心概念3.工作原理4.六大模式三、模式一&#xff1a;"Hello World!"1.依赖2.生产者代码3.消费者代码四、模式二&#xff1a;Work Queues1.工作原理2.工具类代码&#xff1a;连接工厂3.消费者代码4.生产者代码5.分发策略不公平分发预…

计算机网络第2章(物理层)学习笔记

❤ 作者主页&#xff1a;欢迎来到我的技术博客&#x1f60e; ❀ 个人介绍&#xff1a;大家好&#xff0c;本人热衷于Java后端开发&#xff0c;欢迎来交流学习哦&#xff01;(&#xffe3;▽&#xffe3;)~* &#x1f34a; 如果文章对您有帮助&#xff0c;记得关注、点赞、收藏、…

Metasploit 使用篇

文章目录前言一、msfconsole启动msfconsole命令分类核心命令模块命令作业命令资源脚本命令后台数据库命令二、使用案例更改提示和提示字符运行shell命令信息收集&#xff1a;HTTP头检测前言 理解了Meatasploit框架架构、原理之后&#xff0c;自然就很好理解它的使用逻辑 find…

springmvc java ssm药店库存进销存管理系统带前台

基于JSP技术、SSM框架、B/S机构、Mysql数据库设计并实现了龙康药店管理系统。系统主要包括药店简介管理、客户信息管理、药品信息管理、入库信息管理、出库信息管理、进货单管理等功能模块。其特点一是方便学习&#xff0c;方便快捷&#xff1b;二是有非常大的信息储存量&#…

Games101-202作业1

一. 将模型从模型空间变换到世界空间下 在这个作业下&#xff0c;我们主要进行旋转的变换。 二.视图变换 ,将相机移动到坐标原点&#xff0c;同时保证物体和相机进行同样的变换&#xff08;这样对形成的图像没有影响&#xff09; 在这个作业下我们主要进行摄像机的平移变换&am…

【深度学习编译器系列】1. 为什么需要深度学习编译器?

本系列是自学深度学习编译器过程中的一些笔记和总结&#xff0c;参考文献在文末。 1. 概述 深度学习&#xff08;DL&#xff09;编译器的产生有两方面的因素&#xff1a;深度学习模型的广泛应用&#xff0c;以及深度学习芯片的层出不穷。 一方面&#xff0c;我们现在有非常多…

剑指 Offer 34. 二叉树中和为某一值的路径(java解题)

剑指 Offer 34. 二叉树中和为某一值的路径&#xff08;java解题&#xff09;1. 题目2. 解题思路3. 数据类型功能函数总结4. java代码1. 题目 给你二叉树的根节点 root 和一个整数目标和 targetSum &#xff0c;找出所有 从根节点到叶子节点 路径总和等于给定目标和的路径。 叶…

关于MySQL的limit优化

1、前提 提示&#xff1a;只适用于InnoDB引擎 2、InnoDB存储特点 它把索引和数据放在了一个文件中&#xff0c;就是聚集索引。这与MyISAM引擎是不一样的。 3、SQL示例 -- 给cve字段建立索引 select * from cnnvd where cveCVE-2022-24808 limit 300000,10&#xff1b;由于M…

ACWING/1824. 钻石收藏家

输出格式 输出贝茜可以在展示柜中展示的钻石最大数量。 数据范围 1≤N≤1000 0≤K≤10000 钻石的尺寸范围 [1,10000]输入样例&#xff1a; 5 3 1 6 4 3 1输出样例&#xff1a; 4排序双指针 常规解法 将数据进行排序&#xff0c;形成一个有序单调增加的数组。然后左指针不…

【C语言每日一题】——猜凶手

【C语言每日一题】——猜名次&#x1f60e;前言&#x1f64c;猜凶手&#x1f64c;解题思路分享&#xff1a;&#x1f60d;解题源码分享&#xff1a;&#x1f60d;总结撒花&#x1f49e;&#x1f60e;博客昵称&#xff1a;博客小梦 &#x1f60a;最喜欢的座右铭&#xff1a;全神…

Spring中的数据校验

数据校验基础 参考&#xff1a; Java Bean Validation 规范 Spring对Bean Validation的支持 Spring定义了一个接口org.springframework.validation.Validator&#xff0c;用于应用相关的对象的校验器。 这个接口完全从基础设施或者上下文中脱离的&#xff0c;这意味着它没有…