【Java集合夜话】第1篇:拨开迷雾,探寻集合框架的精妙设计
欢迎来到Java集合框架系列的第一篇文章!🌹
本系列文章将以通俗易懂的语言,结合实际开发经验,带您深入理解Java集合框架的设计智慧。🌹
若文章中有任何不准确或需要改进的地方,欢迎大家指出,让我们一起在交流中进步!🌹
文章中的所有示例代码都经过实际验证,您可以直接在项目中参考使用。如果觉得有帮助,欢迎点赞转发!🌹
目录
-
一、基础概念
- 1. 集合框架层次结构
- List家族
- Set家族
- Queue家族
- 2. Collection与Map的区别
- Collection特性
- Map特性
- 选择建议
- 3. 集合与数组对比
- 1. 集合框架层次结构
-
二、进阶特性
- 4. 泛型在集合中的应用
- 5. 集合框架的常用接口
- 6. fail-fast与fail-safe机制
- 快速失败机制
- 安全失败机制
- 使用建议
-
三、并发处理
- 7. 同步包装器详解
- 基本概念
- 工作原理
- 使用注意事项
- 最佳实践
- 7. 同步包装器详解
阅读提示:
- 如果你是初学者,建议按顺序阅读
- 如果你是有经验的开发者,可以直接跳转到感兴趣的章节
- 每个主题都配有代码示例和实践建议
Java集合框架基础精讲
在上一篇《Java集合框架学习指南》中,我们绘制了一张完整的知识地图。今天,让我们迈出探索的第一步,深入理解Java集合框架的设计精髓。
一、基础概念
1. 集合框架层次结构
Java集合框架主要分为两大体系:
- Collection:存储元素的集合,就像一个容器,可以存放多个元素
- Map:存储键值对的映射,就像一个字典,可以通过键快速找到值
Collection 体系
Collection
├── List(有序、可重复)
│ ├── ArrayList
│ ├── LinkedList
│ └── Vector
├── Set(无序、不重复)
│ ├── HashSet
│ ├── TreeSet
│ └── LinkedHashSet
└── Queue(队列)├── PriorityQueue└── Deque
List家族
List代表有序、可重复的集合,就像我们排队一样,每个人都有明确的位置。
- ArrayList
// 创建和添加元素
List<String> fruits = new ArrayList<>();
fruits.add("苹果");
fruits.add("香蕉");
fruits.add("橙子");// 按索引访问
System.out.println(fruits.get(0)); // 输出:苹果// 插入和删除
fruits.add(1, "葡萄"); // 在索引1处插入
fruits.remove("香蕉"); // 删除指定元素
- 特点:查询快,增删慢
- 场景:适合频繁查询,较少增删的场景
- 应用:展示列表、数据缓存等
- LinkedList
// 创建和操作LinkedList
LinkedList<String> tasks = new LinkedList<>();
tasks.addFirst("任务1"); // 添加到开头
tasks.addLast("任务2"); // 添加到结尾
tasks.removeFirst(); // 移除第一个
tasks.removeLast(); // 移除最后一个
- 特点:增删快,查询慢
- 场景:适合频繁增删,较少查询的场景
- 应用:任务队列、消息队列等
Set家族
Set代表无序、不重复的集合,就像一个集市,东西只能有一份。
- HashSet
// 创建和使用HashSet
Set<String> tags = new HashSet<>();
tags.add("Java");
tags.add("Python");
tags.add("Java"); // 重复元素不会被添加
System.out.println(tags); // 输出:[Java, Python]// 判断元素是否存在
boolean hasJava = tags.contains("Java"); // 返回true
- 特点:查询快,无序
- 场景:需要去重的场景
- 应用:标签系统、去重统计等
- TreeSet
// 创建和使用TreeSet
TreeSet<Integer> scores = new TreeSet<>();
scores.add(85);
scores.add(92);
scores.add(78);
System.out.println(scores); // 自动排序:[78, 85, 92]// 范围查询
System.out.println(scores.ceiling(80)); // 大于等于80的最小值
System.out.println(scores.floor(90)); // 小于等于90的最大值
- 特点:有序(自然顺序或自定义顺序)
- 场景:需要排序的去重场景
- 应用:排行榜、成绩统计等
Queue家族
Queue代表队列,就像排队买票,先来先服务。
- PriorityQueue
// 创建优先队列
PriorityQueue<String> taskQueue = new PriorityQueue<>((a, b) -> b.length() - a.length()); // 按字符串长度降序排列
taskQueue.offer("短任务");
taskQueue.offer("非常长的任务");
taskQueue.offer("中等任务");// 获取任务
System.out.println(taskQueue.poll()); // 输出:非常长的任务
- 特点:自动排序的队列
- 场景:需要按优先级处理的场景
- 应用:任务调度、事件处理等
- ArrayDeque
// 作为栈使用
ArrayDeque<String> stack = new ArrayDeque<>();
stack.push("第一层");
stack.push("第二层");
System.out.println(stack.pop()); // 输出:第二层// 作为队列使用
ArrayDeque<String> queue = new ArrayDeque<>();
queue.offer("第一个");
queue.offer("第二个");
System.out.println(queue.poll()); // 输出:第一个
- 特点:双端队列,可以作为栈或队列使用
- 场景:需要同时支持栈和队列操作的场景
- 应用:撤销重做、历史记录等
实践建议
- 选择合适的集合类型
// 需要频繁查询
List<User> userList = new ArrayList<>();// 需要频繁增删
List<Message> messageQueue = new LinkedList<>();// 需要去重
Set<String> uniqueEmails = new HashSet<>();// 需要排序
Set<Score> scoreBoard = new TreeSet<>();
- 合理估计初始容量
// 如果知道大概需要存储100个元素
List<String> list = new ArrayList<>(100); // 避免频繁扩容
- 使用接口类型声明
// 推荐
List<String> list = new ArrayList<>();
Set<Integer> set = new HashSet<>();// 不推荐
ArrayList<String> list = new ArrayList<>();
HashSet<Integer> set = new HashSet<>();
小贴士:选择合适的集合类型可以显著提升程序性能。在实际开发中,要根据具体场景选择合适的实现类。
但是这里一直有个误区,大家都会觉得ArrayList查询快,插入慢,插入场景应使用LinkedList,其实道理上是这样,但是大部分场景如果是尾插法,其实ArrayList的性能和LinkedList是差不多的,大部分场景ArrayList甚至更甚一筹,在使用ArrayList的时候推荐从一开始就定义好集合大小,减少扩容成本。
2. Collection 与 Map 的区别与应用
Collection 特性与应用
Collection 接口是所有集合的根接口,主要用于存储单个元素的集合。
基本操作
// 创建集合
Collection<String> collection = new ArrayList<>();// 添加元素
collection.add("Java");
collection.addAll(Arrays.asList("Python", "Go"));// 删除元素
collection.remove("Java");// 判断包含
boolean hasGo = collection.contains("Go");// 获取大小
int size = collection.size();// 直接遍历(实现了Iterable接口)
for (String item : collection) {System.out.println(item);
}
常见应用场景
- 存储同类型元素集合
// 存储用户列表
Collection<User> users = new ArrayList<>();
users.add(new User("张三"));
users.add(new User("李四"));// 存储唯一标识符
Collection<String> ids = new HashSet<>();
ids.add("A001");
ids.add("A002");
- 批量数据处理
// 批量处理订单
Collection<Order> orders = getOrders();
orders.removeIf(order -> order.getStatus().equals("CANCELLED"));
orders.forEach(Order::process);
Map 特性与应用
Map 接口专门用于存储键值对映射关系,通过键可以快速找到对应的值。
基本操作
// 创建Map
Map<String, User> userMap = new HashMap<>();// 添加键值对
userMap.put("u001", new User("张三"));
userMap.put("u002", new User("李四"));// 获取值
User user = userMap.get("u001");// 检查键是否存在
boolean hasKey = userMap.containsKey("u001");
boolean hasValue = userMap.containsValue(user);// 删除键值对
userMap.remove("u001");// 获取所有键或值
Set<String> keys = userMap.keySet();
Collection<User> values = userMap.values();// 遍历键值对
for (Map.Entry<String, User> entry : userMap.entrySet()) {System.out.println(entry.getKey() + ": " + entry.getValue());
}
常见应用场景
- 缓存实现
// 简单的缓存实现
Map<String, Object> cache = new HashMap<>();
cache.put("userConfig", loadUserConfig());
cache.put("systemConfig", loadSystemConfig());// 带过期时间的缓存
Map<String, CacheEntry> timeCache = new HashMap<>();
timeCache.put("data", new CacheEntry(data, System.currentTimeMillis() + 3600000));
- 数据分组统计
// 统计单词出现频率
Map<String, Integer> wordCount = new HashMap<>();
for (String word : words) {wordCount.merge(word, 1, Integer::sum);
}// 按类别分组
Map<String, List<Product>> productByCategory = new HashMap<>();
for (Product product : products) {productByCategory.computeIfAbsent(product.getCategory(), k -> new ArrayList<>()).add(product);
}
选择建议
- 使用Collection场景:
- 需要存储同类型元素集合
- 需要频繁遍历数据
- 不需要键值对关联
- 使用Map场景:
- 需要键值对应关系
- 需要根据键快速查找值
- 需要数据分组或统计
性能考虑
// Collection性能优化
Collection<String> list = new ArrayList<>(1000); // 预设容量
Collection<String> set = new HashSet<>(1000, 0.75f); // 预设容量和负载因子// Map性能优化
Map<String, User> map = new HashMap<>(1000); // 预设容量
Map<String, User> linkedMap = new LinkedHashMap<>(1000, 0.75f, true); // 按访问顺序
小贴士:
- Collection 适合对元素进行遍历和批量操作
- Map 适合需要快速查找和建立关联关系的场景
- 在初始化时,尽可能指定合适的初始容量,避免频繁扩容
- 根据实际需求选择合适的实现类,如需要排序可以选择 TreeMap/TreeSet
3. 集合与数组对比
让我们通过一个简单的对比表格来了解数组和集合的主要区别:
特性 | 数组 | 集合 |
---|---|---|
长度 | 固定长度 | 动态可变 |
数据类型 | 支持基本类型和引用类型 | 只支持引用类型 |
性能 | 较好(直接内存寻址) | 较差(需要额外开销) |
功能特性 | 只能存储和访问 | 提供丰富的操作方法 |
// 数组示例
int[] numbers = new int[5]; // 固定长度为5
String[] names = {"Tom", "Jerry"}; // 初始化固定内容// 集合示例
List<Integer> numberList = new ArrayList<>(); // 动态长度
numberList.add(1); // 自动扩容
选择建议:当数据量固定且明确时使用数组,需要动态管理数据时使用集合。
4. 泛型在集合中的应用
// 不使用泛型
List list = new ArrayList();
list.add("Hello");
String str = (String) list.get(0); // 需要强制转换// 使用泛型
List<String> list = new ArrayList<>();
list.add("Hello");
String str = list.get(0); // 无需转换
正常来说,已知类型的情况下,我们将类型给定义出来是最好的,可以为我们减少很多不必要的麻烦。
5. 集合框架的常用接口
Java集合框架提供了几个核心接口,每个接口都有其特定用途:
接口 | 特点 | 常用实现类 | 典型应用场景 |
---|---|---|---|
Collection | 集合的根接口,定义基本操作 | - | 作为通用集合类型使用 |
List | 有序、可重复、可按索引访问 | ArrayList, LinkedList | 存储列表数据,如商品列表 |
Set | 无序、不可重复 | HashSet, TreeSet | 存储唯一元素,如用户ID |
Queue | 队列接口,FIFO | LinkedList, PriorityQueue | 任务队列,消息处理 |
Map | 键值对映射 | HashMap, TreeMap | 缓存,键值对应关系 |
// 常见使用示例
List<String> list = new ArrayList<>(); // 存储有序数据
Set<Integer> set = new HashSet<>(); // 存储唯一数字
Queue<Task> queue = new LinkedList<>(); // 任务队列
Map<String, User> map = new HashMap<>(); // 用户信息映射
小贴士:选择合适的接口类型可以让代码更灵活,建议优先使用接口类型声明变量。
6. fail-fast 与 fail-safe 机制
快速失败机制
fail-fast 是Java集合的一种错误检测机制,在用迭代器遍历集合时,如果发现集合被修改,会立即抛出 ConcurrentModificationException 异常。
示例代码
// fail-fast示例
List<String> list = new ArrayList<>();
list.add("A");
list.add("B");// 错误示范:直接修改集合
for (String item : list) {if ("A".equals(item)) {list.remove(item); // 会抛出ConcurrentModificationException}
}// 正确做法:使用迭代器修改
Iterator<String> it = list.iterator();
while (it.hasNext()) {String item = it.next();if ("A".equals(item)) {it.remove(); // 正确的删除方式}
}
多线程场景
// 多线程环境下的fail-fast
List<String> list = new ArrayList<>();
list.add("A");
list.add("B");// 一个线程遍历
new Thread(() -> {for (String item : list) {try {Thread.sleep(100);System.out.println(item);} catch (InterruptedException e) {}}
}).start();// 另一个线程修改
new Thread(() -> {try {Thread.sleep(50);list.add("C"); // 可能触发ConcurrentModificationException} catch (InterruptedException e) {}
}).start();
安全失败机制
fail-safe 机制在遍历时会在集合的副本上操作,因此在遍历过程中对原集合的修改不会反映在遍历结果中。
示例代码
// fail-safe示例
CopyOnWriteArrayList<String> safeList = new CopyOnWriteArrayList<>();
safeList.add("A");
safeList.add("B");// 可以安全遍历和修改
for (String item : safeList) {System.out.println(item);safeList.add("C"); // 不会抛出异常,但遍历的是副本
}// 并发Map的安全遍历
ConcurrentHashMap<String, String> safeMap = new ConcurrentHashMap<>();
safeMap.put("key1", "value1");
safeMap.put("key2", "value2");for (Map.Entry<String, String> entry : safeMap.entrySet()) {System.out.println(entry.getKey());safeMap.put("key3", "value3"); // 安全的并发修改
}
使用建议
- 单线程环境:
// 使用迭代器的remove方法
Iterator<String> it = list.iterator();
while (it.hasNext()) {String item = it.next();if (needToRemove(item)) {it.remove();}
}// 或者使用removeIf方法
list.removeIf(item -> needToRemove(item));
- 多线程环境:
// 使用线程安全的集合类
List<String> safeList = new CopyOnWriteArrayList<>();
Map<String, String> safeMap = new ConcurrentHashMap<>();// 或者使用同步包装器
List<String> syncList = Collections.synchronizedList(new ArrayList<>());
synchronized (syncList) {for (String item : syncList) {// 安全地处理元素}
}
小贴士:
- 在单线程环境中,优先使用迭代器的方法进行修改
- 在多线程环境中,优先使用线程安全的集合类
- fail-safe 虽然安全,但会带来额外的内存开销
- 需要实时性的场景要谨慎使用 fail-safe 机制
7. 同步包装器详解
同步包装器(Synchronized Wrappers)是 Collections 工具类提供的一种将非线程安全的集合转换为线程安全的集合的机制。
基本概念
同步包装器通过装饰器模式,在原有集合的所有方法上添加 synchronized 关键字来实现线程安全:
// 创建同步包装器
List<String> syncList = Collections.synchronizedList(new ArrayList<>());
Set<String> syncSet = Collections.synchronizedSet(new HashSet<>());
Map<String, String> syncMap = Collections.synchronizedMap(new HashMap<>());
工作原理
// 简化的内部实现原理
public static <T> List<T> synchronizedList(List<T> list) {return new SynchronizedList<>(list); // 返回一个同步包装类
}static class SynchronizedList<E> implements List<E> {private final List<E> list; // 持有原始列表的引用private final Object mutex; // 同步锁对象// 所有方法都使用 synchronized 关键字public void add(E e) {synchronized (mutex) {list.add(e);}}public E get(int index) {synchronized (mutex) {return list.get(index);}}
}
使用注意事项
- 迭代时需要手动同步
List<String> list = Collections.synchronizedList(new ArrayList<>());// 错误方式:可能导致并发问题
for (String item : list) {// 处理元素
}// 正确方式:手动同步
synchronized (list) {for (String item : list) {// 处理元素}
}
- 批量操作的原子性
List<String> list = Collections.synchronizedList(new ArrayList<>());// 非原子操作,需要额外同步
synchronized (list) {if (!list.contains("A")) {list.add("A");}
}
最佳实践
- 低并发场景的简单实现
// 适合并发访问少的场景
List<String> syncList = Collections.synchronizedList(new ArrayList<>());
- 高并发场景应选择并发集合
// 更适合并发访问多的场景
List<String> concurrentList = new CopyOnWriteArrayList<>();
Map<String, String> concurrentMap = new ConcurrentHashMap<>();
- 需要原子操作时使用同步块
List<String> list = Collections.synchronizedList(new ArrayList<>());
synchronized (list) {// 需要原子性的复合操作if (list.isEmpty()) {list.add("First");}
}
核心提示:
- 同步包装器适合简单的线程安全需求
- 高并发场景建议使用专门的并发集合
- 注意迭代和复合操作时的同步处理
- 考虑性能需求选择合适的线程安全实现
总结
本文详细介绍了Java集合框架的基础知识,主要包括:
-
集合框架的整体架构
- Collection和Map两大体系
- 主要接口和实现类的特点
-
核心特性
- 泛型的应用
- fail-fast与fail-safe机制
- 同步包装器的使用
-
实践建议
- 集合类选择的考虑因素
- 性能优化的关键点
- 并发场景的最佳实践
通过本文的学习,相信你已经对Java集合框架有了一个清晰的认识。在实际开发中,合理使用这些知识点,可以帮助我们写出更优雅、更高效的代码。
下期预告
在Java集合框架这片广阔的知识海洋中,我们今天完成了第一次探索。而这仅仅是一个开始,在下一篇文章中,我们将继续深入Collection这个核心接口家族,去探索更多的技术奥秘:
🌟 我们将一起:
- 揭开Collection接口的设计智慧
- 解读核心方法的源码实现
- 探究各个实现类的内部机制
- 分享实战中的性能优化技巧
- 传授源码级别的最佳实践
技术的学习是一场永不停歇的旅程,而我很幸运能够与你同行。如果这篇文章对你有所帮助,请别忘了点赞、关注、收藏!你的每一个互动,都是我继续创作的动力!🌹🌹🌹
愿我们能在技术的道路上携手前行,共同进步!🌹🌹🌹
“种一棵树最好的时间是十年前,其次是现在。”
学习也是如此,让我们一起在技术的沃土上耕耘,收获知识的果实!
相关文章:
【Java集合夜话】第1篇:拨开迷雾,探寻集合框架的精妙设计
欢迎来到Java集合框架系列的第一篇文章!🌹 本系列文章将以通俗易懂的语言,结合实际开发经验,带您深入理解Java集合框架的设计智慧。🌹 若文章中有任何不准确或需要改进的地方,欢迎大家指出,让我…...
Prometheus使用
介绍:Prometheus 是一个开源的 监控与告警系统,主要用于采集和存储时间序列数据(Time Series Data) Prometheus的自定义查询语言PromQL Metric类型 为了能够帮助用户理解和区分这些不同监控指标之间的差异,Prometheu…...
Java学习打卡-Day19-Set、HashSet、LinkedHashSet
Set 接口 无序(添加和取出顺序不一致)(但取出顺序固定)。没有索引。不允许重复,所以最多一个null。遍历方式 迭代器增强for循环不能使用普通for循环索引方式。 HashSet 实现了Set接口,具有相应特征。底…...
冯・诺依曼架构深度解析
一、历史溯源:计算机科学的革命性突破 1.1 前冯・诺依曼时代 在 1940 年代之前,计算机领域呈现 "百家争鸣" 的格局: 哈佛 Mark I(1944):采用分离的指令存储与数据存储ENIAC(1946&a…...
单片机学完开发板,如何继续提升自己的技能?
很多人学完开发板后都会卡在一个尴尬的阶段:觉得自己会的东西不少,但又不知道下一步该干啥。会点C语言,能烧录程序,能点亮LED,玩转按键,搞定串口等等,能用开发板做点小玩意儿,但面对…...
Nginx 日志格式
默认日志格式配置 log_format main $remote_addr - $remote_user [$time_local] "$request" $status $body_bytes_sent "$http_referer" "$http_user_agent" "$http_x_forwarded_for";该格式记录了客户端IP、用户、时间、请求、状态…...
Spring Boot 整合 Elasticsearch 实践:从入门到上手
引言 Elasticsearch 是一个开源的分布式搜索引擎,广泛用于日志分析、搜索引擎、数据分析等场景。本文将带你通过一步步的教程,在 Spring Boot 项目中整合 Elasticsearch,轻松实现数据存储与查询。 1. 创建 Spring Boot 项目 首先ÿ…...
STM32 —— 嵌入式系统、通用计算机系统、物联网三层架构
目录 一、嵌入式系统的概念 二、通用计算机系统与嵌入式系统的比较 用途 硬件 软件 性能与功耗 开发与维护 三、嵌入式系统与物联网的关系 四、物联网的三层架构 1. 感知层(Perception Layer) 2. 网络层(Network Layer) …...
SARAD 解读
出处:NIPS 2024 代码链接:https://github.com/daidahao/SARAD/ 一 文章动机 ① 时间建模(Temporal Modeling)的局限性: a. 时间维度上 感受野极小;b. 变量间时间戳错位 (时间建模、空间建模不统一) →…...
【愚公系列】《高效使用DeepSeek》017-知识点思维导图生成
🌟【技术大咖愚公搬代码:全栈专家的成长之路,你关注的宝藏博主在这里!】🌟 📣开发者圈持续输出高质量干货的"愚公精神"践行者——全网百万开发者都在追更的顶级技术博主! 👉 江湖人称"愚公搬代码",用七年如一日的精神深耕技术领域,以"…...
【linux】scp和rsync
scp 和 rsync 都是 Linux 系统中用于文件传输的命令行工具,它们都可以通过网络在本地和远程主机之间传输文件。 scp 命令 定义 scp 是 “secure copy” 的缩写,它是一个基于 SSH 协议的文件传输工具,用于在本地和远程主机之间安全地复制文…...
软件需求分类、需求获取(高软46)
系列文章目录 软件需求分类,需求获取 文章目录 系列文章目录前言一、软件需求二、获取需求三、真题总结 前言 本节讲明软件需求分类、需求获取的相关知识。 一、软件需求 二、获取需求 三、真题 总结 就是高软笔记,大佬请略过!...
蓝桥杯单片机之AT24C02(基于自己对AT24C02的学习和理解)
一、先用抽象法说明原理,让原理变得简单易懂: 1、向AT24C02写入数据: 有个关系户,他想安排自己的儿子进某个大厦里某个楼层的公司,那么他就要先找到这个公司的地址,然后再找到该公司是第几楼,最…...
【Qt】Qt + Modbus 服务端学习笔记
《Qt Modbus 服务端学习笔记》 1.因为项目的需要,要写一个modbus通信,csdn上感觉有些回答,代码是人工智能生成的,有些细节不对。我这个经过实测,是可以直接用的。 首先要包含Qt 的相关模块 Qt Modbus 模块主要包含以…...
抖音用户视频批量下载工具开发全解析
一、逆向工程原理剖析 1.1 抖音Web端防护体系 抖音采用五层防御机制保护数据接口: graph LRA[浏览器指纹检测] --> B[请求参数签名]B --> C[Cookie动态验证]C --> D[请求频率限制]D --> E[IP信誉评级] 1.2 核心参数解密 参数名称作用原理生成方式有效期x-bogu…...
DeepSeek写打台球手机小游戏
DeepSeek写打台球手机小游戏 提问 根据提的要求,让DeepSeek整理的需求,进行提问,内容如下: 请生成一个包含以下功能的可运行移动端打台球小游戏H5文件: 要求 可以重新开始游戏 可以暂停游戏 有白球和其他颜色的球&am…...
清晰易懂的 Swift 安装与配置教程
初学者也能看懂的 Swift 安装与配置教程 本教程将手把手教你如何在 macOS 系统上安装 Swift,配置依赖包缓存位置,并指出新手容易踩坑的细节。即使你是零基础小白,也能快速上手! 一、安装 Swift(macOS 环境)…...
Post-Training Quantization, PTQ
Post-Training Quantization(PTQ) 是 模型训练完成后,对其参数(权重 & 激活值)进行量化 的方法,目的是 减少存储占用 & 提高推理速度,同时尽可能保持模型精度。 相比于 量化感知训练&a…...
linux Redhat9.5采用DNS主从实现跨网段解析
文章目录 主从服务器DNS实现跨网段解析一、服务器规划二、主服务器配置1、安装bind2、修改主配置文件3、配置区域配置文件4、配置正向解析文件5、配置反向解析文件6、检查并启动服务 三、从服务器配置1、安装bind2、配置主配置文件3、修改区域配置文件4、检查并启动服务 四、路…...
Python个人学习笔记(18):模块(异常处理、traceback、日志记录)
七、异常处理 语法错误不属于异常,处理的是程序运行时的一些意外情况 代码: a int(input(>>>:)) b int(input(>>>:)) print(a / b) # 在运行的时候由于数据不对,导致出错 # 此时程序会中断 prin…...
记一次MyBatis分页莫名其妙的失效,首次执行合适,后续执行分页失效且异常
代码几乎一样,为啥这个xml配置的就会出现莫名其妙的问题呢 org.mybatis.spring.MyBatisSystemException: nested exception is org.apache.ibatis.type.TypeException: Could not set parameters for mapping: ParameterMapping{propertymybatis_plus_first, modeI…...
Claude是什么?适合哪些场景?
Claude 是由人工智能公司 Anthropic 开发的一款 大型语言模型(LLM),旨在通过自然语言交互帮助用户完成复杂任务。以下是关于 Claude 的核心信息: 1. 核心定位 • 安全可靠: 采用 Constitutional AI(宪法AI…...
基于yolov11的持刀检测系统python源码+pytorch模型+评估指标曲线+精美GUI界面
【算法介绍】 基于YOLOv11的持刀检测系统 随着公共安全问题的日益突出,特别是在公共场所如机场、车站、学校等地,持刀等危险行为频发,对人们的生命财产安全构成严重威胁。传统的监控手段往往依赖于人工观察,但这种方式不仅效率低…...
openEuler24.03 LTS下安装Hive3
目录 前提条件 安装MySQL 卸载原有mysql及mariadb 下载mysql 解压mysql 安装mysql 启动mysql服务 开机自启动mysql服务 登录mysql 修改mysql密码 远程连接mysql 安装Hive 下载安装包 解压 设置环境变量 解决日志包冲突 将mysql驱动拷贝到lib目录 配置Hive 创…...
13-动态规划-最长公共子序列
题目 来源 24. 最长公共子序列 思路 不想打字,援引自最长公共子序列 (LCS) 详解例题模板(全)-CSDN博客 图示举例: 其余详见代码 代码 #include<bits/stdc.h> using namespace std; const int N110; int f[N][N]; int m…...
golang 生成单元测试报告
在 Go 语言中,你可以使用 go test 生成单元测试报告。以下是几种方法: 1. 生成基本测试报告(文本格式) go test -v ./... > test_report.txt-v:显示详细的测试信息./...:递归测试所有子目录> test_r…...
让AI看见世界:MCP协议与服务器的工作原理
让AI看见世界:MCP协议与服务器的工作原理 MCP(Model Context Protocol)是一种创新的通信协议,旨在让大型语言模型能够安全、高效地与外部资源进行交互。在AI技术快速发展的今天,MCP正成为连接AI与现实世界的重要桥梁。…...
设备健康管理系统是什么,设备健康管理系统多少钱?
想象一下,你的汽车在仪表盘报警前 3 天,手机就收到 “发动机轴承剩余寿命 1500 公里” 的提醒 —— 这就是 ** 设备健康管理系统(EHM)** 的日常。在制造业,设备故障每年造成全球 3.4 万亿美元损失,而 80% 的…...
设计模式(创建型)-抽象工厂模式
摘要 在软件开发的复杂世界中,设计模式作为解决常见问题的最佳实践方案,一直扮演着至关重要的角色。抽象工厂模式,作为一种强大的创建型设计模式,在处理创建一系列或相关依赖对象的场景时,展现出了独特的优势和灵活性。它通过提供一个创建对象的接口,让开发者能够在不指定…...
docker 部署elk 设置账号密码
1. 先把 kibana 停掉 2.进入es 容器 docker exec -it 75895a078cbc /bin/bash 找到 bin 目录 执行 ./elasticsearch-setup-passwords interactive 全部设置一样的密码 ,不一样自己要记住,设置成功会输出如下内容 Changed password for user [apm_system] Chang…...
<table>内有两行<tr>,第一行设定高度为60,剩余第二行,和右侧元素高度补齐。
实现 <table> 内第一行高度设定为 60px,第二行和右侧元素高度补齐的效果,你可以通过 CSS 样式来控制。示例: 为第一行 <tr> 设置固定高度 60px。对于右侧元素,假设它是一个 <div> 或者其他容器,将其…...
QT5.15.2加载pdf为QGraphicsScene的背景
5.15.2使用pdf 必须要安装QT源码,可以看到编译器lib目录已经有pdf相关的lib文件,d是debug 1.找到源码目录:D:\soft\QT\5.15.2\Src\qtwebengine\include 复制这两个文件夹到编译器的包含目录中:D:\soft\QT\5.15.2\msvc2019_64\include 2.找…...
常见的工具和技术
Mockito Mockito 是一个流行的 Java Mocking 框架,用于创建和配置模拟对象(Mock Objects),以便在单元测试中模拟复杂依赖关系的行为 使用场景:单元测试、隔离测试 EclipseEclipse 是一个开源的集成开发环境(…...
Linux怎样源码安装Nginx
1. 安装必要的依赖 在编译 Nginx 之前,你需要安装一些必要的依赖包,像编译工具和库文件等。以 CentOS 系统为例,可借助yum命令来安装: bash sudo yum install -y gcc pcre-devel zlib-devel openssl-devel要是使用的是 Ubuntu 系…...
汇编指令(20250319)
SOC常用总线 AHB(Advanced High-performance Bus):先进高性能总线,连接RAM,ROM等高速设备APB(Advanced Peripheral Bus):先进外设总线,连接外设等一些低速设备 CISC和R…...
PlainUSR|LIA: 追求更快的卷积网络实现高效的超分辨率重建
PlainUSR|LIA: 追求更快的卷积网络实现高效的超分辨率重建 引言 在深度学习领域,图像处理始终是一个热门话题。而超分辨率重建(Super-Resolution Reconstruction, SR)作为其中一个重要的研究方向,旨在通过算法将低分辨率图像恢复…...
神经网络基础之正则化
引言:正则化 (Regularization) 是机器学习中一种用于防止模型过拟合技术。核心思想是通过在模型损失函数中添加一个惩罚项 (Penalty Term),对模型的复杂度进行约束,从而提升模型在新数据上的泛化…...
JAVA序列化与反序列化URLDNS链CC1链
1、序列化的实现 java序列化的是对象属性的,只有实现了Serializable或者Externalizable接口的类的对象才能被序列化为字节序列。(不是则会抛出异常),静态成员变量是属于类的,所以静态成员变量是不能被序列化的&#x…...
孤儿进程和僵尸进程
本文讲述了什么是孤儿进程和僵尸进程,会带来怎样的问题以及如何处理 一、孤儿进程 1)什么是孤儿进程 孤儿进程,顾名思义,就是父进程提前终止,但是子进程还在运行中,父进程无法对子进程进行监管&#…...
用ASCII字符转化图片
代码 from PIL import Image# 定义 ASCII 字符集,从最暗到最亮 ASCII_CHARS "%#*-:. "def resize_image(image, new_width100):width, height image.sizeratio height / widthnew_height int(new_width * ratio)resized_image image.resize((new_wi…...
从WebRTC到嵌入式:EasyRTC如何借助大模型提升音视频通信体验
随着人工智能技术的快速发展,WebRTC与大模型的结合正在为音视频通信领域带来革命性的变革。WebRTC作为一种开源实时通信技术,以其低延迟、跨平台兼容性和强大的音视频处理能力,成为智能硬件和物联网设备的重要技术支撑。 而EasyRTC作为基于W…...
自动化测试工具-Playwright介绍和快速实例
Playwright 是什么 Playwright 是由 Microsoft 开发的开源自动化测试工具,专为现代 Web 应用设计。它支持 Chromium、Firefox 和 WebKit 内核的浏览器,能够跨平台(Windows、macOS、Linux)运行,提供强大的浏览器自动化能力,适用于测试、爬虫和监控等场景。 Playwright的…...
Java:Apache HttpClient中HttpRoute用法的介绍
当使用Apache HttpClient组件时,经常会用到它的连接池组件。典型的代码如下: PoolingHttpClientConnectionManager connectionManager new PoolingHttpClientConnectionManager();connectionManager.setMaxTotal(httpConfig.getMaxPoolTotal());connect…...
万字长文详解|蚂蚁数据湖深度探索与业务应用实践
作者介绍:黄超傑,蚂蚁数据智能部实时数据团队数据研发工程师,致力于数据湖技术在蚂蚁落地应用、蚂蚁广告实时数仓建设、蚂蚁数据成本治理等工作。 在开始之前 如果: 你想要提升数据时效,能够接受分钟级延迟你好奇准实…...
rk3568 以太网eth1 , 定制板 phy调试问题。
问题: 客户定制底板的 phy 网络不通。 解决逻辑: phy 问题, 就是 照着 公司底板对硬件就行,完全是硬件问题,不用改 软件。 原理图如下:...
React相关面试题
以下是150道React面试题及其详细回答,涵盖了React的基础知识、组件、状态管理、路由、性能优化等多个方面,每道题目都尽量详细且简单易懂: React基础概念类 1. 什么是React? React是一个用于构建用户界面的JavaScript库ÿ…...
IT工具 | node.js 进程管理工具 PM2 大升级!支持 Bun.js
P(rocess)M(anager)2 是一个 node.js 下的进程管理器,内置负载均衡,支持应用自动重启,常用于生产环境运行 node.js 应用,非常好用👍 🌼概述 2025-03-15日,PM2发布最新版本v6.0.5,这…...
php 高性能,高并发,有哪些框架,扩展,推荐一下,或者技术的实现有哪些
以下是针对PHP高性能、高并发场景的框架、扩展及技术实现推荐,结合最新技术趋势和行业实践进行总结: 一、高性能框架推荐 1. C扩展类框架 YAF (Yet Another Framework) 特点:由C语言编写,直接嵌入PHP内核,仅提供核心M…...
golang单机锁实现
1、锁的概念引入 首先,为什么需要锁? 在并发编程中,多个线程或进程可能同时访问和修改同一个共享资源(例如变量、数据结构、文件)等,若不引入合适的同步机制,会引发以下问题: 数据竞…...
面试中JVM常被问到的问题以及对应的答案
在面试中,关于JVM常被问到的问题以及对应的答案可能包括: 什么是JVM?它的作用是什么? 答:JVM是Java虚拟机的缩写,是Java程序运行的环境。它负责将Java源代码编译成字节码并运行在不同平台上。 请解释一下J…...