
本文介绍如何从 php 中类似 @loc = 1, @username = tom, 的非标准字符串格式中,精准提取键值对(如 username → "tom"),并转换为关联数组,适用于解析自定义协议头、日志标记或遗留系统返回的类 ini 风格文本。
在实际开发中,我们偶尔会遇到结构不规范的字符串数据——例如 $result['Header'] 返回的并非标准数组或 JSON,而是一段形如 header @LOC = 1, @USERNAME = Tom, 的混合文本。此时直接使用数组下标(如 $result['Header']['@USERNAME'])会失败,因为该值本质上是字符串,而非嵌套数组。
正确的处理思路是:先解析字符串,再结构化为关联数组。推荐使用正则表达式配合 preg_match_all() 提取所有 @key = value 模式,再通过 array_combine() 组装键值对:
// 假设 $result['Header'] 的值为:"header @LOC = 1, @USERNAME = Tom,"
$headerStr = trim($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])) {
$headerData = array_combine($matches[1], array_map('trim', $matches[2]));
// 现在可安全访问:
echo $headerData['USERNAME']; // 输出:Tom
echo $headerData['LOC']; // 输出:1
} else {
$headerData = [];
error_log("Warning: Failed to parse Header string.");
}✅ 关键说明:
- 正则 /@(\w+)\s*=\s*([^,]+?),/ 中,\w+ 匹配字母/数字/下划线(更安全替代 \S+,避免截断含空格的值);[^,]+? 非贪婪匹配值直到逗号前;\s* 容忍等号两侧空格。
- array_map('trim', $matches[2]) 清除值两端空格,提升健壮性。
- 务必检查 $matches 是否非空,防止 array_combine() 因空数组报错(PHP Warning)。
⚠️ 注意事项:
立即学习“PHP免费学习笔记(深入)”;
- 若值中可能包含逗号(如 @NOTE = "Hello, world",),上述正则将失效,此时需升级为支持引号包裹的解析器(如使用 str_getcsv() 预处理或 PCRE 的递归模式)。
- 不建议用 eval() 或 parse_ini_string() 直接解析——前者存在严重安全风险,后者要求严格 INI 格式(不支持 @ 前缀和逗号分隔)。
此方法轻量、无外部依赖,适用于绝大多数简单头信息解析场景,兼顾可读性与执行效率。











