被正则玩疯了,为什么$matches[0]为空?

php中文网
发布: 2016-06-23 13:47:41
原创
1080人浏览过

写个正则,能正常匹配到,但我还需要他返回匹配的整条字符串

正则如下

preg_match( '/^<!--\sLayout\sname\s*=\s*"(.*)"\s-->/', $_content, $matches);
登录后复制


$_content的内容是
<!-- Layout name="header" --><!-- Layout name="footer" -->
登录后复制


我知道只能匹配到第一条,但根据官方文档写的: $matches[0]将包含完整模式匹配到的文本
但我测试了很久,只能得到以下结果
Array(    [0] =>     [1] => header)
登录后复制

$matches[0]为什么会为空?说好的匹配到的文本呢?(我原意是想$matches[0]值为  的)

还有一个问题,不知道为什么断言后就匹配不到,
preg_match( '/^<!--\sLayout\sname\s*=\s*"(.*)"\s-->$/', $_content, $matches);
登录后复制

就这样,末尾加了个美元符,就匹配不到任何东西了,虽然这个不影响,但我还是想知道是为什么,看网上的文档,一堆 术语头都晕了,希望高人解答,谢谢。

晕,怎么只能给100分啊,我有1000分的。。。

音疯
音疯

音疯是昆仑万维推出的一个AI音乐创作平台,每日可以免费生成6首歌曲。

音疯 84
查看详情 音疯

回复讨论(解决方案)

$_content =<<< HTML<!-- Layout name="header" --><!-- Layout name="footer" -->HTML;preg_match( '/^<!--\sLayout\sname\s*=\s*"(.*)"\s-->/', $_content, $matches);print_r($matches);
登录后复制
Array(    [0] => <!-- Layout name="header" -->    [1] => header)
登录后复制
没有问题

只不过你是被你自己弄糊涂了
$matches[0] 是一个html标记,只在文本方式下才能看到

$_content='<!-- Layout name="header" --><!-- Layout name="footer" -->';preg_match_all('//', $_content, $matches);print_r($matches);/*Array(    [0] => Array        (            [0] =>             [1] =>         )    [1] => Array        (            [0] => header            [1] => footer        ))*/
登录后复制

$matches[0]将包含完整模式匹配到的文本,它后面还有一句: $matches[1]将包含第一个捕获子组匹配到的文本, 以此类推. 
意思应该是只包含你反捕获(也就是你的小括号括起来的内容),而不是执行的全局匹配,所以用preg_match_all来执行全局匹配
还有你的正则在最前面加了^,它表示你执行匹配的字符串必须已^之后的内容开始,所以是不能匹配到了
同理,$表示匹配的字符串必须以$之前的结尾,所以要匹配到内容,$_content='';或者$_content='‘;才能被匹配到

$_content =<<< HTML<!-- Layout name="header" --><!-- Layout name="footer" -->HTML;preg_match( '/^<!--\sLayout\sname\s*=\s*"(.*)"\s-->/', $_content, $matches);print_r($matches);
登录后复制
Array(    [0] => <!-- Layout name="header" -->    [1] => header)
登录后复制
没有问题

只不过你是被你自己弄糊涂了
$matches[0] 是一个html标记,只在文本方式下才能看到



我了个去!!!忘记了HTML的注释标记,晕,真的昏了一晚上啊,感谢提醒。

$_content='<!-- Layout name="header" --><!-- Layout name="footer" -->';preg_match_all('//', $_content, $matches);print_r($matches);/*Array(    [0] => Array        (            [0] =>             [1] =>         )    [1] => Array        (            [0] => header            [1] => footer        ))*/
登录后复制

$matches[0]将包含完整模式匹配到的文本,它后面还有一句: $matches[1]将包含第一个捕获子组匹配到的文本, 以此类推. 
意思应该是只包含你反捕获(也就是你的小括号括起来的内容),而不是执行的全局匹配,所以用preg_match_all来执行全局匹配
还有你的正则在最前面加了^,它表示你执行匹配的字符串必须已^之后的内容开始,所以是不能匹配到了
同理,$表示匹配的字符串必须以$之前的结尾,所以要匹配到内容,$_content='';或者$_content='‘;才能被匹配到



还是不太懂,是不是用了 ^ 之后,只能从 $content[0] 开始比较?前面的字串相同就进行匹配, $ 同理?

还有一个问题希望能解决一下

存在一字符串

$string = 'bbs/csdn/net/xxx';
登录后复制


如果字符串不是以bbs开头的就进行匹配
比如 
$string = 'www/csdn/net/xxx';
登录后复制

因为不以 bbs 开头,继续匹配

我想到一种写法
preg_match( '#((?!bbs/).)*#', 'www/csdn/net/xxx/', $matches);
登录后复制

但感觉这个效率有点低下,不知道还能不能优化一下?

^是匹配输入字符串的开始位置,若待匹配的字符串不是以^之后的内容开始,根本就不会执行匹配

若是只是判断 字符串不是以bbs开头,这样比较快

$str='/bbs/www/csdn/net/xxx/';if(strpos($str,'bbs')!=0 || strpos($str,'bbs')===false){	//等于0就是以bbs开头,未找到返回false(全等于false)	echo 'exe';}
登录后复制

^是匹配输入字符串的开始位置,若待匹配的字符串不是以^之后的内容开始,根本就不会执行匹配

若是只是判断 字符串不是以bbs开头,这样比较快

$str='/bbs/www/csdn/net/xxx/';if(strpos($str,'bbs')!=0 || strpos($str,'bbs')===false){	//等于0就是以bbs开头,未找到返回false(全等于false)	echo 'exe';}
登录后复制



不是,其实我想写个类似django这样的URL路由功能,如果按上面的方法的话,扩展性不够强,所以想用正则表达

$str='www/bbs/net/xxx/';if(preg_match('/^bbs.+?/', $str)){	echo '以bbs开头';}else{	echo 'exe';}
登录后复制

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号