
本文介绍如何将字符串中的元音和辅音分别提取、转为小写并去除空格后,按**各字符在原字符串中首次出现的顺序**进行分组排序(即相同字符按频次重复,但整体顺序由首次出现位置决定)。
要实现“按首次出现顺序排序”,关键在于:不是简单地对字符数组调用 sort(),而是先统计每类字符(元音/辅音)的出现频次,再按它们在原始字符串中第一次出现的先后顺序依次拼接重复字符。
原代码的问题在于:它只是按遍历顺序收集字符($v 和 $c 是按出现顺序追加),但未对同类字符做归并排序——例如 's' 在 "Sample Case" 中首次出现在索引 1(S),第二次出现在索引 7(s),而 'm' 首次出现在索引 2;因此最终辅音序列应为 s(先)、m(后)、p、l、c,且每个字符重复其实际出现次数 → ssmplc。
下面是一个清晰、健壮的实现方案:
function sortCharactersByFirstOccurrence($text) {
$text = strtolower(str_replace(' ', '', $text));
$vowels = ['a', 'e', 'i', 'o', 'u'];
// 分离元音与辅音,并记录首次出现位置
$vowelChars = [];
$consonantChars = [];
$seenVowels = [];
$seenConsonants = [];
for ($i = 0; $i < strlen($text); $i++) {
$char = $text[$i];
if (in_array($char, $vowels)) {
if (!isset($seenVowels[$char])) {
$seenVowels[$char] = $i; // 记录首次出现索引
$vowelChars[] = $char;
}
} else {
if (!isset($seenConsonants[$char])) {
$seenConsonants[$char] = $i;
$consonantChars[] = $char;
}
}
}
// 统计总频次
$allChars = str_split($text);
$counts = array_count_values($allChars);
// 按首次出现顺序拼接(每个字符重复对应频次)
$vowelResult = '';
foreach ($vowelChars as $v) {
$vowelResult .= str_repeat($v, $counts[$v] ?? 0);
}
$consonantResult = '';
foreach ($consonantChars as $c) {
$consonantResult .= str_repeat($c, $counts[$c] ?? 0);
}
echo "Vowels : {$vowelResult}\n";
echo "Consonants : {$consonantResult}\n";
}
// 测试
sortCharactersByFirstOccurrence("Sample Case");输出:
立即学习“PHP免费学习笔记(深入)”;
Vowels : aaee Consonants : ssmplc
✅ 核心逻辑说明:
- 第一次遍历确定元音/辅音的首次出现顺序(存入 $vowelChars / $consonantChars);
- 利用 array_count_values() 获取每个字符的全局出现次数;
- 最后按首次顺序循环拼接,用 str_repeat() 补足重复次数。
⚠️ 注意事项:
- 此方法严格区分大小写处理(已统一转小写),不依赖 ASCII 排序;
- 空格与非字母字符(如标点)默认被 str_replace(' ', '') 移除;如需支持其他过滤,可改用正则 preg_replace('/[^a-z]/', '', $text);
- 若需扩展支持 Unicode 元音(如带重音的 á, é),应使用 mb_ 系列函数并定义更完整的元音数组。
该方案兼顾可读性、可维护性与正确性,适用于教学及中小型文本处理场景。











