应使用ArrayList替代硬编码数组,Contact需重写equals()和hashCode(),遍历时删除须用Iterator.remove()或倒序循环,搜索用stream().filter(),存储优选JSON或CSV而非ObjectOutputStream。

用 ArrayList 存通讯录对象,别用数组硬编码
硬写 String[] names 或 Contact[] list 会导致增删困难、长度固定。真实场景下,联系人数量动态变化,必须用可变集合。直接上 ArrayList,Contact 是你自己定义的类,含 name、phone、email 等字段。
注意点:
-
Contact类必须重写equals()和hashCode(),否则list.remove(contact)可能失效(默认比较引用) - 不要在遍历
ArrayList时直接调用remove(),会抛ConcurrentModificationException;改用Iterator.remove()或倒序 for 循环 - 初始化别写
new ArrayList(100)—— 容量设太大浪费内存,设太小频繁扩容(每次 1.5 倍)影响性能;默认构造即可
按姓名查联系人?优先用 stream().filter(),不是手写 for
用户输入“张三”,你要返回所有匹配的 Contact。别再写三层 for 循环加 if 判断——Java 8+ 就该用流式处理,语义清晰且不易出错。
示例代码片段:
立即学习“Java免费学习笔记(深入)”;
list.stream()
.filter(c -> c.getName().contains(input))
.collect(Collectors.toList());
关键细节:
- 用
contains()而非equals(),支持模糊搜索;若要精确匹配,换成c.getName().equals(input) - 如果搜索频繁且数据量大(>1000 条),考虑提前建
Map做索引,但得同步维护一致性> -
stream()是惰性求值,不触发collect()就不会执行;调试时别只写 filter 就以为有结果
保存到文件用 ObjectOutputStream?风险高,改用 JSON 或 CSV
新手常想“序列化整个 ArrayList 到文件”,但 ObjectOutputStream 生成的二进制格式不可读、跨 JDK 版本不兼容、反序列化有安全风险(如恶意 payload),纯属自找麻烦。
更务实的做法:
- 轻量需求(学生作业/本地小工具):用
Files.write()写 CSV,每行张三,13800138000,zhang@xx.com,用String.split(",")解析 - 稍正式点:引入
com.fasterxml.jackson.core:jackson-databind,调new ObjectMapper().writeValue(file, list)输出 JSON,结构清晰、易调试、方便后续扩展 - 绝对别把密码、token 等敏感字段放进序列化或明文文件里,哪怕只是练习项目
删除联系人后没刷新界面?本质是没触发视图更新逻辑
如果你用控制台模拟界面(比如打印全部联系人列表),删完忘了调 System.out.println(list),就会以为“删没成功”。这不是集合问题,是逻辑断点缺失。
真正要注意的是:
- 所有修改集合的操作(add/remove/set)之后,如果后续逻辑依赖最新状态,必须确保你操作的是同一个对象引用,而不是副本(比如方法参数传了
new ArrayList(original)) - 如果封装了
ContactManager类,删完应返回boolean success或抛明确异常(如ContactNotFoundException),别让调用方靠“没报错”来判断成功 - 涉及多线程(比如后台定时保存),
ArrayList不是线程安全的,得换Collections.synchronizedList()或CopyOnWriteArrayList,但后者写多读少时性能差
Contact 实例创建后有没有被正确加入集合?删除时是不是删了另一个新 new 出来的临时对象?这些隐式引用关系,比 for 循环少个括号还难 debug。










