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

SynchronousQueueworkQueue.offer() 和 poll() 方法的超时控制及线程池的讨论

2024.11.12 11:26

问:workQueue.offer(command) 这里 workQueue.poll(keepAlive, TimeUnit.NANOSECONDS) : 这里的 workQueue.offer(command) 没有超时时间控制?

Claude 3.5 Haiku

非常好的问题!我来详细解释 offer() 和 poll() 方法的区别:

  1. offer() 方法:
 

java

// LinkedBlockingQueue 的 offer 方法 public boolean offer(E e) { if (e == null) throw new NullPointerException(); final AtomicInteger count = this.count; if (count.get() == capacity) return false; // 队列已满直接返回 false final ReentrantLock putLock = this.putLock; putLock.lock(); try { if (count.get() < capacity) { enqueue(new Node<E>(e)); c = count.getAndIncrement(); if (c + 1 < capacity) notFull.signal(); } } finally { putLock.unlock(); } return true; }

  1. poll() 方法:
 

java

// LinkedBlockingQueue 的 poll 方法(带超时) public E poll(long timeout, TimeUnit unit) throws InterruptedException { E x = null; int c = -1; final long nanos = unit.toNanos(timeout); final AtomicInteger count = this.count; final ReentrantLock takeLock = this.takeLock; takeLock.lockInterruptibly(); try { while (count.get() == 0) { if (nanos <= 0) return null; // 等待指定时间 nanos = notEmpty.awaitNanos(nanos); } x = dequeue(); c = count.getAndDecrement(); if (c > 1) notEmpty.signal(); } finally { takeLock.unlock(); } return x; }

关键区别:

  1. offer():
  • 无超时机制
  • 队列满返回 false
  • 立即返回
  • 不会阻塞
  1. poll(timeout):
  • 有超时机制
  • 可以等待指定时间
  • 队列为空时会等待
  • 超时返回 null

示例代码:

 

java

public class BlockingQueueDemo { public static void main(String[] args) throws InterruptedException { // 容量为2的阻塞队列 BlockingQueue<String> queue = new LinkedBlockingQueue<>(2); // offer 测试:无阻塞 System.out.println("offer 1: " + queue.offer("A")); // true System.out.println("offer 2: " + queue.offer("B")); // true System.out.println("offer 3: " + queue.offer("C")); // false,队列已满 // poll 测试:带超时 System.out.println("poll with timeout: " + queue.poll(100, TimeUnit.MILLISECONDS)); // 返回 "A" } }

线程池中的应用:

 

java

// 线程池中的任务提交逻辑 if (isRunning(c) && workQueue.offer(command)) { // 不会阻塞,队列满直接返回 false int recheck = ctl.get(); if (!isRunning(recheck) && remove(command)) reject(command); else if (workerCountOf(recheck) == 0) addWorker(null, false); }

如果需要超时控制的 offer,可以自定义:

 

java

public boolean offerWithTimeout(BlockingQueue<Runnable> queue, Runnable command, long timeout, TimeUnit unit) throws InterruptedException { return queue.offer(command, timeout, unit); }

建议:

  1. offer() 适合快速非阻塞场景
  2. poll(timeout) 适合需要等待的场景
  3. 线程池默认使用 offer(),快速失败
  4. 根据具体需求选择合适的队列和提交策略

问:如果是SynchronousQueue这里用offer() 方法,Synchrounous的长度为0 ,那不是直接返回false了,offer不会阻塞吗?

Claude 3.5 Haiku

对于 SynchronousQueue,offer() 方法确实不会阻塞。让我详细解释 SynchronousQueue 的特性:

SynchronousQueue 的 offer() 方法源码:

 

java

public boolean offer(E e) { if (e == null) throw new NullPointerException(); // 直接尝试传递,没有空闲接收者直接返回 false return transferer.transfer(e, false, 0) != null; }

关键特性:

  1. 队列长度始终为 0
  2. offer() 不会阻塞
  3. 必须有消费者立即接收

示例代码:

 

java

public class SynchronousQueueDemo { public static void main(String[] args) throws InterruptedException { // 公平模式 SynchronousQueue<String> queue = new SynchronousQueue<>(true); // offer 测试 boolean offered = queue.offer("Test"); // 直接返回 false System.out.println("Offered: " + offered); // false // 正确使用方式 ExecutorService executor = Executors.newFixedThreadPool(2); // 消费者 executor.submit(() -> { try { // 等待并接收 String item = queue.take(); System.out.println("Received: " + item); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } }); // 生产者 executor.submit(() -> { try { // 阻塞等待传递 queue.put("Hello"); System.out.println("Put successful"); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } }); executor.shutdown(); } }

线程池中的处理:

 

java

public void execute(Runnable command) { if (command == null) throw new NullPointerException(); int c = ctl.get(); // 如果线程数小于核心线程数,直接创建线程 if (workerCountOf(c) < corePoolSize) { if (addWorker(command, true)) return; c = ctl.get(); } // 对于 SynchronousQueue,offer 会立即返回 false if (isRunning(c) && workQueue.offer(command)) { // 这里实际上不会执行 int recheck = ctl.get(); if (! isRunning(recheck) && remove(command)) reject(command); else if (workerCountOf(recheck) == 0) addWorker(null, false); } // 立即尝试创建新线程 else if (!addWorker(command, false)) reject(command); }

SynchronousQueue 的两种模式:

  1. 非公平模式(默认):
 

java

SynchronousQueue<String> queue = new SynchronousQueue<>();

  1. 公平模式:
 

java

SynchronousQueue<String> queue = new SynchronousQueue<>(true);

正确使用 SynchronousQueue 的方法:

  1. put() 方法:阻塞直到有消费者
 

java

queue.put("item"); // 阻塞直到被取走

  1. offer(E e, long timeout, TimeUnit unit):带超时的阻塞
 

java

boolean success = queue.offer("item", 1, TimeUnit.SECONDS);

线程池建议:

  1. SynchronousQueue 适合:

    • 任务处理非常快
    • 不希望任务排队
    • 希望立即创建线程
  2. 使用注意:

    • 配合合适的拒绝策略
    • 控制最大线程数
    • 评估线程创建开销

完整示例:

 

java

public class ThreadPoolSynchronousQueueDemo { public static void main(String[] args) { // 使用 SynchronousQueue 的线程池 ThreadPoolExecutor executor = new ThreadPoolExecutor( 8, // 核心线程数 32, // 最大线程数 60L, // 空闲线程保留时间 TimeUnit.SECONDS, new SynchronousQueue<>(), // 无缓冲队列 new ThreadPoolExecutor.CallerRunsPolicy() // 由调用线程执行 ); // 提交大量任务 for (int i = 0; i < 50; i++) { final int taskId = i; executor.submit(() -> { System.out.println("Task " + taskId + " executed by " + Thread.currentThread().getName()); try { Thread.sleep(100); // 模拟任务执行 } catch (InterruptedException e) { Thread.currentThread().interrupt(); } }); } executor.shutdown(); } }

关键点总结:

  1. 推荐使用 put() 或 offer(E, timeout)
  2. 线程池会立即尝试创建新线程
  3. 选择合适的拒绝策略很重要

问:SynchronousQueue 如果有空闲线程在pool ,那是不是直接能offer

Claude 3.5 Haiku

非常好的追问!我们来深入分析 SynchronousQueue 的 offer 行为。

关键源码:

 

java

public boolean offer(E e) { // 调用 transfer 方法 return transferer.transfer(e, false, 0) != null; }

transfer 方法的核心逻辑:

  1. 如果有空闲线程(消费者)在等待,直接传递任务
  2. 没有空闲线程,立即返回 false

示例验证代码:

 

java

public class SynchronousQueueDemo { public static void main(String[] args) throws InterruptedException { // 创建线程池 ThreadPoolExecutor executor = new ThreadPoolExecutor( 8, // 核心线程数 32, // 最大线程数 60L, // 空闲线程保留时间 TimeUnit.SECONDS, new SynchronousQueue<>(), // 无缓冲队列 new ThreadPoolExecutor.CallerRunsPolicy() ); // 先创建一些线程,使线程池有空闲线程 for (int i = 0; i < 5; i++) { executor.submit(() -> { try { Thread.sleep(1000); // 模拟长任务 System.out.println("Long running task"); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } }); } // 等待线程创建 Thread.sleep(100); // 检查是否能直接 offer SynchronousQueue<Runnable> queue = (SynchronousQueue<Runnable>) executor.getQueue(); // 模拟任务 Runnable task = () -> System.out.println("Quick task executed"); // 尝试 offer boolean offered = queue.offer(task); System.out.println("Task offered: " + offered); // 关闭线程池 executor.shutdown(); } }

深入源码分析 transfer 方法:

 

java

// SynchronousQueue 内部 TransferQueue 实现 E transfer(E e, boolean timed, long nanos) { QNode s = null; boolean isData = (e != null); while (true) { QNode m = head; // 如果有等待的消费者线程 if (m != null && m.isData != isData) { // 直接匹配并传递任务 Object x = m.match(e); if (x != null) { return (E) x; } } // 没有匹配的线程,返回 null return null; } }

关键点:

  1. 如果线程池有空闲线程,SynchronousQueue 的 offer 会成功
  2. 成功的前提是:
    • 有空闲线程在等待任务
    • 线程处于 WAITING 状态
    • 线程在 workQueue 中

验证代码:

 

java

public class SynchronousQueueDetailDemo { public static void main(String[] args) throws Exception { // 创建线程池 ThreadPoolExecutor executor = new ThreadPoolExecutor( 8, // 核心线程数 32, // 最大线程数 60L, // 空闲线程保留时间 TimeUnit.SECONDS, new SynchronousQueue<>(), // 无缓冲队列 new ThreadPoolExecutor.CallerRunsPolicy() ); // 场景1:有空闲线程 CountDownLatch latch = new CountDownLatch(1); // 提交一个长任务,但很快返回 Future<?> longTask = executor.submit(() -> { try { latch.await(); // 模拟线程等待 System.out.println("Long task completed"); } catch (InterruptedException e) { Thread.currentThread().interrupt(); } }); // 等待线程创建 Thread.sleep(100); // 获取队列 SynchronousQueue<Runnable> queue = (SynchronousQueue<Runnable>) executor.getQueue(); // 尝试 offer Runnable quickTask = () -> System.out.println("Quick task executed"); boolean offered = queue.offer(quickTask); System.out.println("Task offered: " + offered); // 释放长任务 latch.countDown(); // 关闭线程池 executor.shutdown(); } }

总结:

  1. SynchronousQueue 的 offer 行为取决于是否有空闲线程
  2. 有空闲线程且处于等待状态可以直接 offer 成功
  3. 没有空闲线程会立即返回 false
  4. 线程池会立即尝试创建新线程

建议:

  • 监控线程池状态
  • 合理设置核心线程数和最大线程数
  • 选择合适的拒绝策略
  • 根据业务特点选择队列类型

相关文章:

SynchronousQueueworkQueue.offer() 和 poll() 方法的超时控制及线程池的讨论

2024.11.12 11:26 问&#xff1a;workQueue.offer(command) 这里 workQueue.poll(keepAlive, TimeUnit.NANOSECONDS) : 这里的 workQueue.offer(command) 没有超时时间控制&#xff1f; Claude 3.5 Haiku 非常好的问题&#xff01;我来详细解释 offer() 和 poll() 方法的区别…...

自动驾驶---“火热的”时空联合规划

1 背景 早期的不少规划算法都是横纵分离的(比如Apollo),先求解path之后,依赖path的结果再进行speed的求解。这种横纵解耦的规划方式具有以下特点: 相对较为简单,计算量通常较小,容易实现实时性要求。但是由于分别规划横向和纵向运动,可能会忽略两者之间的耦合关系,导致…...

命令行工具进阶指南

&#x1f680; 命令行工具进阶指南&#xff1a;Git、Shell与效率工具的进阶之路 掌握命令行工具&#xff0c;让你的开发效率突飞猛进。本文将深入探讨 Git 高级技巧、Shell 脚本自动化以及各种效率倍增的 CLI 工具。 &#x1f4d1; 目录 Git 高级技巧与工作流Shell 脚本自动化…...

扫雷游戏代码分享(c基础)

hi , I am 36. 代码来之不易&#x1f44d;&#x1f44d;&#x1f44d; 创建两个.c 一个.h 1&#xff1a;test.c #include"game.h"void game() {//创建数组char mine[ROWS][COLS] { 0 };char show[ROWS][COLS] { 0 };char temp[ROWS][COLS] { 0 };//初始化数…...

基于vue框架的的社区居民服务管理系统8w86o(程序+源码+数据库+调试部署+开发环境)系统界面在最后面。

系统程序文件列表 项目功能&#xff1a;居民,楼房信息,报修信息,缴费信息,维修进度 开题报告内容 基于Vue框架的社区居民服务管理系统开题报告 一、研究背景与意义 随着城市化进程的加速&#xff0c;社区居民数量激增&#xff0c;社区管理面临着前所未有的挑战。传统的社区…...

一分钟快速熟悉makedown

Markdown 是一种轻量级标记语言&#xff0c;广泛用于编写文档、撰写博客、创建 README 文件等。它的语法简单易学&#xff0c;能够快速生成格式化的文本。以下是 Markdown 的一些常用语法和示例&#xff1a; 1. 标题 Markdown 支持六级标题&#xff0c;使用 # 符号表示。 # …...

P8649 [蓝桥杯 2017 省 B] k 倍区间:同余,前缀和,组合数,区间个数

题目描述 给定一个长度为 NN 的数列&#xff0c;A1,A2,⋯ANA1​,A2​,⋯AN​&#xff0c;如果其中一段连续的子序列 Ai,Ai1,⋯Aj(i≤j)Ai​,Ai1​,⋯Aj​(i≤j) 之和是 KK 的倍数&#xff0c;我们就称这个区间 [i,j][i,j] 是 KK 倍区间。 你能求出数列中总共有多少个 KK 倍区…...

产业与学术相互促进,2024年OEG海上能源博览会助力全球能源可持续发展

10月30日至31日&#xff0c;2024年OEG海上能源全产业链博览会在上海跨国采购会展中心成功举办。本次大会系全球海洋工程与高端装备领域的年度国际交流盛会——第十一届全球FPSO&FLNG&FSRU大会&#xff0c;同期举办第七届亚洲海洋风能大会。本次大会暨博览会由上海船舶工…...

【GDB调试】智慧中控项目的调试

一.在执行的智慧中控项目的时候&#xff0c;喊语音模块唤醒(小欣小欣)的时候遇到了&#xff1a;Segmentation fault 段错误 二.遇到段错误&#xff0c;一般是以下情况&#xff1a; “Segmentation fault”&#xff08;段错误&#xff09;是Linux系统中常见的程序异常终止信号。…...

《一本书讲透 Elasticsearch》京东评论采集+存储+可视化全 AI 实现

经常和出版社编辑老师交流读者的反馈。毕竟是小众书籍&#xff0c;豆瓣评分的人并不多。 而京东作为主要读书销售渠道&#xff0c;非常有必要整合一下京东读者评论&#xff0c;看看读者们都说了什么&#xff0c;以便后续的改进&#xff01; 一条条的翻看非常不方便&#xff0c;…...

uniapp中webview全屏不显示导航栏解决方案

uniapp官网文档地址&#xff1a;https://uniapp.dcloud.net.cn/api/window/window.html#getappwebview <template><view class"index"><u-navbar :is-back"true" title"标题"" :title-width"650"></u-navb…...

Dear ImGui 使用VS2022编译为静态库

Dear ImGui 是一个无臃肿的 C++ 图形用户界面库。它输出优化的顶点缓冲区,您可以在支持 3D 管道的应用程序中随时渲染这些缓冲区。它速度快、可移植、与渲染器无关且自成一体(无外部依赖项)。 Dear ImGui 旨在实现快速迭代,并让程序员能够创建内容创建工具和可视化/调试工具…...

5G 现网信令参数学习(3) - RrcSetup(1)

目录 1. rlc-BearerToAddModList 1.1 rlc-Config 1.1.1 ul-AM-RLC 1.1.2 dl-AM-RLC 1.2 mac-LogicalChannelConfig 2. mac-CellGroupConfig 2.1 schedulingRequestConfig 2.2 bsr-Config 2.3 tag-Config 2.4 phr-Config 2.5 skipUplinkTxDynamic 3. physicalCellG…...

PHP实现身份证OCR识别API接口

随着社会的发展&#xff0c;身份认证需求不断增长&#xff0c;这与身份证OCR识别技术的发展密切相关。在当今社会&#xff0c;各个领域都需要进行身份认证。传统的人工手动录入身份证信息费时费力&#xff0c;速度慢且容易出错&#xff0c;体验不佳。而身份证 OCR 识别技术通过…...

关于 Qt+Osg中使用背景图HUD受到后绘制几何图形顶点颜色影响 的解决方法

若该文为原创文章&#xff0c;转载请注明出处 本文章博客地址&#xff1a;https://hpzwl.blog.csdn.net/article/details/143607816 长沙红胖子Qt&#xff08;长沙创微智科&#xff09;博文大全&#xff1a;开发技术集合&#xff08;包含Qt实用技术、树莓派、三维、OpenCV、Op…...

[CKS] K8S AppArmor Set Up

最近准备花一周的时间准备CKS考试&#xff0c;在准备考试中发现有一个题目关于AppArmor Pod操作权限的问题。 ​ 专栏其他文章: [CKS] Create/Read/Mount a Secret in K8S-CSDN博客[CKS] Audit Log Policy-CSDN博客 -[CKS] 利用falco进行容器日志捕捉和安全监控-CSDN博客[CKS] …...

redis笔记-数据结构

zset zset一方面它是一个 set&#xff0c;保证了内部value 的唯一性&#xff0c;另一方面它可以给每个 value 赋予一个 score&#xff0c;代表这个 value 的排序权重。 zset的底层是由字典和跳表实现。 字典主要用来存储value和score的对应关系。跳表这个数据结构主要用来提…...

webpack的常见配置

Webpack 是一个现代 JavaScript 应用的模块打包工具&#xff0c;用于将项目中的多个文件和依赖打包成浏览器可以识别的文件&#xff0c;通常是一个或多个 JavaScript、CSS 或其他静态资源的 bundle&#xff08;将多个模块或文件合并成一个或几个文件的过程&#xff0c;这些合并…...

text-embedding-ada-002;BGE模型;M3E模型是Moka Massive Mixed Embedding;BERT

目录 text-embedding-ada-002 一、模型概述 二、模型功能 三、模型特点 四、模型应用 五、模型优势 BGE模型 一、模型背景与特点 二、模型性能与表现 三、模型迭代与发展 M3E模型是Moka Massive Mixed Embedding 一、基本信息 二、技术特点 三、应用场景 四、性能…...

WebRTC 环境搭建

主题 本文主要描述webrtc开发过程中所需的环境搭建 环境&#xff1a; 运行环境&#xff1a;ubuntu 20.04 Node.js环境搭建 安装编译 Node.js 所需的依赖包: sudo apt-get update sudo apt-get install -y build-essential libssl-dev 下载 Node.js 源码: curl -sL htt…...

FastHTML快速入门:http方法,CSS文件和内联样式,其他静态媒体文件位置

HTTP方法 FastHTML通过函数名与HTTP方法进行匹配。到目前为止&#xff0c;我们定义的URL路由都是针对HTTP GET方法的&#xff0c;这是网页最常见的方法。 表单提交通常作为HTTP POST发送。在处理更动态的网页设计时&#xff0c;也就是所谓的单页应用&#xff08;SPA&#xff0…...

项目管理和研发管理中的痛点及其解决方案

在现代企业中&#xff0c;研发管理和项目管理面临着多重挑战&#xff0c;包括资源配置不当、沟通不畅、目标不明确、进度控制困难等。这些痛点不仅影响项目的顺利推进&#xff0c;还可能导致企业在市场竞争中处于劣势。尤其是在资源配置不当方面&#xff0c;企业往往难以合理分…...

机器学习(基础1)

数据集 sklearn玩具数据集 数据量小&#xff0c;数据在sklearn库的本地&#xff0c;只要安装了sklearn&#xff0c;不用上网就可以获取 sklearn现实世界数据集 数据量大&#xff0c;数据只能通过网络获取&#xff08;为国外数据集&#xff0c;下载需要梯子&#xff09; skle…...

我谈维纳(Wiener)复原滤波器

Rafael Gonzalez的《数字图像处理》中&#xff0c;图像复原这章内容几乎全错。上篇谈了图像去噪&#xff0c;这篇谈图像复原。 图像复原也称为盲解卷积&#xff0c;不处理点扩散函数&#xff08;光学传递函数&#xff09;的都不是图像复原。几何校正不属于图像复原&#xff0c…...

怎么看真假国企啊?怎么识别假冒国企的千层套路?

一、怎么看真假国企啊&#xff1f; 1.使用具有迷惑性的名称&#xff1a;假冒国企往往在名称中使用“中国”、“中”、“国”等字样&#xff0c;或与知名国企名称相似的字号&#xff0c;以增加其可信度。 2.注册资本虚高&#xff1a;为了显示实力&#xff0c;假冒国企可能会在…...

C#中break和continue的区别?

在C#编程语言中&#xff0c;break和continue是两个用于控制循环流程的关键字&#xff0c;但它们的作用和用途有所不同。 break关键字 break关键字用于立即终止它所在的最内层循环或switch语句&#xff0c;并跳出该循环或switch块。程序执行将继续进行循环或switch语句之后的下一…...

Linux部署nginx访问文件403

问题描述&#xff1a;在linux服务器上通过nginx部署&#xff0c;访问文件403 新配置了一个用户来部署服务&#xff0c;将部署文件更新到原有目录下&#xff0c;结果nginx访问403 原因&#xff1a;没有配置文件的读写权限&#xff0c;默认不可读写&#xff0c;nginx无法访问到文…...

华为OD机试 - 数字排列 - 深度优先搜索dfs算法(Python/JS/C/C++ 2024 C卷 200分)

华为OD机试 2024E卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试真题&#xff08;Python/JS/C/C&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;私信哪吒&#xff0c;备注华为OD&#xff0c;加入华为OD刷题交流群&#xff0c;…...

Scrapy爬取heima论坛所有页面内容并保存到数据库中

前期准备&#xff1a; Scrapy入门_win10安装scrapy-CSDN博客 新建 Scrapy项目 scrapy startproject mySpider03 # 项目名为mySpider03 进入到spiders目录 cd mySpider03/mySpider03/spiders 创建爬虫 scrapy genspider heima bbs.itheima.com # 爬虫名为heima &#…...

Kafka参数了解

Kafka配置参数完整说明 1. 基础配置 参数名说明推荐值参考值broker.idbroker的唯一标识符每个节点唯一的整数1delete.topic.enable是否允许删除topictruetruelistenersbroker监听地址SASL_PLAINTEXT://host:9092SASL_PLAINTEXT://172.24.77.15:9092advertised.listeners对外发…...

现在网站后台有哪几种模板形式/什么软件可以免费发广告

一&#xff1a;下载虚拟机 VMware 官网&#xff1a;https://www.vmware.com/cn/products/workstation-pro/workstation-pro-evaluation.html 选择你所对应的系统 我这里是windows操作系统 下载成功后安装 打开界面是这样子的 二&#xff1a;介绍一下 CentOS CentOS是免费的、…...

做康复医院网站/怎样能在百度上搜索到自己的店铺

HTML5 Web SQL Web SQL 数据库可以在最新版的 Safari, Chrome 和 Opera 浏览器中工作。 三个核心方法&#xff1a; openDatabase&#xff1a;这个方法使用现有的数据库或者新建的数据库创建一个数据库对象。transaction&#xff1a;这个方法让我们能够控制一个事务&#xff…...

企业网站建设 安全/网页友情链接

1&#xff0c;冒泡排序 基本思想&#xff1a;每趟排序从待排关键字记录中的第一个关键字记录与第二个关键字记录相比较&#xff0c;若第一个关键字大于第二个关键字&#xff0c;则将两者交换&#xff0c;再次比较第二个关键字和第三个关键字&#xff0c;若为逆序&#xff0c;则…...

国内企业建站模板/制作网站需要什么技术

修改mysql的root密码后&#xff0c;出现Host localhost is not allowed to connect to this MySQL server 错误。 解决办法&#xff1a; C:\Program Files\MySQL\MySQL Server 5.5\my.ini 在[mysqld]下加下面两行&#xff0c; skip-name-resolveskip-grant-tables 重启mysql的w…...

类似凡科互动的网站/磁力帝

视频中用到的是<bean id...> <constructor-arg ref"userDao4Mysql"/> </bean> 表示在构造函数中引用userDao4Mysql这个类。 ref是引用的意思。...

戚墅堰做网站/拉新推广平台有哪些

过去的自己&#xff0c;你好&#xff01; 又是好久没有写东西了&#xff0c;工作和生活都很忙&#xff0c;总是没空思考。你以后会变得和我一样忙碌&#xff0c;千万不要安逸&#xff0c;因为所谓的“艰难困苦”来的比你想象的快得多。Life is tough, you must be tougher。…...