
本文介绍如何从 php 中类似 `@loc = 1, @username = tom,` 的字符串格式中,准确提取键(如 loc、username)和对应值,并构造成关联数组,便于后续调用(如 `$vars['username']`)。
在实际开发中,我们偶尔会遇到结构不规范但具有一定规律的字符串数据——例如从某些旧系统、日志或自定义协议返回的 Header 字段,其内容并非标准 JSON 或序列化格式,而是以 @键 = 值, 形式拼接的文本。此时直接使用数组下标(如 $result['Header']['@USERNAME'])无法访问,因为 $result['Header'] 实际是一个纯字符串,而非嵌套数组。
正确的处理思路是:先解析字符串,再构造关联数组。推荐使用正则表达式配合 preg_match_all() 提取所有 @key = value 对,再用 array_combine() 绑定键与值:
// 假设 $result['Header'] 的值为:'@LOC = 1, @USERNAME = Tom,'
$headerStr = $result['Header'];
// 使用正则匹配所有 '@{key} = {value},' 模式(支持空格、忽略大小写)
preg_match_all('/@(\w+)\s*=\s*([^,]+?),/', $headerStr, $matches);
// $matches[1] 是所有键(如 ['LOC', 'USERNAME']),$matches[2] 是对应值(如 ['1', 'Tom'])
if (!empty($matches[1]) && !empty($matches[2])) {
$vars = array_combine($matches[1], array_map('trim', $matches[2]));
echo $vars['USERNAME']; // 输出:Tom
print_r($vars);
} else {
echo "未匹配到有效的键值对";
}✅ 关键说明:
- 正则 /@(\w+)\s*=\s*([^,]+?),/ 中:
- (\w+) 捕获键名(仅字母数字下划线,更安全);
- \s*=\s* 匹配等号及前后任意空格;
- ([^,]+?) 非贪婪捕获值(直到逗号前),避免跨字段误匹配;
- array_map('trim', ...) 清除值两端空格,提升健壮性;
- 务必检查 $matches 是否非空,防止 array_combine() 报 Warning(参数数量不一致)。
⚠️ 注意事项:
立即学习“PHP免费学习笔记(深入)”;
- 若值中可能含逗号(如 @NOTE = "Hello, world",),需改用更复杂的正则(如支持引号包裹)或考虑 str_getcsv() 配合预处理;
- 不建议用 eval() 或 parse_str() 解析此类字符串,存在严重安全风险;
- 如该 Header 格式固定且高频使用,可封装为复用函数:
function parseHeaderString(string $header): array {
preg_match_all('/@(\w+)\s*=\s*([^,]+?),/', $header, $m);
return !empty($m[1]) ? array_combine($m[1], array_map('trim', $m[2])) : [];
}
// 调用:$data = parseHeaderString($result['Header']);通过该方法,你不仅能获取 @USERNAME,还能统一管理所有 Header 字段,代码清晰、可维护性强,且完全符合 PHP 最佳实践。











