Not a complete newbie, but I still don't understand everything about Regular expressions. I was trying to use Regex to strip out <p> tags and my first attempt
<p\s*.*>
was so greedy it caught the whole line
<p someAttributes='example'>SomeText</p>
I got it to work with
((.|\s)*?)
This seems like it should be just as greedy, can anyone help me understand why it isnt?
Trying to make this question as language non-specific as possible, but I was doing this wi开发者_如何转开发th ColdFusion's reReplaceNoCase if it makes a lot of difference.
The key difference is the *?
part, which creates a reluctant quantifier, and so it tries to match as little as possible. The standard quantifier *
is a greedy quantifier and tries to match as much as possible.
See e.g. Greedy vs. Reluctant vs. Possessive Quantifiers
As Seth Robertson noted, you might want to use a regex that does not depend on the greedy/reluctant behaviour. Indeed, you can write a possessive regex for best performance:
<p\s*+[^>]*+>
Here, \s*+
matches any number of white space, while [^>]*+
matches any number of characters except >
. Both quantifiers do not track back in case of a mismatch, which improves runtime in case of a mismatch, and for some regex implementations also in case of a match (because internal backtracking data can be omitted).
Note that, if there are other tags starting with <p
(didn't write HTML directly for a long time), you match these too. If you don't want that, use a regex like this:
<p(\s++[^>]*+)?>
This makes the whole section between <p
and >
optional.
Well, either regex will match absolutely anything, so the question is moot. Using non-greedy parser will probably come closer to what you want but still could have very unexpected results.
While you should not be matching html/xml with a RE, you probably want something like:
<p\s*([^>]*)>
Which would put any attributes of p into $1.
<p\s*.*>
Is looking for 'p', 0 or more spaces, 0 or more characters, '>' . The group "any character" contains '>', so the regular expression find the whole line.
精彩评论