如何匹配开始和结束大括号,标签和指定属性内的文本

时间:2015-11-21 08:13:30

标签: php regex preg-match-all

我正在为我的CMS系统实现一个插件代码。类似于短代码的东西,但在许多场景中都适用。我想要一个管理员写这样的代码的案例:

示例1:

{COMMAND_NAME}Strings of texts that conatains htmltags,symbols,just anything{/COMMAND_NAME}

示例2

{COMMAND_NAME}

示例3

{COMMAND_NAME{attriute1=value attribute2=value}}

示例4

{COMMAND_NAME{attriute1=value attribute2=value}}Strings of anything including texts, htmltags and anything at all {/COMMAND_NAME}

正则表达式可以匹配上面的字符串。获取COMMAND_NAME,从中获取文本,并从单个正则表达式模式中获取结束{/COMMAND_NAME}

在正则表达式中,我想捕获COMMAND_NAME,提供的属性,{COMMAND_NAME}有结尾{/COMMAND_NAME}和结束{/COMMAND_NAME}之间的文本如果提供的话。

查看我到目前为止所做的工作并得出一些不完整的结果。

$regex = #\{(RAW|ACCESS|DWNLINK|MODL)[\{]{0,1}([\w\W\s]*?)\}{0}\}([\w\s]+)([\{/RAW|ACCESS|DWNLINK|MODL]*)\}#i

$strings = '<div class="blog-list-item blog"><header class="entry-title">
        <h1>Welcome to our website</h1>
    </header><article id="entry-72" class="entry post-72 page et-bg-layout-dark et-white-bg"><div class="jumbotron row">
<div class="col-md-8">
<ul>
<li>You have a pending job on your neck?&hellip;</li>
<li>Do your company need a website makeover ?&hellip;</li>
<li>Or a competitive web application ? ?&hellip;</li>
<li>Do you need a customized plugin, or a tweak ?&hellip;</li>
<li>Maybe you want a personal website ?&hellip;</li>
<li>Or a graphic for your new project ?&hellip;</li>
</ul>
<div class="bg-primary well">
<h4 class="text-center text-white shadow">Track your project as we work it         to perfection...</h4>
</div>
</div>
<div class="pull-right col-md-4">
<h4 class="bg-primary text-white well">Other services we offer</h4>
{ACCESS{type=500}}
<ul>
<li>SEO work for an existing website or new</li>
<li>Bulk SMS</li>
<li>E-currency exchange</li>
<li>Facebook AD</li>
<li>Google AD</li>
</ul>
{/ACCESS}</div>
{RAW{say=email,access=500}} {RAW} <a class="btn button large tall green"     href="client-area">Place new Job now as we deliver at the quickest   <em>reasonable time</em></a>{/RAW}</div></article></div>';

And doing a php var_dump, gives the following result:
array(5) {
  [0]=>
  array(1) {
    [0]=>
    string(224) "{ACCESS{type=500}}
<ul>
<li>SEO work for an existing website or new</li>
<li>Bulk SMS</li>
<li>E-currency exchange</li>
<li>Facebook AD</li>
<li>Google AD</li>
</ul>
{/ACCESS}</div>
{RAW{say=email,access=500}} {RAW}"
  }
  [1]=>
  array(1) {
    [0]=>
    string(6) "ACCESS"
  }
  [2]=>
  array(1) {
    [0]=>
    string(209) "type=500}}
<ul>
<li>SEO work for an existing website or new</li>
<li>Bulk SMS</li>
<li>E-currency exchange</li>
<li>Facebook AD</li>
<li>Google AD</li>
</ul>
{/ACCESS}</div>
{RAW{say=email,access=500}"
  }
  [3]=>
  array(1) {
    [0]=>
    string(1) " "
  }
  [4]=>
  array(1) {
    [0]=>
    string(4) "{RAW"
  }
}

这实际上不是我需要检索的内容。 我再次想要捕获COMMAND_NAME,仅在提供的情况下捕获属性,如果{COMMAND_NAME}具有结束{/COMMAND_NAME}则关闭文本,如果提供则关闭{/COMMAND_NAME}。这意味着该命令可以是内联{COMMAND_NAME},也可以不是{COMMAND_NAME}某些字符串{/COMMAND_NAME},或者具有属性{COMMAND_NAME{attr1=value attr2=value2}}

1 个答案:

答案 0 :(得分:1)

此正则表达式将按您的指定工作:

$regex = '~

#opening tag
\{(RAW|ACCESS|DWNLINK|MODL|\w+)
 #optional attributes
 (?>
     \{   ([^}]*)   }
 )?

}


#optional text and closing tag
(?:
    (   #text:= any char except "{", or a "{" not followed by /commandname
        [^{]*+
        (?>\{(?!/?\1[{}])[^{]*)*+
    )

    #closing tag
    (   \{/\1}   )
)?

~ix';

regex101 demo

与您所拥有的相比:

首先,我使用了/x修饰符(在结尾处),它忽略了空格和#comments

在开始标记中,我使用了您的选项,但您也可以使用\w+来匹配任何命令名称:

\{(RAW|ACCESS|DWNLINK|MODL|\w+)

对于可选属性,您有[\{]{0,1}([\w\W\s]*?)\}{0},这是尝试使每个部分都可选的无效尝试。相反,我使用(?> group )?(请参阅non-capturing groupsatomic groups)使整个子模式可选(使用? quantifier)。< / p>

 (?>
     \{   ([^}]*)   }
 )?

对text和closing标签应用相同的逻辑,使其成为可选项。

您使用[\w\s]+来匹配与单词字符和空格匹配的文本,但无法匹配标点符号和其他字符。我本可以使用.*?,它可以正常工作。但是,我使用了以下构造,它匹配相同,但表现更好:

    (   #text:= any char except "{", or a "{" not followed by /commandname
        [^{]*+
        (?>\{(?!/?\1[{}])[^{]*)*?
    )

最后,我使用\1匹配结束标记,这是对第1组中匹配的文本(开头标记名称)的反向引用:

\{/\1}

假设:

  • 某个属性在"te}xt"等引号中没有右括号可能会使其中断。