多线程多进程

news2025/1/23 13:16:19

秋招面试的java八股文知识点补充以及iot

这里有一点阅读补充

线程和进程区别

什么是进程?

进程 (Process) 是计算机中的一个独立执行单元,是操作系统资源分配的基本单位。每个进程有各自独立的内存空间和资源,它们之间相互独立,相互之间不能访问彼此的内存空间和寄存器等资源。

什么是线程?

线程 (Thread) 是指进程中的一个执行单元,也被称为轻量级进程。一个进程可以包含多个线程,它们共享进程的内存和资源·线程拥有自己的栈和程序计数器,但共享进程内的数据和资源。

Linux中进程线程相关指令

ps 命令用于显示当前正在运行的进程的信息。
ps aux | grep <进程名>:查看包含指定进程名的所有进程信息。
ps -eLf:显示所有线程的详细信息。
ps -T <进程ID>:显示指定进程ID下的所有线程。

netstat -tuln 显示当前正在使用的TCP和UDP端口。
lsof -i :80 查看80端口上的进程信息

top:实时显示系统的运行情况,包括进程和线程信息,可以按 H 键来切换显示线程或进程。

进程上下文切换

进程上下文切换是操作系统将CPU从一个进程切换到另一个进程的过程
简要来说,它包括以下步骤
1.中断当前进程。
2.保存当前进程的上下文 (如程序计数器、CPU寄存器)
3.根据调度策略选择新进程
4更新新进程的进程控制块(PCB)
5.恢复新进程的上下文。
6.开始执行新进程。
进程的上下文切换非常昂贵,它涉及到需要保存很多的上下文资源

何时使用多进程和多线程?

多进程:

对资源保护和管理要求高,不限制开销和效率时可以使用多进程。
每个进程有独立的内存空间,相互之间不会影响,更安全,适合于需要并发执行但需要隔离资源的任务。

多线程:

要求效率高,切换频繁时使用多线程。
多线程共享内存空间,因此线程之间的通信和切换更加快速,适合于需要频繁切换和共享数据的任务。

进程间通信方式有哪些?

文件: 进程可以通过读写共享文件来进行通信。一个进程写入文件,另一个进程读取文件。

Socket通信: 基于网络编程的Socket通信是一种进程间通信的常见方式。可以通过套接字实现进程间的数据传输,包括本地通信和远程通信。

管道(Pipes): java.io.PipedInputStream 和 java.io.PipedOutputStream 可以用于在两个相关的Java进程之间进行通信。一个进程通过输出流写入数据,另一个进程通过输入流读取数据。

Java RMI(Remote Method Invocation): Java RMI 允许一个Java虚拟机中的对象调用另一个Java虚拟机中的对象的方法,实现了远程通信。

消息队列(Message Queue): 使用消息队列工具,比如 RabbitMQ 或 Apache Kafka,可以在不同Java进程之间传递消息,实现进程间通信。

共享内存: Java的共享内存通常通过使用 Java 的并发工具,例如 java.util.concurrent 包中的类,实现多线程间的共享数据。

进程间通信的选择:

管道通信:用于进程间互发短小、频率很高的消息。
共享内存:用于进程间共享数据量庞大、读写频繁的数据。
其他考虑套接字(Socket)。

线程间通信方法有哪些?

临界区(Critical Section): 通过在代码中标记临界区域,保证在任何给定时间内只有一个线程能够进入该区域,从而确保对共享资源的互斥访问。

互斥量(Mutex): 使用互斥量实现互斥访问,一个线程进入互斥区时,其他线程必须等待。一旦线程离开互斥区,其他线程可以获得对共享资源的访问权。

信号量(Semaphore): 通过信号量控制对共享资源的访问。信号量维护一个计数器,当资源被占用时,计数器减少;当资源释放时,计数器增加。线程在访问资源前必须获取信号量,如果计数器为0,线程将等待。

事件(Event): 通过事件对象进行通信。一个线程可以等待事件的发生,而其他线程负责发出事件信号。当事件发生时,等待的线程被唤醒。

条件变量(Condition Variable): 用于在线程之间传递信息,通常与互斥量一起使用。一个线程等待条件变量的某个条件为真,而其他线程则可以通过触发条件变量来通知等待的线程条件已满足。

读写锁(Read-Write Lock): 用于共享资源的读写控制。多个线程可以同时获取共享资源的读取权限,但只有一个线程能够获得写入权限。这可以提高读取操作的并发性。

操作系统

创建进程的方式

系统初始化,例如后台进程、守护进程。
一个进程开启另一个进程,使用 fork() 函数。
用户的交互式请求,例如在命令行中启动一个新的进程。

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>

int main() {
    pid_t pid;

    // 创建一个新的进程
    pid = fork();

    if (pid < 0) {
        // 错误处理
        fprintf(stderr, "Fork failed\n");
        return 1;
    } else if (pid == 0) {
        // 子进程执行的代码
        printf("This is the child process (PID: %d)\n", getpid());
    } else {
        // 父进程执行的代码
        printf("This is the parent process (PID: %d)\n", getpid());
    }

    // 公共代码,父子进程都会执行
    printf("This is common code (PID: %d)\n", getpid());

    return 0;
}

进程有几种状态?

进程有五种状态:创建、就绪、运行、阻塞、终止。在Linux系统中,进程的生命周期是从创建到终止。

什么是内核线程和用户线程?

用户线程:由用户进行管理,创建、调度、同步和销毁由用户空间的库函数完成,不需要内核帮助,开销较小。
内核线程:由操作系统创建和销毁,完全由内核管理。

如何实现守护进程?

创建子进程,终止父进程。
调用 setsid() 创建一个新的会话。
将当前目录更改为根目录。
重设文件权限掩码。
关闭不再需要的文件描述符。

线程里面的java函数

继承 Thread 类或实现 Runnable 接口来创建线程

可以通过继承 Thread 类或实现 Runnable 接口来创建线程。

继承 Thread 类和实现 Runnable 接口的区别:继承 Thread 类只能创建一个线程,而实现 Runnable 接口可以创建多个线程。

start() 方法启动一个线程

可以通过调用 start() 方法来启动一个线程。

class MyThread extends Thread {
    public void run() {
        // 线程执行的任务
        for (int i = 0; i < 5; i++) {
            System.out.println(Thread.currentThread().getId() + " Value " + i);
        }
    }
}

public class Main {
    public static void main(String args[]) {
        MyThread t1 = new MyThread();
        t1.start(); // 启动线程

        MyThread t2 = new MyThread();
        t2.start(); // 启动另一个线程
    }
}

sleep()和wait()的区别

sleep()是 Thread类的方法,wait() 是 Object类的方法。
sleep() 不释放锁,wait() 会释放锁。
wait() 必须在同步方法或同步代码块中执行。

sleep() 方法暂停线程的执行

try {
    Thread.sleep(1000);
} catch (InterruptedException e) {
    e.printStackTrace();
}

wait() 方法等待其他线程的通知

try {
    obj.wait();
} catch (InterruptedException e) {
    e.printStackTrace();
}

yield() 方法让出CPU资源

Thread.yield();

join() 方法等待线程执行完毕

try {
    thread.join();
} catch (InterruptedException e) {
    e.printStackTrace();
}

notify() 方法通知其他线程

obj.notify();

notifyAll() 方法通知所有等待的线程

obj.notifyAll();

synchronized 关键字同步代码块

synchronized (obj) {
    // 同步代码块
}

synchronized 关键字同步方法

public synchronized void method() {
    // 同步方法
}

volatile 关键字保证变量的可见性

private volatile int count;

AtomicInteger 类保证变量的原子性

private AtomicInteger count = new AtomicInteger();

CountDownLatch 类等待多个线程完成

CountDownLatch latch = new CountDownLatch(2);
Thread thread1 = new Thread(() -> {
    // 线程1执行的代码
    latch.countDown();
});
Thread thread2 = new Thread(() -> {
    // 线程2执行的代码
    latch.countDown();
});
thread1.start();
thread2.start();
latch.await();

Semaphore 类控制并发线程数

Semaphore semaphore = new Semaphore(1);
Thread thread1 = new Thread(() -> {
    try {
        semaphore.acquire();
        // 线程1执行的代码
        semaphore.release();
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
});
Thread thread2 = new Thread(() -> {
    try {
        semaphore.acquire();
        // 线程2执行的代码
        semaphore.release();
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
});
thread1.start();
thread2.start();

ReentrantLock 类实现同步

private ReentrantLock lock = new ReentrantLock();

public void method() {
    lock.lock();
    try {
        // 同步代码块
    } finally {
        lock.unlock();
    }
}

ReadWriteLock 类实现读写分离

private ReadWriteLock lock = new ReentrantReadWriteLock();

public void readMethod() {
    lock.readLock().lock();
    try {
        // 读操作
    } finally {
        lock.readLock().unlock();
    }
}

public void writeMethod() {
    lock.writeLock().lock();
    try {
        // 写操作
    } finally {
        lock.writeLock().unlock();
    }
}

ThreadLocal 类实现线程本地变量

private ThreadLocal<Integer> count = new ThreadLocal<>();

public void method() {
    count.set(1);
    // 访问 count 变量
    count.remove();
}

Executor 框架管理线程池

ExecutorService executor = Executors.newFixedThreadPool(2);

executor.execute(() -> {
    // 线程1执行的代码
});

executor.execute(() -> {
    // 线程2执行的代码
});

executor.shutdown();

程序是怎么运行的

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

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

相关文章

【保姆级】Protobuf详解及入门指南

目录 Protobuf概述 什么是Protobuf 为什么要使用Protobuf Protobuf实战 环境配置 创建文件 解析/封装数据 附录 AQin.proto 完整代码 Protobuf概述 什么是Protobuf Protobuf&#xff08;Protocol Buffers&#xff09;协议&#x1f609; Protobuf 是一种由 Google 开…

flink的分组聚合、over聚合、窗口聚合对比

【背景】 flink有几种聚合&#xff0c;使用上是有一些不同&#xff0c;需要加以区分&#xff1a; 分组聚合&#xff1a;group agg over聚合&#xff1a;over agg 窗口聚合&#xff1a;window agg 省流版&#xff1a; 触发计算时机 结果流类型 状态大小 分组聚合group ag…

【RabbitMQ】RabbitMQ的交换机

交换机类型 在上文中&#xff0c;都没有交换机&#xff0c;生产者直接发送消息到队列。而一旦引入交换机&#xff0c;消息发送的模式会有很大变化&#xff1a;可以看到&#xff0c;在订阅模型中&#xff0c;多了一个exchange角色&#xff0c;而且过程略有变化&#xff1a; Pub…

ThreadLocal出现内存泄露原因分析

ThreadLocal 导致内存泄漏的主要原因是它的工作方式。在 Java 中&#xff0c;ThreadLocal 通过维护一个以 Thread 为键&#xff0c;以用户设置的值为值的映射来工作。每个线程都拥有其自身的线程局部变量副本&#xff0c;不同线程间的这些变量互不干扰。这个映射是存储在每个 T…

EpiCypher—SMARCAL1介导的肿瘤免疫逃避

资深作者Alberto Ciccia博士(左)和第一作者Giuseppe Leuzzi博士。 在新的博客系列中&#xff0c;我们将对话最近发表染色质研究工作的首席科学家们&#xff0c;了解他们的幕后工作。在EpiCypher&#xff0c;我们很感兴趣染色质技术是如何整合到不同领域的&#xff0c;如免疫学、…

【Ubuntu】gonme桌面的 gdm 和 lightdm 区别

总结&#xff1a;都可以 gdm: 【Gnome Display Manager】 完整&#xff0c;体积大 lightdm: 【Light Display Manager】 轻量

HBase非关系型数据库

HBase非关系型数据库 1 什么是HBase2 HBase的特点3 什么时候需要HBase4 HBase的数据模型5 HBase架构5.1 架构5.2 HBase如何列式储存 6 如何正确设计RowKey 1 什么是HBase HBase – Hadoop Database&#xff0c;是一个高可靠性、高性能、面向列、可伸缩、 实时读写的分布式数据…

趣味看图-Linux 文件系统的组成

/&#xff08;根目录&#xff09;&#xff1a;根目录是Linux文件系统中的顶级目录。所有其他目录都是根目录的子目录&#xff0c;使其成为整个文件系统的父目录。 /bin&#xff1a;包含启动系统和执行基本操作所需的基本二进制可执行文件。这些对所有用户都可用。 /boot&…

手撕栈和队列

接下来的日子会顺顺利利&#xff0c;万事胜意&#xff0c;生活明朗-----------林辞忧 引言 栈和队列作为数据结构的重要组成部分&#xff0c;可以用栈实现非递归等&#xff0c;为后面学习打基础。栈由数组来实现&#xff0c;队列由链表来实现&#xff0c;接下来将详细介绍 …

C++实现引用计数(二)

实现引用计数 引言实现集成开发环境项目结构实现代码运行结果 注意 引言 C中经常使用智能指针来管理内存。对于共享指针shared_ptr的原理&#xff1a;每当有一个指针指向这块内存&#xff0c;引用计数的值加一&#xff0c;每当一个指针不再指向这块内存&#xff0c;引用计数的…

【OpenGL手册14】投光物

目录 一、说明二、平行光三、点光源四、衰减五、选择正确的值六、实现衰减七、聚光八、手电筒九、平滑/软化边缘练习 一、说明 我们目前使用的光照都来自于空间中的一个点。它能给我们不错的效果&#xff0c;但现实世界中&#xff0c;我们有很多种类的光照&#xff0c;每种的表…

员工私单亡羊补牢!这个监管神器让你从此放心!

对于企业而言&#xff0c;如何有效监管员工的微信使用成为了一项重要的任务。幸运的是&#xff0c;现在有一个监管神器——微信管理系统&#xff0c;可以帮助企业从根本上解决这个问题。 接下来&#xff0c;让我们一起来看看这个监管神器究竟有哪些神奇的功能吧&#xff01; …

5款好用的AI办公软件,一键轻松制作PPT、视频,提升工作效率!

众所周知&#xff0c;AI 人工智能技术已渗透到生活的方方面面&#xff0c;无论是很多人早已用上的智能音箱、语音助手&#xff0c;还是新近诞生的各种 AI 软件工具&#xff0c;背后都离不开 AI 人工智能技术的加持。 对于各类新生的 AI 软件工具&#xff0c;人们很容易「选边站…

亚马逊多账号怎么防关联?超级浏览器来帮你!

很多做亚马逊跨境电商的小伙伴都会遇到的问题就是多登店铺账号被关联&#xff0c;我们要知道&#xff0c;如果在亚马逊上运营多个店铺&#xff0c;保持账户之间的独立性是很重要的。一旦账户之间被平台识别为关联&#xff0c;不仅可能导致收入损失&#xff0c;还可能面临账号被…

短视频,文案素材哪里找?找短视频文案素材指南

有很多创业者&#xff0c;耗费了很长时间创作出了很多短视频&#xff0c;文案素材没写好&#xff0c;导致浏览量下降&#xff0c;或者写文案的同时就已经花费了很长时间&#xff0c;导致没有时间发布视频&#xff0c;从而影响了流量&#xff0c;导致流量下滑&#xff0c;其实我…

React-嵌套路由

1.概念 说明&#xff1a;在一级路由中又内嵌了其他路由&#xff0c;这种关系就叫做嵌套路由&#xff0c;嵌套至一级路由内的路由又称作二级路由。 2.实现步骤 说明&#xff1a;使用childen属性配置路由嵌套关系&#xff0c;使用<Outlet/>组件配置二级路由渲染的位置。…

RK3588-hdmiin

1. HDMI-IN简介 HDMI IN功能可以通过桥接芯⽚的⽅式实现&#xff0c;将HDMI信号转换成MIPI信号接收RK3588芯⽚平台⾃带HDMI RX模块&#xff0c;可以直接接收HDMI信号&#xff0c;无需通过桥接芯⽚实现。在ArmSoM系列产品中&#xff0c;ArmSoM-W3支持HDMI-IN功能HDMI-IN功能框图…

容灾演练双月报|美创DRCC助力银行高效验证数据库高可用架构

了解更多灾备行业动态 守护数字化时代业务连续 目录 CONTENTS 01 灾备法规政策 02 热点安全事件 03 容灾演练典型案例 01 灾备法规政策 2月&#xff0c;工信部印发《工业领域数据安全能力提升实施方案&#xff08;2024—2026年&#xff09;》&#xff0c;要求到2026年…

Java语言的演变之路:从过去到现在,以及未来的展望与探索

引言 Java&#xff0c;这个在全球范围内广受欢迎的编程语言&#xff0c;自其诞生之日起&#xff0c;就注定要在计算机科学的历史上留下浓墨重彩的一笔。起源于20世纪90年代&#xff0c;Java的初衷是打造一种能够在多种平台上运行的、安全且易于使用的编程语言。它不仅成功地实…

MySQL-QA-异常问题及解决方案(持续更新)

MySQL-Q&A(持续更新) 1.1 PID文件找不到 问题描述 错误详情&#xff1a; ERROR&#xff01;The server quit without updating PID file (/usr/local/mysql/data/localhost.localdomain.pid) 解决方案 首先排查配置文件&#xff0c;一般路径为&#xff1a;/etc/my.cnf 检查…