当前位置: 首页 > news >正文

5.Java并发编程—JUC线程池架构

JUC线程池架构

在Java开发中,线程的创建和销毁对系统性能有一定的开销,需要JVM和操作系统的配合完成大量的工作。

  1. JVM对线程的创建和销毁:
    • 线程的创建需要JVM分配内存、初始化线程栈和线程上下文等资源,这些操作会带来一定的时间和内存开销。
    • JVM需要通过与操作系统的交互进行系统调用,涉及到资源分配、权限检查等操作。
    • 线程的销毁需要释放已分配的内存和其他系统资源,这也需要一定的开销。
  2. 操作系统对线程的管理:
    • 操作系统在内核层面管理线程,每个线程都需要占用操作系统的资源,包括内存、CPU时间片、线程调度等。
    • 创建和销毁线程涉及到操作系统的系统调用,如创建线程栈、设置线程上下文、更新线程调度信息等。
    • 过多的线程会增加操作系统的负担,导致资源竞争和上下文切换的开销增加。

为了减少这些开销,Java引入了线程池的概念。线程池可以预先创建一定数量的线程,并重用这些线程来处理任务,从而减少线程的创建和销毁频率,提高系统的性能和效率。

使用线程池的主要优势包括:

  • 减少线程创建和销毁的开销:线程池在应用启动时创建一定数量的线程,并将它们保存在池中,避免了频繁的创建和销毁操作。
  • 线程重用:线程池可以重用线程来执行多个任务,避免了反复创建线程的开销。
  • 动态调整线程数量:线程池可以根据任务负载情况动态调整线程数量,提高系统的处理能力和响应性能。
  • 管理和监控线程:线程池提供管理和监控线程的功能,可以设置线程的优先级、超时时间等,提供更好的线程控制和调优能力。

1.JUC线程池架构

在多线程编程中,任务都是一些经过抽象的工作单元,而线程就是让任务异步执行的基本机制。随着我们应用开发的扩张,线程和任务的管理也开始变得非常复杂,为了简化这些复杂的线程管理,这个时候就需要一个“管理者”,来统一管理线程及任务分配,这个就是线程池。

有关线程池接口 和 类的架构图大致如下

Executor结构-1

Executor(执行任务)

Executor是Java异步目标任务的执行者接口,其目的就是来执行目标任务。Executor通过execute()接口来执行已经提交的Runnable目标实例。其目的就是将任务执行者任务提交者分离开来。

Executor框架的主要目的是将任务的提交和执行进行解耦,将任务的创建和执行逻辑分离开来。通过使用Executor框架,可以将任务的提交和执行过程进行灵活管理,并提供了一些常用的线程池实现,简化了多线程编程的复杂性。

它只包含了一个方法

public interface Executor {/*** Executes the given command at some time in the future.  The command* may execute in a new thread, in a pooled thread, or in the calling* thread, at the discretion of the {@code Executor} implementation.** @param command the runnable task* @throws RejectedExecutionException if this task cannot be* accepted for execution* @throws NullPointerException if command is null*/void execute(Runnable command);
}

ExecutorService(提交任务)

ExecutorService继承Executor。他是Java中异步目标任务的执行者服务接口对外提供异步任务的接收服务。ExecutorService对外提供了接收异步任务并转发给执行者的方法,例如submit系列方法,invoke系列方法等。

// 向线程池提交单个任务
<T> Future<T> submit(Callable<T> task);
<T> Future<T> submit(Runnable task, T result);
Future<?> submit(Runnable task);// 批量向线程池提交任务 
<T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks) throws InterruptedException;
<T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks,long timeout, TimeUnit unit) throws InterruptedException;

AbstractExecutorService

AbstractExecutorService是Java中ExecutorService接口的一个抽象实现类。它提供了ExecutorService接口的部分默认实现,以便更方便地创建自定义的ExecutorService实现。

AbstractExecutorService实现了ExecutorService接口中的大部分方法,包括提交任务、关闭执行器、等待任务完成等。它还提供了一些辅助方法和模板方法,可以供子类进行扩展和定制。

以下是AbstractExecutorService中一些重要的方法和概念:

  1. submit(Runnable task)submit(Callable<T> task)方法:这些方法用于将任务提交给执行器并返回一个代表任务结果的Future对象。submit(Runnable task)方法接受Runnable任务,不返回结果;submit(Callable<T> task)方法接受Callable任务,返回一个Future对象,可以通过它获取任务的执行结果。
  2. shutdown()shutdownNow()方法:shutdown()方法用于平缓关闭执行器,不再接受新的任务提交,但会等待已提交的任务执行完成。shutdownNow()方法更加强制,它尝试停止执行器并取消所有未执行的任务。
    1. shutdown()方法:
      • shutdown()方法是一个平缓的关闭方法。调用该方法后,ExecutorService将停止接受新的任务提交,但会等待已提交的任务执行完成。
      • shutdown()方法不会中断正在执行的任务,而是会等待任务自行完成或等待超时。
      • 在调用shutdown()方法后,可以使用awaitTermination()方法来等待所有任务执行完成,或者使用isTerminated()方法来判断是否所有任务都已经执行完成。
    2. shutdownNow()方法:
      • shutdownNow()方法是一个强制关闭方法。调用该方法后,ExecutorService将尝试停止当前正在执行的任务,并取消所有未执行的任务。
      • shutdownNow()方法会中断正在执行的任务,即使任务正在阻塞中,它们也会收到InterruptedException
      • shutdownNow()方法返回一个List<Runnable>,其中包含所有未执行的任务。这样可以检查和处理未执行的任务。
  3. invokeAny(Collection<? extends Callable<T>> tasks)invokeAll(Collection<? extends Callable<T>> tasks)方法:这些方法用于提交一组任务,并等待其中一个或所有任务完成。invokeAny()方法返回其中一个任务的结果,invokeAll()方法返回所有任务的结果。
  4. isShutdown()isTerminated()方法:isShutdown()方法用于判断执行器是否已经关闭,isTerminated()方法用于判断是否所有任务都已经执行完成。

ThreadPoolExecutor

下面我们就来了解一下,大名鼎鼎的线程池实现类,ThreadPoolExecutor,它继承与AbstractExecutorService抽象类。ThreadPoolExecutor是JUC线程池的核心实现类,线程的创建和中止都需要很大的开销,线程池为我们提供了指定数量的可重用的线程,所以使用线程池可以很大程度上节省系统资源,并且每个线程池都维护了一些基础的数据统计,方便线程的管理和监控。

我们先来简单了解一下ThreadPoolExecutor的参数,后面会详细讲解每个参数具体的意义。

ThreadPoolExecutor的一些重要概念:

  1. 核心线程(Core Threads)核心线程是线程池中保持活动状态的最小线程数。即使线程是空闲的,也会保持这个数量的线程。核心线程在线程池的整个生命周期中都会存在,除非显式地调用了allowCoreThreadTimeOut()方法来允许核心线程超时退出。
  2. 最大线程数(Maximum Threads)最大线程数指定了线程池中允许的最大线程数。当任务数量超过核心线程数且工作队列已满时,线程池会创建新的线程,直到达到最大线程数。超过最大线程数的任务将被拒绝执行,默认情况下会抛出RejectedExecutionException异常。
  3. 工作队列(Work Queue)工作队列用于存储还未执行的任务。当线程池的线程都处于忙碌状态时,新提交的任务会被放入工作队列中等待执行。ThreadPoolExecutor提供了不同的工作队列实现,例如ArrayBlockingQueue、LinkedBlockingQueue、SynchronousQueue等。
  4. 拒绝策略(Rejected Execution Policy)当线程池无法接受新的任务时(例如达到最大线程数并且工作队列已满),拒绝策略定义了如何处理这些被拒绝的任务。ThreadPoolExecutor提供了几种预定义的拒绝策略,如AbortPolicy、CallerRunsPolicy、DiscardPolicy和DiscardOldestPolicy,同时还支持自定义拒绝策略。
  5. 线程池生命周期(ThreadPool Lifecycle):线程池存在着不同的生命周期阶段,包括初始化运行中关闭。线程池的生命周期可以通过调用shutdown()shutdownNow()方法进行关闭。在关闭线程池之前,可以使用isShutdown()方法检查线程池是否已经关闭,使用isTerminated()方法判断是否所有任务都已经执行完成。
CallerRunsPolicy

CallerRunsPolicy是Java线程池中的一个饱和策略(RejectedExecutionHandler)。当线程池无法接受新的任务时,该策略会将任务交给调用线程来执行。

当线程池中的工作队列已满且无法继续接受新的任务时,饱和策略会决定如何处理这些被拒绝的任务。CallerRunsPolicy是其中一种饱和策略,它的行为是将被拒绝的任务交给调用线程来执行,而不是在新的线程中执行。

CallerRunsPolicy的特点如下:

  1. 任务在调用线程中执行:当线程池无法接受新的任务时,被拒绝的任务会由调用线程直接执行,而不是在新的线程中执行。这意味着任务的执行会阻塞调用线程,可能导致调用线程的执行时间延长。
  2. 保证任务不会丢失:CallerRunsPolicy保证被拒绝的任务不会丢失。尽管任务可能无法在期望的时间内执行,但它们最终会被执行,从而确保任务不会被丢弃。
  3. 适用于一些特定场景:CallerRunsPolicy适用于一些需要保证任务执行顺序或依赖调用线程的场景。当线程池的工作队列已满时,将任务交给调用线程执行可以确保任务的顺序性和正确性。
AbortPolicy

AbortPolicy是Java线程池中的一个饱和策略(RejectedExecutionHandler)。当线程池无法接受新的任务时,该策略会抛出RejectedExecutionException异常,拒绝执行新的任务。

当线程池中的工作队列已满且无法继续接受新的任务时,饱和策略会决定如何处理这些被拒绝的任务。AbortPolicy是其中一种饱和策略,它的行为是直接抛出异常,不执行被拒绝的任务。

AbortPolicy的特点如下:

  1. 抛出异常:当线程池无法接受新的任务时,被拒绝的任务会导致RejectedExecutionException异常被抛出。这意味着无法处理的任务会被拒绝执行,并且调用方可以捕获该异常进行处理。
  2. 不保证任务的执行:AbortPolicy不保证被拒绝的任务一定不会执行。由于线程池已经饱和且无法接受新的任务,被拒绝的任务可能无法执行,并且不会进入工作队列等待执行。
  3. 适用于需要严格控制任务提交的场景:AbortPolicy适用于一些需要严格控制任务提交的场景。当线程池无法接受新的任务时,抛出异常可以通知调用方任务无法执行,从而确保任务提交的可控性。
DiscardPolicy

DiscardPolicy是Java线程池中的一个饱和策略(RejectedExecutionHandler)。当线程池无法接受新的任务时,该策略会默默地丢弃被拒绝的任务,不做任何处理。

当线程池中的工作队列已满且无法继续接受新的任务时,饱和策略会决定如何处理这些被拒绝的任务。DiscardPolicy是其中一种饱和策略,它的行为是直接丢弃被拒绝的任务,不执行也不抛出异常。

DiscardPolicy的特点如下:

  1. 默默丢弃任务:当线程池无法接受新的任务时,被拒绝的任务会被默默地丢弃,不做任何处理。这意味着无法处理的任务会被直接丢弃,不会被执行。
  2. 不保证任务的执行:DiscardPolicy不保证被拒绝的任务一定不会执行。由于线程池已经饱和且无法接受新的任务,被拒绝的任务会被直接丢弃,不会进入工作队列等待执行。
  3. 适用于忽略任务执行的场景:DiscardPolicy适用于一些对任务执行无关紧要的场景。当任务被拒绝执行时,丢弃任务可以忽略任务执行,从而避免对整个系统产生影响。
DiscardOldestPolicy

DiscardOldestPolicy是Java线程池中的一个饱和策略(RejectedExecutionHandler)。当线程池无法接受新的任务时,该策略会丢弃工作队列中最旧的任务,为新的任务腾出空间,并尝试将新任务添加到工作队列中。

当线程池中的工作队列已满且无法继续接受新的任务时,饱和策略会决定如何处理这些被拒绝的任务。DiscardOldestPolicy是其中一种饱和策略,它的行为是丢弃工作队列中最旧的任务,并尝试将新任务添加到工作队列中。

DiscardOldestPolicy的特点如下:

  1. 丢弃最旧的任务:当线程池无法接受新的任务时,被拒绝的任务会导致工作队列中最旧的任务被丢弃。这意味着最旧的任务会被抛弃,为新的任务腾出空间。
  2. 尝试添加新任务:DiscardOldestPolicy会尝试将新的任务添加到工作队列中。如果添加成功,则新任务可以被执行;如果添加失败(例如工作队列已满),则新任务也会被丢弃。
  3. 不保证任务的执行:DiscardOldestPolicy不保证被拒绝的任务一定不会执行。由于线程池已经饱和且无法接受新的任务,被拒绝的任务可能无法执行,并且不会进入工作队列等待执行。
ThreadFactory

ThreadFactory是Java中的一个接口,用于创建线程对象。它提供了一种自定义线程创建方式的机制,允许开发者在创建线程时进行一些额外的配置或处理。

ThreadFactory接口定义了一个方法newThread,该方法接收一个Runnable对象作为参数,并返回一个新创建的Thread对象。开发者可以根据自己的需求,在newThread方法中实现自定义的线程创建逻辑。

下面是ThreadFactory接口的定义:

public interface ThreadFactory {Thread newThread(Runnable r);
}

ThreadFactory接口只有一个方法newThread,它接收一个Runnable参数,并返回一个Thread对象。在实现ThreadFactory接口时,开发者需要提供自己的线程创建逻辑,例如设置线程的名称、优先级、异常处理等。

Worker

Worker指的是执行任务的工作线程。线程池是一种用于管理和复用线程的机制,它通过预先创建一组工作线程来执行提交的任务,从而提供了一种有效地处理并发任务的方式。

Worker是线程池中的核心组成部分,它负责从任务队列中获取任务并执行。线程池中的每个Worker都是一个独立的工作线程,可以并发地执行任务。

Worker的主要职责包括:

  1. 从任务队列中获取任务:Worker会不断地从任务队列中获取待执行的任务。任务队列可以是一个阻塞队列,当队列为空时,Worker会等待直到有新的任务被提交到队列中。
  2. 执行任务:一旦Worker获取到任务,它会执行任务的具体逻辑。任务可以是一个Runnable对象或Callable对象,根据任务的类型进行执行。
  3. 处理异常:Worker负责捕获和处理任务执行过程中可能抛出的异常。它可以根据需要记录异常信息、进行错误处理或通知相关方。
  4. 生命周期管理:Worker的生命周期由线程池进行管理。线程池可以动态地创建、销毁和管理Worker线程,以适应不同的工作负载。

通过使用Worker线程,线程池可以提供以下好处:

  1. 重用线程:Worker线程可以被线程池重用,避免了频繁地创建和销毁线程的开销。
  2. 并发执行:线程池中的多个Worker线程可以并发地执行多个任务,提高任务处理的效率。
  3. 控制并发度:线程池可以限制并发执行的任务数量,防止系统资源被过度占用。
  4. 提供任务排队和调度:线程池通过任务队列来管理待执行的任务,可以灵活地控制任务的排队和调度策略。

ScheduledExecutorService

ScheduledExecutorService是Java中的一个接口,继承自ExecutorService接口,用于在预定的时间间隔内执行任务。它提供了一种方便的方式来调度任务的执行,并且可以支持延迟执行和周期性执行。

ScheduledExecutorService的主要特点如下:

  1. 任务调度:ScheduledExecutorService可以安排任务在指定的时间点执行,或者延迟一定时间后执行。它提供了一些方法,如schedule()scheduleAtFixedRate()scheduleWithFixedDelay(),用于安排任务的执行时间。
  2. 延迟执行:ScheduledExecutorService可以延迟任务的执行。通过schedule()方法,可以指定任务的延迟时间,任务将在延迟时间过后执行一次。
  3. 周期性执行:ScheduledExecutorService支持周期性执行任务。通过scheduleAtFixedRate()方法,可以指定任务的初始延迟时间和执行周期,任务将在指定的延迟时间过后开始执行,并在每个周期结束后立即重新执行。
  4. 线程池支持:ScheduledExecutorService通常使用线程池来执行任务。它可以使用线程池中的线程来执行任务,以便高效地管理和复用线程资源。线程池可以通过Executors类的方法创建,例如newScheduledThreadPool()
  5. 可取消任务:ScheduledExecutorService允许取消已安排的任务。通过返回的ScheduledFuture对象,可以使用cancel()方法取消任务的执行。取消的任务将不再执行,已经在执行的任务可以被中断。

通过使用ScheduledExecutorService,可以实现定时任务、周期性任务和延迟任务的调度。它提供了灵活的任务调度功能,使得在应用程序中执行定时或延迟任务变得简单和可靠。

请注意,ScheduledExecutorService是一个接口,它的具体实现类是ScheduledThreadPoolExecutor,它是ThreadPoolExecutor的扩展,提供了调度任务的功能。

ScheduledThreadPoolExecutor

ScheduledThreadPoolExecutor是Java中的一个类,它是ScheduledExecutorService接口的具体实现类。它继承自ThreadPoolExecutor类,并增加了调度任务的功能。

ScheduledThreadPoolExecutor具有以下特点:

  1. 任务调度:ScheduledThreadPoolExecutor可以安排任务在指定的时间点执行,或者延迟一定时间后执行。它提供了一些方法,如schedule()scheduleAtFixedRate()scheduleWithFixedDelay(),用于安排任务的执行时间。
  2. 线程池支持:ScheduledThreadPoolExecutor使用线程池来执行任务。它继承自ThreadPoolExecutor类,因此可以充分利用线程池的功能,包括线程的复用、线程池大小的控制和线程的生命周期管理。
  3. 优雅关闭:ScheduledThreadPoolExecutor可以通过调用shutdown()shutdownNow()方法来关闭线程池。在关闭之前,它会等待所有已安排的任务执行完成,并停止接受新的任务。可以使用isShutdown()方法检查线程池是否已经关闭,使用isTerminated()方法判断是否所有任务都已经执行完成。
  4. 灵活的任务调度策略:ScheduledThreadPoolExecutor提供了多种任务调度策略。例如,scheduleAtFixedRate()方法可以按照固定的速率执行任务,无论任务的执行时间是否超过周期;scheduleWithFixedDelay()方法可以在任务执行完成后,等待固定的延迟时间再执行下一个任务。
  5. 可取消任务:ScheduledThreadPoolExecutor允许取消已安排的任务。通过返回的ScheduledFuture对象,可以使用cancel()方法取消任务的执行。取消的任务将不再执行,已经在执行的任务可以被中断。

使用ScheduledThreadPoolExecutor可以轻松地实现定时任务、周期性任务和延迟任务的调度。它提供了灵活的任务调度功能,并且通过线程池的管理,能够高效地执行任务。

请注意,ScheduledThreadPoolExecutor是ScheduledExecutorService接口的默认实现,它提供了最常用的功能和调度策略。如果需要更高级的功能,也可以通过自定义实现ScheduledExecutorService接口来实现。

DelayedWorkQueue

DelayedWorkQueue是Java中的一个工作队列实现,用于存储延迟执行的任务。它是ScheduledThreadPoolExecutor类中使用的默认工作队列。

DelayedWorkQueue的特点如下:

  1. 存储延迟任务:DelayedWorkQueue用于存储延迟执行的任务。每个任务都有一个延迟时间,任务将在延迟时间过后才能被执行。延迟时间可以是固定的时间段,也可以是任务提交的时间点到执行时间点的时间间隔。
  2. 按照延迟时间排序:DelayedWorkQueue按照任务的延迟时间进行排序。队列中的任务按照延迟时间从小到大排列,即延迟时间最小的任务排在队列的前面,最先被取出执行。这样可以确保任务按照预定的延迟时间顺序执行。
  3. 基于优先级的比较:DelayedWorkQueue使用任务的延迟时间和优先级进行比较。在延迟时间相等的情况下,优先级高的任务会被排在前面,优先执行。这样可以确保在延迟时间相等的情况下,具有高优先级的任务先被执行。

ScheduledFutureTask

ScheduledFutureTask是Java中的一个类,它是ScheduledThreadPoolExecutor类中使用的任务实现。它继承自FutureTask类,并实现了ScheduledFuture接口。

ScheduledFutureTask的特点如下:

  1. 延迟执行和周期性执行:ScheduledFutureTask用于表示延迟执行和周期性执行的任务。它可以安排任务在指定的延迟时间后执行一次,也可以按照固定的周期执行任务。
  2. 可取消任务:ScheduledFutureTask允许取消已安排的任务。通过调用cancel()方法,可以取消任务的执行。已取消的任务将不再执行,已经在执行的任务可以被中断。
  3. 任务调度信息:ScheduledFutureTask包含任务的调度信息,如延迟时间、周期和任务的执行时间等。它提供了一些方法,如getDelay()getPeriod(),用于获取任务的延迟时间和周期。
  4. 任务执行状态:ScheduledFutureTask提供了任务执行状态的管理。它可以通过isDone()方法判断任务是否已完成,通过isCancelled()方法判断任务是否已取消。

相关文章:

5.Java并发编程—JUC线程池架构

JUC线程池架构 在Java开发中&#xff0c;线程的创建和销毁对系统性能有一定的开销&#xff0c;需要JVM和操作系统的配合完成大量的工作。 JVM对线程的创建和销毁&#xff1a; 线程的创建需要JVM分配内存、初始化线程栈和线程上下文等资源&#xff0c;这些操作会带来一定的时间和…...

llama2c(4)之forward、sample、decode

1、forward float* logits forward(transformer, token, pos); 输入transformer的参数&#xff0c;当前token&#xff0c;pos位置&#xff0c;预测出下一个token的预测值&#xff08;用矩阵乘&#xff0c;加减乘除等运算构成Transformer&#xff09; 其中&#xff0c;logits如…...

20240312-2-贪心算法

贪心算法 是每次只考虑当前最优&#xff0c;目标证明每次是考虑当前最优能够达到局部最优&#xff0c;这就是贪心的思想&#xff0c;一般情况下贪心和排序一起出现&#xff0c;都是先根据条件进行排序&#xff0c;之后基于贪心策略得到最优结果。 面试的时候面试官一般不会出贪…...

前端 --- HTML

1. HTML 结构 1.1 HTML 文件基本结构 <html><head><title>第一个html程序</title></head><body>hello world!</body> </html> html 标签是整个 html 文件的根标签(最顶层标签)head 标签中写页面的属性.body 标签中写的是页…...

curl c++ 实现HTTP GET和POST请求

环境配置 curl //DV2020T环境下此步骤可省略 https://curl.se/download/ 笔者安装为7.85.0版本 ./configure --without-ssl make sudo make install sudo rm /usr/local/lib/curl 系统也有curl库&#xff0c;为防止冲突&#xff0c;删去编译好的curl库。 对以json数据的解析使…...

12、设计模式之代理模式(Proxy)

一、什么是代理模式 代理模式属于结构型设计模式。为其他对象提供一种代理以控制对这个对象的访问。 在某些情况下&#xff0c;一个对象不适合或者不能直接引用另一个对象&#xff0c;而代理对象可以在客户端和目标对象之间起到中介的作用。 二、分类 代理模式分为三类&#…...

springboot集成Quartz定时任务组件

文章目录 前言一、Quartz 是什么&#xff1f;下面是对 Java 中 Quartz 的主要概念的简单描述&#xff1a; 二、使用步骤总结 前言 平时开发中相信大家都经常用到定时任务吧&#xff0c;最近简单的就是直接使用Scheduled注解标注到方法上用注解的方式在项目运行时无法去对任务进…...

代码随想录算法训练营第38天—动态规划06 | ● 完全背包 ● *518. 零钱兑换 II ● 377. 组合总和 Ⅳ

完全背包 视频讲解&#xff1a;https://www.bilibili.com/video/BV1uK411o7c9 https://programmercarl.com/%E8%83%8C%E5%8C%85%E9%97%AE%E9%A2%98%E7%90%86%E8%AE%BA%E5%9F%BA%E7%A1%80%E5%AE%8C%E5%85%A8%E8%83%8C%E5%8C%85.html 题目描述&#xff1a;有n件物品和一个最多能…...

C语言每日一题(63)复写零

题目链接 力扣网 1089 复写零 题目描述 给你一个长度固定的整数数组 arr &#xff0c;请你将该数组中出现的每个零都复写一遍&#xff0c;并将其余的元素向右平移。 注意&#xff1a;请不要在超过该数组长度的位置写入元素。请对输入的数组 就地 进行上述修改&#xff0c;不…...

ElasticSearch聚合查询

数据准备 索引创建 PUT product {"mappings": {"properties": {"createtime": {"type": "date"},"desc": {"type": "text","fields": {"keyword": {"type": …...

【毕设级项目】基于AI技术的多功能消防机器人(完整工程资料源码)

基于AI技术的多功能消防机器人演示效果 竞赛-基于AI技术的多功能消防机器人视频演示 前言&#xff1a; 随着“自动化、智能化”成为数字时代发展的关键词&#xff0c;机器人逐步成为社会经济发展的重要主体之一&#xff0c;“机器换人”成为发展的全新趋势和时代潮流。在可预见…...

【一】【设计模式】类关系UML图

1. 继承&#xff08;Generalization&#xff09; 继承是对象间的一种层次关系&#xff0c;允许子类继承并扩展父类的功能。 UML线&#xff1a;带有空心箭头的直线&#xff0c;箭头指向基类&#xff08;父类&#xff09;。 class Parent {public void parentMethod() {System.…...

【DevOps基础篇】容器化架构基础设施监控方案

【DevOps基础篇】容器化架构基础设施监控方案 目录 【DevOps基础篇】容器化架构基础设施监控方案要监视什么不同监控系统方案比较1. Datadog2. Prometheus3. ELK(Elasticsearch、Logstash、Kibana)4. Sysdig5. 自行打造!如何选择总结推荐超级课程: Docker快速入门到精通 当…...

【QT】文件流操作(QTextStream/QDataStream)

文本流/数据流&#xff08;二级制格式&#xff09; 文本流 &#xff08;依赖平台&#xff0c;不同平台可能乱码&#xff09;涉及文件编码 #include <QTextStream>操作的都是基础数据类型&#xff1a;int float string //Image Qpoint QRect就不可以操作 需要下面的 …...

CentOS 7 devtoolset编译addressSanitizer版本失败的问题解决

在我的一个Cent OS7开发环境中&#xff0c;按https://yeyongjin.blog.csdn.net/article/details/134178420的方法升级GCC版本到8.3.1。 这两天&#xff0c;要用Google的addressSanitizer检验内存问题&#xff0c;加上编译参数后&#xff0c;却发现编译不通过。configure时直接退…...

ubuntu2004桌面系统英伟达显卡驱动安装方法

#如何查看显卡型号 lspci | grep -i vga#----output------ 01:00.0 VGA compatible controller: NVIDIA Corporation Device 1f06 (rev a1)根据 Device 后的 值 进入网站查询 pci-ids.ucw.cz/mods/PC/10de?actionhelp?helppci #根据显卡型号&#xff0c;下载对应系统的驱动…...

Java通过Excel批量上传数据!!!

一、首先在前端写一个上传功能。 <template><!-- 文件上传 --><el-upload class"upload-demo" drag action"" :on-change"onChange" :auto-upload"false"><el-icon class"el-icon--upload"><up…...

【PyQT/Pysider】控件背景渐变

默认渐变配色说明 background-color: qlineargradient(spread:pad, x1:0, y1:0, x2:1, y2:0, stop:0 rgba(255, 178, 102, 255), stop:0.55 rgba(235, 148, 61, 255), stop:0.98 rgba(0, 0, 0, 255), stop:1 rgba(0, 0, 0, 0));这段样式表使用了qlineargradient函数来创建…...

ChatGPT-4 VS 文心一言4.0

在线体验 地址&#xff08;含 gpt 3.5 / 4.0&#xff0c;文心 3.5 / 4.0&#xff09;&#xff1a;https://chat.tool4j.com 点击访问 文心一言和ChatGPT-4都是非常强大的自然语言处理模型&#xff0c;它们都能够在对话系统和其他NLP应用中发挥巨大的作用。然而&#xff0c;它们…...

MYSQL------从概述到DQL

数据库&#xff08;数据管理&#xff0c;数据存储的仓库&#xff09; 数据库管理系统&#xff08;操纵和管理数据库的大型软件&#xff09; SQL是操作关系型的编程语言&#xff0c;是一套标准 MySQL下载安装完成以后&#xff0c;可以进行启动和停止操作&#xff0c;对于启动和停…...

Cursor实现用excel数据填充word模版的方法

cursor主页&#xff1a;https://www.cursor.com/ 任务目标&#xff1a;把excel格式的数据里的单元格&#xff0c;按照某一个固定模版填充到word中 文章目录 注意事项逐步生成程序1. 确定格式2. 调试程序 注意事项 直接给一个excel文件和最终呈现的word文件的示例&#xff0c;…...

linux之kylin系统nginx的安装

一、nginx的作用 1.可做高性能的web服务器 直接处理静态资源&#xff08;HTML/CSS/图片等&#xff09;&#xff0c;响应速度远超传统服务器类似apache支持高并发连接 2.反向代理服务器 隐藏后端服务器IP地址&#xff0c;提高安全性 3.负载均衡服务器 支持多种策略分发流量…...

C++_核心编程_多态案例二-制作饮品

#include <iostream> #include <string> using namespace std;/*制作饮品的大致流程为&#xff1a;煮水 - 冲泡 - 倒入杯中 - 加入辅料 利用多态技术实现本案例&#xff0c;提供抽象制作饮品基类&#xff0c;提供子类制作咖啡和茶叶*//*基类*/ class AbstractDr…...

利用ngx_stream_return_module构建简易 TCP/UDP 响应网关

一、模块概述 ngx_stream_return_module 提供了一个极简的指令&#xff1a; return <value>;在收到客户端连接后&#xff0c;立即将 <value> 写回并关闭连接。<value> 支持内嵌文本和内置变量&#xff08;如 $time_iso8601、$remote_addr 等&#xff09;&a…...

K8S认证|CKS题库+答案| 11. AppArmor

目录 11. AppArmor 免费获取并激活 CKA_v1.31_模拟系统 题目 开始操作&#xff1a; 1&#xff09;、切换集群 2&#xff09;、切换节点 3&#xff09;、切换到 apparmor 的目录 4&#xff09;、执行 apparmor 策略模块 5&#xff09;、修改 pod 文件 6&#xff09;、…...

【JavaEE】-- HTTP

1. HTTP是什么&#xff1f; HTTP&#xff08;全称为"超文本传输协议"&#xff09;是一种应用非常广泛的应用层协议&#xff0c;HTTP是基于TCP协议的一种应用层协议。 应用层协议&#xff1a;是计算机网络协议栈中最高层的协议&#xff0c;它定义了运行在不同主机上…...

(二)TensorRT-LLM | 模型导出(v0.20.0rc3)

0. 概述 上一节 对安装和使用有个基本介绍。根据这个 issue 的描述&#xff0c;后续 TensorRT-LLM 团队可能更专注于更新和维护 pytorch backend。但 tensorrt backend 作为先前一直开发的工作&#xff0c;其中包含了大量可以学习的地方。本文主要看看它导出模型的部分&#x…...

前端导出带有合并单元格的列表

// 导出async function exportExcel(fileName "共识调整.xlsx") {// 所有数据const exportData await getAllMainData();// 表头内容let fitstTitleList [];const secondTitleList [];allColumns.value.forEach(column > {if (!column.children) {fitstTitleL…...

【SQL学习笔记1】增删改查+多表连接全解析(内附SQL免费在线练习工具)

可以使用Sqliteviz这个网站免费编写sql语句&#xff0c;它能够让用户直接在浏览器内练习SQL的语法&#xff0c;不需要安装任何软件。 链接如下&#xff1a; sqliteviz 注意&#xff1a; 在转写SQL语法时&#xff0c;关键字之间有一个特定的顺序&#xff0c;这个顺序会影响到…...

ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放

简介 前面两期文章我们介绍了I2S的读取和写入&#xff0c;一个是通过INMP441麦克风模块采集音频&#xff0c;一个是通过PCM5102A模块播放音频&#xff0c;那如果我们将两者结合起来&#xff0c;将麦克风采集到的音频通过PCM5102A播放&#xff0c;是不是就可以做一个扩音器了呢…...