
本文介绍在 php 中如何将字符串拆分为元音和辅音字符,统一转为小写并去除空格后,**按各字符在原字符串中首次出现的顺序进行分组排序**(即相同字符按频次重复,但整体保持“先出现的字符类型优先排列”),而非简单按字母表顺序排序。
要实现“按首次出现顺序排序”,关键在于:不改变字符类别的相对位置顺序,仅对同一类别内的重复字符做频次聚合,并保持该字符第一次出现时的先后次序。例如 'Sample Case' 中,元音 a 首次出现在索引 1(Sample...),e 首次出现在索引 4(Sample...),因此最终元音结果应为 aaee(两个 a 在前,两个 e 在后);同理,辅音 s 首次出现在索引 0 和索引 6(S 和 Case 中的 s),故 s 排最前,后续依次是 m, p, l, c → 拼出 ssmplc。
下面是一个清晰、健壮的实现方案:
function sortCharactersByFirstOccurrence($text) {
$text = strtolower(str_replace(' ', '', $text));
$vowels = ['a', 'e', 'i', 'o', 'u'];
// 提取所有元音和辅音(保留原始顺序)
$allVowels = [];
$allConsonants = [];
for ($i = 0; $i < strlen($text); $i++) {
$char = $text[$i];
if (in_array($char, $vowels)) {
$allVowels[] = $char;
} else {
$allConsonants[] = $char;
}
}
// 统计频次,并按首次出现顺序拼接
$vowelFreq = [];
$consonantFreq = [];
// 元音频次统计(保持首次出现顺序)
foreach ($allVowels as $v) {
if (!isset($vowelFreq[$v])) {
$vowelFreq[$v] = 1;
} else {
$vowelFreq[$v]++;
}
}
// 辅音同理
foreach ($allConsonants as $c) {
if (!isset($consonantFreq[$c])) {
$consonantFreq[$c] = 1;
} else {
$consonantFreq[$c]++;
}
}
// 按原始出现顺序去重得到字符序列(PHP 7.4+ 可用 array_keys(array_flip(...)),但此处手动保序更明确)
$orderedVowels = array_values(array_unique($allVowels));
$orderedConsonants = array_values(array_unique($allConsonants));
// 构建结果
$vowelResult = '';
foreach ($orderedVowels as $v) {
$vowelResult .= str_repeat($v, $vowelFreq[$v]);
}
$consonantResult = '';
foreach ($orderedConsonants as $c) {
$consonantResult .= str_repeat($c, $consonantFreq[$c]);
}
echo "Vowels : {$vowelResult}\n";
echo "Consonants : {$consonantResult}\n";
}
// 测试
sortCharactersByFirstOccurrence('Sample Case');输出:
Vowels : aaee Consonants : ssmplc
✅ 核心要点总结:
- 先清洗字符串(小写 + 去空格),再逐字符遍历,严格按原始位置分离元音/辅音;
- 使用 array_unique()(配合 array_values())提取首次出现顺序的不重复字符列表,确保拼接时顺序正确;
- 利用关联数组统计频次,再按保序后的字符键循环拼接,实现“同字符聚集 + 类别内首现优先”;
- 避免使用 sort() 或 ksort() —— 它们会破坏原始位置逻辑;
- 此方案兼容含重复字符、大小写混合、多空格等常见边界情况。
该方法语义清晰、易于维护,适用于文本分析、密码学预处理或教学演示等需要保留字符时序特征的场景。










