FutureTask详解
目录
- FutureTask详解
- 1、FutureTask简介
- 2、FutureTask内部结构
- 继承结构
- 类属性
- 构造方法
- 内部类`WaitNode`
- 3、`Runnable`、`Callable`、`Future`、`RunnableFuture`接口
- ①、`Runnable`接口
- ②、`Callable`接口
- ③、`Future`接口
- ④、`RunnableFuture`接口
- 总结对比
- 4、FutureTask的使用示例
- 普通Thread使用
- 线程池使用
- 5、FutureTask的核心方法
- `run()`方法
- `get()`方法
- `cancel()`方法
- 6、补充知识点`Treiber` 栈
- Treiber 栈的核心思想
- Treiber 栈的特点
- `Treiber` 栈的Java简易实现
- `Treiber` 栈在`FutureTask`中的应用
- 7、补充FutureTask子类ScheduledFutureTask
- 总结
FutureTask详解
1、FutureTask简介
FutureTask主要用于异步任务的执行和结果获取。其最重要的特性就是可以被提交到线程池中执行,同时也可以用来获取执行结果或检查任务的状态。
2、FutureTask内部结构
继承结构
public class FutureTask<V> implements RunnableFuture<V> {}

FutureTask 实现了 RunnableFuture 接口,而 RunnableFuture 又继承 Runnable 和 Future 。
所以FutureTask 可以被提交到线程池中执行,同时也可以用来获取执行结果或检查任务的状态。
类属性
// FutureTask 的状态字段,用于标记任务的不同状态
private volatile int state; // volatile 确保线程间对该字段的可见性
private static final int NEW = 0; // 任务刚创建,尚未开始执行
private static final int COMPLETING = 1; // 任务正在完成(即运行中或正在设置结果)
private static final int NORMAL = 2; // 任务正常完成
private static final int EXCEPTIONAL = 3; // 任务完成时抛出了异常
private static final int CANCELLED = 4; // 任务已被取消
private static final int INTERRUPTING = 5; // 任务中断中
private static final int INTERRUPTED = 6; // 任务已中断/** 包含要执行的 callable 对象;在运行后会被置为 null */
private Callable<V> callable;/** 任务的结果或者在执行时抛出的异常,使用 Object 类型来存储结果或异常 */
private Object outcome; // non-volatile, 由 state 字段的读写保护/** 正在运行任务的线程;在 run() 方法中使用 CAS 操作 */
private volatile Thread runner;/** 用于存储等待任务完成的线程的 Treiber 栈 */
// 可以理解成是一个 无锁且线程安全的栈
private volatile WaitNode waiters;
构造方法
- ①、
FutureTask(Callable<V> callable)
public FutureTask(Callable<V> callable) {if (callable == null) // 检查 callable 是否为空throw new NullPointerException(); // 如果为空,抛出空指针异常this.callable = callable; // 保存 callable 对象this.state = NEW; // 初始化状态为 NEW,表示任务刚创建
}
- ②、
FutureTask(Runnable runnable, V result)
public FutureTask(Runnable runnable, V result) {// 使用 Executors.callable() 方法将 Runnable 转换为 Callable,并指定结果this.callable = Executors.callable(runnable, result);this.state = NEW; // 初始化状态为 NEW,表示任务刚创建
}
Executors.callable方法
public static <T> Callable<T> callable(Runnable task, T result) {if (task == null)throw new NullPointerException();return new RunnableAdapter<T>(task, result);}// 利用RunnableAdapter 适配器类 把Runnable 转换成 Callable
static final class RunnableAdapter<T> implements Callable<T> {final Runnable task;final T result;RunnableAdapter(Runnable task, T result) {this.task = task;this.result = result;}public T call() {task.run();return result;}}
内部类WaitNode
WaitNode 是 FutureTask 类中的一个静态内部类,用于支持等待任务完成的线程管理。
WaitNode 主要作为一个节点,在 waiters 栈中构建链表,以便能够高效地管理等待线程。
static final class WaitNode {// 用于存储等待的线程volatile Thread thread;// 用于指向下一个等待节点,形成链表结构volatile WaitNode next;// 构造函数,初始化当前节点的线程为当前线程WaitNode() { thread = Thread.currentThread(); }
}
3、Runnable、Callable、Future、RunnableFuture接口
①、Runnable接口
用于实现线程要执行的任务。可以将 Runnable 对象传递给 Thread 构造函数或线程池的 execute() 方法。
@FunctionalInterface
public interface Runnable {void run();
}
Runnable 是一个函数式接口,定义了一个 run() 方法。
run() 方法没有返回值,也没有抛出检查型异常(checked exceptions)。
主要用于线程执行的任务,无需处理任务结果。
②、Callable接口
用于执行有返回值的任务。通常与 Future 配合使用,可以在任务完成后获取结果。
@FunctionalInterface
public interface Callable<V> {V call() throws Exception;
}
Callable 是一个泛型接口,定义了一个 call() 方法。
call() 方法可以返回一个结果,并且可以抛出检查型异常。
泛型参数 <V> 表示任务执行后的结果类型。
③、Future接口
用于管理异步计算任务,检查任务状态,获取计算结果或处理异常。
public interface Future<V> {// 尝试取消任务boolean cancel(boolean mayInterruptIfRunning);// 检查任务是否被取消boolean isCancelled();// 检查任务是否完成boolean isDone();// 获取任务的结果,若任务尚未完成则阻塞调用get方法的线程V get() throws InterruptedException, ExecutionException;// 带超时的获取结果方法V get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException;
}
④、RunnableFuture接口
RunnableFuture 主要用于 FutureTask 类的实现。它允许任务以 Runnable 方式提交到线程池中执行,并通过 Future 接口管理结果和状态。
public interface RunnableFuture<V> extends Runnable, Future<V> {
}
RunnableFuture 接口继承了 Runnable 和 Future 接口。
结合了这两个接口的功能,即可以像 Runnable 一样执行任务,也可以像 Future 一样获取任务的结果或取消任务。
总结对比
| 接口 | 功能描述 | 主要方法/特点 | 适用场景 |
|---|---|---|---|
Runnable | 定义要执行的任务,无返回值 | run() | 用于执行不需要结果的任务,如线程的任务 |
Callable | 定义要执行的任务,有返回值,并可能抛出异常 | call() | 用于执行有返回值的任务 |
Future | 管理异步计算的结果,检查状态和处理异常 | get(), cancel(), isDone(), isCancelled() | 用于管理异步任务的结果和状态 |
RunnableFuture | 结合 Runnable 和 Future 的功能,可以执行任务并管理结果 | 继承 Runnable 和 Future | 用于需要同时支持 Runnable 和 Future 功能的场景,如 FutureTask |
4、FutureTask的使用示例
普通Thread使用
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;public class TestA {public static void main(String[] args) {// 创建一个 Callable 对象Callable<Integer> callable = () -> {Thread.sleep(2000); // 模拟长时间运行的任务return 123;};// 用 Callable 对象创建 FutureTaskFutureTask<Integer> futureTask = new FutureTask<>(callable);// 用普通 Thread 执行 FutureTaskThread thread = new Thread(futureTask);thread.start();// 执行其他任务System.out.println("主线程继续执行其他任务...");try {// 获取 FutureTask 的结果Integer result = futureTask.get(); // 这个方法会阻塞直到结果可用System.out.println("获取 FutureTask 的结果: " + result);} catch (InterruptedException | ExecutionException e) {e.printStackTrace();}}
}
执行结果:
主线程继续执行其他任务...
获取 FutureTask 的结果: 123
线程池使用
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;public class TestA {private static final ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(4, // 核心线程数 8, // 最大线程数0, // 非核心线程的存活时间TimeUnit.SECONDS, // 存活时间单位new LinkedBlockingDeque<>(), // 存放任务的 阻塞队列new MyThreadFactory("DogEatBones"), // 创建线程的 工厂 new ThreadPoolExecutor.AbortPolicy() // 线程池的拒绝策略);public static void main(String[] args) throws Exception {String dog1 = "秀逗";// 使用线程池执行Future<String> future1 = threadPoolExecutor.submit(() -> eat(dog1));String result = future1.get();System.out.println("FutureTask执行结果:" + result);// 关闭线程池threadPoolExecutor.shutdown();}public static String eat(String name) {try {TimeUnit.SECONDS.sleep(1);} catch (InterruptedException e) {e.printStackTrace();}return name + "说骨头真好吃!";}}class MyThreadFactory implements ThreadFactory {private final String name;private final AtomicInteger threadNum = new AtomicInteger();// 设置线程名称public MyThreadFactory(String name) {this.name = name;}@Overridepublic Thread newThread(Runnable r) {Thread thread = new Thread(r);// 设置线程名称和编号thread.setName(name + " [#" + threadNum.incrementAndGet() + "]");return thread;}
}
执行结果:
FutureTask执行结果:秀逗说骨头真好吃!
5、FutureTask的核心方法
下面源码基于JDK8
run()方法
public void run() {// 检查任务状态是否为 NEW,且当前线程是否能成功设置为 runnerif (state != NEW ||!UNSAFE.compareAndSwapObject(this, runnerOffset,null, Thread.currentThread()))return;try {Callable<V> c = callable;// 如果 callable 不为 null 且状态仍为 NEW,则执行 callableif (c != null && state == NEW) {V result;boolean ran;try {result = c.call(); // 调用 callable,并存储结果ran = true;} catch (Throwable ex) {result = null;ran = false;setException(ex); // 如果发生异常,设置异常}if (ran)set(result); // 如果 callable 成功执行,设置结果}} finally {// 确保 runner 在 run() 完成后设为 null,以防止多次调用runner = null;// 在将 runner 设为 null 后重新读取状态,以处理可能的中断int s = state;if (s >= INTERRUPTING)handlePossibleCancellationInterrupt(s);}
}
protected void set(V v) {// 使用 CAS 原子性地将状态从 NEW 更新为 COMPLETINGif (UNSAFE.compareAndSwapInt(this, stateOffset, NEW, COMPLETING)) {outcome = v; // 存储结果UNSAFE.putOrderedInt(this, stateOffset, NORMAL); // 设置为最终状态finishCompletion(); // 完成任务}
}
private void finishCompletion() {// 遍历所有等待线程for (WaitNode q; (q = waiters) != null;) {if (UNSAFE.compareAndSwapObject(this, waitersOffset, q, null)) {for (;;) {Thread t = q.thread;if (t != null) {q.thread = null;LockSupport.unpark(t); // 唤醒等待的线程}WaitNode next = q.next;if (next == null)break;q.next = null; // 解除链表节点连接,帮助垃圾回收q = next;}break;}}done(); // 任务完成后执行的操作callable = null; // 释放 callable 对象以减少内存占用
}
总结:
run() 方法: 该方法用于执行任务。
如果任务的状态是 NEW,runner 为空,则将当前线程设置为 runner 并执行 callable。
如果 callable.call() 执行成功,则将结果设置到 FutureTask 中;
如果执行失败,则捕获异常并设置异常。
无论任务成功还是失败,最后都要将 runner 设为 null,并根据状态处理可能的中断。
set(V v) 方法: 该方法用于设置任务的结果。
如果任务的状态是 NEW,则将状态更新为 COMPLETING 并存储结果。然后将状态设置为 NORMAL 表示任务完成,并调用 finishCompletion() 方法完成任务的后续处理。
finishCompletion() 方法: 该方法用于处理任务完成后的操作。首先唤醒所有等待的线程,然后执行 done() 方法以进行额外的完成操作。最后将 callable 设为 null 以减少内存占用。
get()方法
public V get() throws InterruptedException, ExecutionException {int s = state;// 如果状态小于等于 COMPLETING,等待任务完成if (s <= COMPLETING)s = awaitDone(false, 0L);return report(s); // 根据状态报告结果
}
private int awaitDone(boolean timed, long nanos) throws InterruptedException {final long deadline = timed ? System.nanoTime() + nanos : 0L;WaitNode q = null;boolean queued = false;for (;;) {// 如果当前线程被中断,移除等待者并抛出 InterruptedExceptionif (Thread.interrupted()) {removeWaiter(q);throw new InterruptedException();}int s = state;// 如果任务已经完成,返回状态if (s > COMPLETING) {if (q != null)q.thread = null;return s;}// 如果任务正在完成中,当前线程主动让出 CPUelse if (s == COMPLETING)Thread.yield();else if (q == null)q = new WaitNode(); // 创建新的等待节点else if (!queued)queued = UNSAFE.compareAndSwapObject(this, waitersOffset,q.next = waiters, q); // 将当前等待节点加入Treiber栈else if (timed) {nanos = deadline - System.nanoTime();if (nanos <= 0L) {removeWaiter(q); // 超时,移除等待节点并返回状态return state;}LockSupport.parkNanos(this, nanos); // 线程等待指定纳秒时间}elseLockSupport.park(this); // 无超时,线程等待}
}
private void removeWaiter(WaitNode node) {if (node != null) {node.thread = null; // 清除等待节点的线程引用retry:for (;;) { // 当存在竞争条件时,重新尝试移除等待节点for (WaitNode pred = null, q = waiters, s; q != null; q = s) {s = q.next; // 获取当前节点的下一个节点if (q.thread != null)pred = q; // 记录当前节点作为前驱节点else if (pred != null) {pred.next = s; // 如果前驱节点存在且当前节点没有线程,则将前驱节点的 next 指向当前节点的下一个节点if (pred.thread == null) // 检查前驱节点是否也没有线程,可能存在竞争条件continue retry; // 如果存在竞争条件,重新尝试}else if (!UNSAFE.compareAndSwapObject(this, waitersOffset,q, s)) // 如果没有前驱节点,则通过 CAS 将等待队列的头节点更新为下一个节点continue retry; // 如果 CAS 失败,重新尝试}break; // 成功移除等待节点后跳出循环}}
}
static final class WaitNode {volatile Thread thread; // 线程引用volatile WaitNode next; // 下一个等待节点WaitNode() { thread = Thread.currentThread(); } // 构造函数中将当前线程设置为该节点的线程
}
private V report(int s) throws ExecutionException {Object x = outcome;if (s == NORMAL)return (V)x; // 如果状态为 NORMAL,返回结果if (s >= CANCELLED)throw new CancellationException(); // 如果状态为 CANCELLED,抛出取消异常throw new ExecutionException((Throwable)x); // 否则,抛出执行异常
}
总结:
get() 方法: 用于获取任务的结果。如果任务的状态小于等于 COMPLETING,则调用 awaitDone() 方法等待任务完成。最后,调用 report() 方法根据任务的最终状态报告结果或抛出异常。
awaitDone() 方法: 用于等待任务完成。通过检查任务状态并在必要时将当前线程挂起来等待。如果设置了超时,会在超时之前进行等待;如果没有超时,则无限等待。处理竞争条件时,确保等待节点正确地从等待队列中移除。
removeWaiter() 方法: 用于从等待队列中移除指定的等待节点。通过遍历等待队列,调整节点的连接,并处理竞争条件,确保等待节点被正确移除。
WaitNode 类: 用于表示一个等待的节点。包含对线程的引用和指向下一个等待节点的引用。
report() 方法: 根据任务的最终状态,返回结果或抛出异常。如果任务状态为 NORMAL,返回结果;如果状态为 CANCELLED,抛出 CancellationException;否则,抛出 ExecutionException。
cancel()方法
public boolean cancel(boolean mayInterruptIfRunning) {// 如果当前状态是 NEW,且使用 CAS 原子性地将状态设置为 INTERRUPTING 或 CANCELLED,成功则返回 trueif (!(state == NEW &&UNSAFE.compareAndSwapInt(this, stateOffset, NEW,mayInterruptIfRunning ? INTERRUPTING : CANCELLED)))return false;try { // 如果中断当前线程的调用抛出异常,将在 finally 中处理if (mayInterruptIfRunning) {try {Thread t = runner;if (t != null)t.interrupt(); // 如果任务正在运行,尝试中断线程} finally { // 设置最终状态UNSAFE.putOrderedInt(this, stateOffset, INTERRUPTED);}}} finally {finishCompletion(); // 完成任务的清理工作}return true; // 成功取消任务
}
总结:
使用 UNSAFE.compareAndSwapInt 原子性地将任务状态从 NEW 更新为 INTERRUPTING 或 CANCELLED。如果状态更新成功,返回 true;否则,返回 false,表示取消失败(例如任务已经开始执行或已完成)。
如果 mayInterruptIfRunning 为 true,尝试中断正在运行的线程。如果 runner 不为空,则中断该线程。
在 finally 块中,将任务状态设置为 INTERRUPTED,表示任务被中断。
调用 finishCompletion() 方法,完成任务的清理工作,移除等待线程,释放资源等。
如果成功取消任务,返回 true;否则,返回 false。
6、补充知识点Treiber 栈
Treiber 栈是一种并发栈的实现方式,旨在解决多线程环境下的栈操作问题。它由计算机科学家 Robert Treiber 提出,主要用于实现无锁数据结构。
Treiber 栈的核心思想
无锁数据结构: Treiber 栈使用原子操作而不是传统的锁机制来保证线程安全,允许多个线程并发访问而不会阻塞。
基于链表: Treiber 栈通常使用链表实现,每个节点包含数据和指向下一个节点的引用。
原子操作: 使用 compare-and-swap (CAS) 操作来更新栈的头部指针,确保操作的原子性。
Treiber 栈的特点
并发友好: Treiber 栈允许多个线程同时执行 push 和 pop 操作,而不需要传统的锁机制,因此提高了并发性能。
简洁性: 其实现相对简单,主要依赖于 CAS 操作来保证线程安全。
非阻塞: 由于使用了原子操作,Treiber 栈是非阻塞的,避免了锁带来的上下文切换和线程阻塞。
Treiber 栈的Java简易实现
import java.util.concurrent.atomic.AtomicReference;public class TestA<T> {public static void main(String[] args) {SimpleTreiberStack<String> stack = new SimpleTreiberStack<>();// 测试 push 和 popThread t1 = new Thread(() -> {stack.push("秀逗");}, "t1");Thread t2 = new Thread(() -> {stack.push("四眼");}, "t2");Thread t3 = new Thread(() -> {stack.push("大黄");}, "t3");t1.start();t2.start();t3.start();try {t1.join();t2.join();t3.join();} catch (InterruptedException e) {e.printStackTrace();}System.out.println(stack.pop());System.out.println(stack.pop());System.out.println(stack.pop());}
}class SimpleTreiberStack<T> {private final AtomicReference<Node<T>> top = new AtomicReference<>(null);private static class Node<T> {T value;Node<T> next;Node(T value) {this.value = value;this.next = null;}}public void push(T value) {Node<T> newNode = new Node<>(value);Node<T> currentTop;do {currentTop = top.get();newNode.next = currentTop; // 将新节点的 next 指向当前栈顶} while (!top.compareAndSet(currentTop, newNode));}public T pop() {Node<T> oldTop, newTop;do {oldTop = top.get();if (oldTop == null) return null; // 栈为空newTop = oldTop.next;} while (!top.compareAndSet(oldTop, newTop));return oldTop.value;}}
Treiber 栈在FutureTask中的应用
FutureTask在其内部定义了内部类WaitNode
static final class WaitNode {// 用于存储等待的线程volatile Thread thread;// 用于指向下一个等待节点,形成链表结构volatile WaitNode next;// 构造函数,初始化当前节点的线程为当前线程WaitNode() { thread = Thread.currentThread(); }
}
并定义了全局变量 private volatile WaitNode waiters;
waiters指向一个 Treiber 栈,该栈保存着所有等待任务执行结果的线程。
当调用FutureTask的get方法时,如果任务没有完成,则调用线程会被阻塞,本质上就是将要阻塞的线程包装成WaitNode结点保存到waiters指向的 Treiber 栈中。
在上面第5节 get()方法的源码分析中:
下面这段代码就相当于Treiber栈的push操作
else if (q == null)q = new WaitNode(); // 创建新的等待节点else if (!queued)queued = UNSAFE.compareAndSwapObject(this, waitersOffset,q.next = waiters, q); // 将当前等待节点加入Treiber栈
下面这段代码就相当于Treiber栈的pop操作
// 该方法在上面 在上面第5节 `get()`方法的源码分析中已经详细说明 这里不再赘述
removeWaiter(WaitNode node) {...}
最后推荐看下 Java线程池原理剖析和应用指南中对于Future 模式的讲解。
7、补充FutureTask子类ScheduledFutureTask
ScheduledFutureTask 是 ScheduledThreadPoolExecutor 中的一个内部类,它继承自 FutureTask 并实现了 RunnableScheduledFuture 接口。这个类的主要作用是将任务包装成一个可以定时调度的任务 。
private class ScheduledFutureTask<V>extends FutureTask<V> implements RunnableScheduledFuture<V>{}

总结:
继承 FutureTask:
ScheduledFutureTask 继承自 FutureTask,因此它具有 FutureTask 的所有功能,如任务的提交、执行、获取结果等。
实现 RunnableScheduledFuture:
RunnableScheduledFuture 是 Runnable、ScheduledFuture 和 Delayed 的一个组合接口。它结合了这些接口的功能,使得 ScheduledFutureTask 不仅可以作为一个 Runnable 任务执行,还能支持调度和延迟操作。
ScheduledFutureTask 在 ScheduledThreadPoolExecutor 中的应用
在 ScheduledThreadPoolExecutor 中,调度相关的方法(如 schedule 和 scheduleAtFixedRate)会将任务包装成 ScheduledFutureTask 对象。
ScheduledFutureTask的构造方法:
// 构造一个一次性任务,指定任务的执行时间(以纳秒为单位)。
ScheduledFutureTask(Runnable r, V result, long ns) {super(r, result); // 调用父类 FutureTask 的构造函数,传入 Runnable 和结果this.time = ns; // 设置任务的触发时间(纳秒)this.period = 0; // 单次任务,所以周期为0this.sequenceNumber = sequencer.getAndIncrement(); // 设置任务的序列号,用于排序
}/*** 创建一个周期性任务,指定任务的执行时间和周期时间(以纳秒为单位)。*/
ScheduledFutureTask(Runnable r, V result, long ns, long period) {super(r, result); // 调用父类 FutureTask 的构造函数,传入 Runnable 和结果this.time = ns; // 设置任务的初始触发时间(纳秒)this.period = period; // 设置任务的周期时间(纳秒),用于周期性任务this.sequenceNumber = sequencer.getAndIncrement(); // 设置任务的序列号,用于排序
}/*** 创建一个一次性任务,指定任务的触发时间(以纳秒为单位),任务是 Callable 类型。*/
ScheduledFutureTask(Callable<V> callable, long ns) {super(callable); // 调用父类 FutureTask 的构造函数,传入 Callablethis.time = ns; // 设置任务的触发时间(纳秒)this.period = 0; // 单次任务,所以周期为0this.sequenceNumber = sequencer.getAndIncrement(); // 设置任务的序列号,用于排序
}
ScheduledFutureTask关键特性
延迟和调度:
通过实现 Delayed 接口,ScheduledFutureTask 可以管理任务的延迟和调度逻辑。这使得任务可以在指定的时间后执行或周期性执行。
与 ScheduledThreadPoolExecutor 的集成:
在 ScheduledThreadPoolExecutor 中,任务被封装成 ScheduledFutureTask 对象,并根据调度策略插入到队列中。调度器会按照指定的时间或周期触发任务执行。
总结
FutureTask: 主要用于执行异步任务,提供任务的提交、执行、结果获取和取消功能。
ScheduledFutureTask: 在 FutureTask 的基础上扩展,支持定时和周期性任务调度,适用于需要在特定时间或周期内执行的任务。
相关文章:
FutureTask详解
目录 FutureTask详解1、FutureTask简介2、FutureTask内部结构继承结构类属性构造方法内部类WaitNode 3、Runnable、Callable、Future、RunnableFuture接口①、Runnable接口②、Callable接口③、Future接口④、RunnableFuture接口总结对比 4、FutureTask的使用示例普通Thread使用…...
javase综合案例4 -- 考试系统
文章目录 一,项目要求二,创建实体类ExamItem三,创建考试服务类ExamService3.1 全局变量 考题列表itemList(List< ExamItem >类型),答案数组answerArr (String[]类型),得分score3.2 初始化方法init()3.3 打印菜单…...
Logistic回归
Logistic回归模型: 适用于二分类或多分类问题,样本特征是数值型(否则需要转换为数值型) 策略:极大似然估计 算法:随机梯度 或 BFGS算法(改进的拟牛顿法) 线性回归表达式…...
Langchain-Chatchat+Xinference集成部署
Langchain-ChatchatXinference集成部署 安装环境: 系统:Anolis OS 8.9 python版本:Python 3.9.19 Langchain-Chatchat版本:0.3.1.3 Xinference版本:v0.13.3 模型选择(下载时需要科学上网)&#…...
江协科技51单片机学习- p33 PWM呼吸灯和直流驱动电机调速
🚀write in front🚀 🔎大家好,我是黄桃罐头,希望你看完之后,能对你有所帮助,不足请指正!共同学习交流 🎁欢迎各位→点赞👍 收藏⭐️ 留言📝…...
使用Jetbrains.Rider反编译Unity的DLL文件看源码
直接将dll文件的打开方式用Rider打开即可,打开BattleSeqGenertor.dll文件的效果如下:...
【学习笔记】决策单调性优化DP
背景 GDCPC还在发力,清华出题组出的牛客还是 4 题。 这次没有min25筛,不然我能5题(bushi 除了一道用 prufer 序列的恶心 DP 外,还有一道DP题是一个状态难想,并且还需要决策单调性优化的DP,被认为是偏简单…...
【每日一题】【二分图最大匹配】【经典板子题】有大家喜欢的零食吗 河南萌新联赛2024第(一)场:河南农业大学 C题 C++
河南萌新联赛2024第(一)场:河南农业大学 C题 有大家喜欢的零食吗 题目描述 在某幼儿园中共有 n n n个小朋友,该幼儿园的老师为这 n n n 个小朋友准备了 n n n 份不一样的零食大礼包。每个小朋友只能选择一个,但老…...
【python】OpenCV—Image Colorization
文章目录 1、CIELAB 色彩空间2、作色问题定义3、Caffe 模型4、代码实现——Image5、代码实现——Video6、参考 1、CIELAB 色彩空间 Lab颜色空间,也称为Lab色彩空间或CIELAB色彩空间,是一种基于人类视觉感知特性的颜色模型。它是在1931年国际照明委员会&…...
vue 学习笔记
模板语法 1. 插值语法 用于解析标签体内容 { { 表达式 } } ,可以直接读取到 data 中的所有属性 2. 指令语法 解析标签(标签属性, 标签内容, 绑定事件) v-bind : href " url " 或 : href &…...
武汉流星汇聚:‘中国制造’闪耀欧洲站,体育赛事成亚马逊增长点
随着2024年的欧洲体育赛事激情四溢,欧洲杯与奥运会的双重盛会不仅点燃了全球体育迷的热情,更为亚马逊欧洲站带来了前所未有的发展机遇。在这场体育盛宴的推动下,欧洲站正展现出其无限的发展潜力和广阔的市场前景,为中国卖家乃至全…...
RPA是什么?探讨RPA发展的最新趋势 | RPA研究
随着人工智能和自动化技术的飞速发展,机器人流程自动化(Robotic Process Automation,简称RPA)正逐渐成为企业数字化转型的关键工具。RPA通过模拟人类用户的操作行为,自动化执行重复性高、规则性强的任务,从…...
sqlalchemy时间范围查询
1、sqlalchemy时间范围查询 在 SQLAlchemy 中,进行时间范围查询可以通过比较日期或时间字段来实现。假设你有一个模型 Event,它包含一个 timestamp 字段,你想查询在某个时间范围内的所有事件。以下是如何使用 SQLAlchemy 来实现这个查询的示例。 首先,确保你有 SQLAlchem…...
电脑不小心删除的文件怎么恢复?教你文件恢复的绝招
在日常使用电脑的过程中,我们有时会因为误操作或不小心而删除了重要的文件。面对这种情况,很多人可能会感到焦虑和无助。但其实,通过一些专业的方法和工具,我们有可能恢复这些被误删的文件。本文将介绍两种常见的恢复方法…...
stm32:使用和学习--硬件和程序
一硬件 1. GPIO 1.FT, TT功能 ft:five tolerate tt:three tolerate 1. FT(Five-Volt Tolerant)引脚 FT 引脚能够容忍高于 VDD 的输入电压(例如 5V)。这些引脚通常不具有连接到 VDD 的保护二极管&…...
ARM知识点二
一、指令 指令的生成过程 指令执行过程示例 if (a 0) {x 0; } else {x x 3; } //翻译为 cmp r0,#0 MOVEQ R1,#0 ADDGT R1,R1,#3指令获取:从Flash中读取 CMP R0, #0,控制器开始执行。 指令解码:解码器解析 CMP 指令,ALU比较R…...
C# ?的使用
栏目总目录 可空类型标记符(?) 说明: 可空类型标记符?用于指示某个值类型(如int、float等)可以为null。这是C# 2.0引入的一个特性,用于处理数据库查询、JSON解析等场景中可能出现的空值。 示例代码&am…...
【unity小技巧】unity性能优化以及如何进行性能测试
文章目录 前言GPU性能优化打包素材 CPU性能优化代码执行优化 性能测试Vector2.Distance 和 sqrMagnitude哪个好?动画切换优化shader属性优化 URP渲染器资产优化对象池优化删除没必要的空函数图片、音乐音效、贴图等素材压缩ScriptableObject优化参数参考完结 前言 …...
算法参考改进点/知识点
1、clip文章中改进点 图像编码器image encoder: 将全局平均池化层替换为注意力池化机制。注意力池化机制:通过一个单层的“transformer式”多头QKV注意力,其中查询query是基于图像的全局平均池表示。改进VIT(Vision Transformer…...
electron 配置、打包 -报错解决
目录 一、配置途中遇到的问题: 二、 make 配置好后开始打包 三、Electron-builder 打包报错 一、配置途中遇到的问题: 1. 安装 yarn add electron -D 一直卡在这里失败 一直卡可以使用下面这个,然后再重新装依赖 1. 采用新的镜像地址 npm …...
深入浅出Asp.Net Core MVC应用开发系列-AspNetCore中的日志记录
ASP.NET Core 是一个跨平台的开源框架,用于在 Windows、macOS 或 Linux 上生成基于云的新式 Web 应用。 ASP.NET Core 中的日志记录 .NET 通过 ILogger API 支持高性能结构化日志记录,以帮助监视应用程序行为和诊断问题。 可以通过配置不同的记录提供程…...
STM32+rt-thread判断是否联网
一、根据NETDEV_FLAG_INTERNET_UP位判断 static bool is_conncected(void) {struct netdev *dev RT_NULL;dev netdev_get_first_by_flags(NETDEV_FLAG_INTERNET_UP);if (dev RT_NULL){printf("wait netdev internet up...");return false;}else{printf("loc…...
c++ 面试题(1)-----深度优先搜索(DFS)实现
操作系统:ubuntu22.04 IDE:Visual Studio Code 编程语言:C11 题目描述 地上有一个 m 行 n 列的方格,从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子,但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...
渲染学进阶内容——模型
最近在写模组的时候发现渲染器里面离不开模型的定义,在渲染的第二篇文章中简单的讲解了一下关于模型部分的内容,其实不管是方块还是方块实体,都离不开模型的内容 🧱 一、CubeListBuilder 功能解析 CubeListBuilder 是 Minecraft Java 版模型系统的核心构建器,用于动态创…...
Keil 中设置 STM32 Flash 和 RAM 地址详解
文章目录 Keil 中设置 STM32 Flash 和 RAM 地址详解一、Flash 和 RAM 配置界面(Target 选项卡)1. IROM1(用于配置 Flash)2. IRAM1(用于配置 RAM)二、链接器设置界面(Linker 选项卡)1. 勾选“Use Memory Layout from Target Dialog”2. 查看链接器参数(如果没有勾选上面…...
高危文件识别的常用算法:原理、应用与企业场景
高危文件识别的常用算法:原理、应用与企业场景 高危文件识别旨在检测可能导致安全威胁的文件,如包含恶意代码、敏感数据或欺诈内容的文档,在企业协同办公环境中(如Teams、Google Workspace)尤为重要。结合大模型技术&…...
Rust 异步编程
Rust 异步编程 引言 Rust 是一种系统编程语言,以其高性能、安全性以及零成本抽象而著称。在多核处理器成为主流的今天,异步编程成为了一种提高应用性能、优化资源利用的有效手段。本文将深入探讨 Rust 异步编程的核心概念、常用库以及最佳实践。 异步编程基础 什么是异步…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
如何在最短时间内提升打ctf(web)的水平?
刚刚刷完2遍 bugku 的 web 题,前来答题。 每个人对刷题理解是不同,有的人是看了writeup就等于刷了,有的人是收藏了writeup就等于刷了,有的人是跟着writeup做了一遍就等于刷了,还有的人是独立思考做了一遍就等于刷了。…...
【JavaSE】多线程基础学习笔记
多线程基础 -线程相关概念 程序(Program) 是为完成特定任务、用某种语言编写的一组指令的集合简单的说:就是我们写的代码 进程 进程是指运行中的程序,比如我们使用QQ,就启动了一个进程,操作系统就会为该进程分配内存…...
