统计字符频次用HashMap遍历toCharArray()最稳妥,注意大小写和空白符处理;单词统计先清洗标点再split("\s+")并过滤空串;文件读取用Files.lines()防OOM,注意编码和路径;输出用stream排序+String.format对齐。

统计字符串中每个字符出现次数
直接用 HashMap 遍历字符是最稳妥的方式,避免用 String.charAt() 配合 String.indexOf() 这类低效组合。注意区分大小写和空白符——如果需求是“忽略大小写”,得先调用 toLowerCase();如果要跳过空格、换行等,需显式判断 Character.isWhitespace(c)。
常见错误是把 char 当作 String 传给 put(),导致编译失败;或在循环里反复调用 map.get(c) == null 而不是用 getOrDefault(c, 0),多一次哈希查找。
Mapcount = new HashMap<>(); for (char c : text.toCharArray()) { if (Character.isWhitespace(c)) continue; count.put(c, count.getOrDefault(c, 0) + 1); }
按单词统计频次(不依赖正则)
用 String.split("\\s+") 简单但有隐患:它无法处理标点粘连,比如 "hello,world" 会被当做一个词。更健壮的做法是用 StreamTokenizer 或手动扫描——但对简单项目,先用 replaceAll("[^a-zA-Z0-9\\s]", " ") 清洗再分割更直观。
注意 split(" ") 和 split("\\s+") 的区别:前者只切空格,后者切所有空白符(包括制表、换行),且能自动合并连续空白;空字符串过滤必须加 .filter(s -> !s.isEmpty()),否则数组头尾可能有空项。
立即学习“Java免费学习笔记(深入)”;
String cleaned = text.replaceAll("[^a-zA-Z0-9\\s]", " ");
String[] words = cleaned.split("\\s+");
Map wordCount = new HashMap<>();
for (String w : words) {
if (!w.isEmpty()) {
wordCount.put(w.toLowerCase(), wordCount.getOrDefault(w.toLowerCase(), 0) + 1);
}
}
读取文件并统计时的编码与异常处理
用 Files.readAllLines(path, StandardCharsets.UTF_8) 比 FileReader 更安全,后者默认用系统编码,中文 Windows 下容易乱码。必须捕获 IOException,不能只写 throws 丢给上层——命令行工具没人接这个异常。
大文件别用 readAllLines(),会 OOM;改用 Files.lines(path) 返回 Stream,配合 try-with-resources 自动关闭。另外,路径含空格或中文时,确保传入的是 Path 对象而非裸字符串,避免 java.nio.file.InvalidPathException。
-
Paths.get("data.txt")比new File("data.txt").toPath()更推荐 - 统计前检查文件是否存在:
Files.exists(path),避免 FileNotFoundException - 空文件要单独处理,
lines.count()为 0 时别直接进统计逻辑
输出结果时保留排序与格式对齐
Java 默认 HashMap 不保证顺序,想按频次降序输出得转成 LinkedHashMap 或用 stream().sorted()。别用 Collections.sort(list) 手动排序 list 再遍历 map——效率低还容易索引错位。
控制台对齐靠 String.format("%-15s %d", word, count),其中 %-15s 表示左对齐、占15字符宽;数字用 %6d 右对齐更清晰。如果导出 CSV,记得对字段中的逗号、换行做转义,否则 Excel 打开会错列。
wordCount.entrySet().stream()
.sorted(Map.Entry.comparingByValue().reversed())
.limit(10)
.forEach(e -> System.out.println(String.format("%-20s %6d", e.getKey(), e.getValue())));
实际跑起来最常卡在文件编码和标点清洗这两步,尤其是测试用的文本从网页复制过来,藏着零宽空格或软连字符,length() 看着正常,split() 却分不出词。先用 text.codePoints().forEach(System.out::println) 打印码点,比猜快得多。










