Matching a repeated character in regex is simple with a backreference:
(.)\1
However, I would like to match the character after the pair of characters, so I thought I could simply put this in a lookbehind:
(?<=(.)\1).
Unfortunately, this doesn't match anything.
Why is that? In other flavours I wouldn't be surprised because there are strong restrictions on lookbehinds, but .NET usually supports arbitrarily complicated patterns inside lookbehinds.
The short version: Lookbehinds are matched from right to left. That means when the regex engine encounters the
\1
it hasn't captured anything into that group yet, so the regex always fails. The solution is quite simple:Test it here.
Unfortunately, the full story once you start using more complex patterns is a lot more subtle. So here is...
A guide to reading regular expressions in .NET
First, some important acknowledgements. The person who taught me that lookbehinds are matched from right to left (and figured this out on his own through a lot of experimentation), was Kobi in this answer. Unfortunately, the question I asked back then was a very convoluted example which doesn't make for a great reference for such a simple problem. So we figured it would make sense to make a new and more canonical post for future reference and as a suitable dupe target. But please consider giving Kobi an upvote for figuring out a very important aspect of .NET's regex engine that is virtually undocumented (as far as I know, MSDN mentions it in a single sentence on a non-obvious page).
Note that rexegg.com explains the inner workings of .NET's lookbehinds differently (in terms of reversing the string, the regex, and any potential captures). Although that wouldn't make a difference to the result of the match, I find that approach much harder to reason about, and from looking at the code it's fairly clear that this is not what the implementation actually does.
So. The first question is, why is it actually more subtle than the bolded sentence above. Let's try matching a character that is preceded by either
a
orA
using a local case-insensitive modifier. Given the right-to-left matching behaviour, one might expect this to work:However, as you can see here this doesn't seem to use the modifier at all. Indeed, if we put the modifier in front:
...it works.
Another example, that might be surprising with right-to-left matching in mind is the following:
Does the
\2
refer to the left or right capturing group? It refers to the right one, as this example shows.A final example: when matched against
abc
, does this captureb
orab
?It captures
b
. (You can see the captures on the "Table" tab.) Once again "lookbehinds are applied from right to left" isn't the full story.Hence, this post tries to be a comprehensive reference on all things regarding directionality of regex in .NET, as I'm not aware of any such resource. The trick to reading a complicated regex in .NET is doing so in three or four passes. All but the last pass are left-to-right, regardless of lookbehinds or
RegexOptions.RightToLeft
. I believe this is the case, because .NET processes these when parsing and compiling the regex.First pass: inline modifiers
This is basically what the above example shows. If anywhere in your regex, you had this snippet:
Regardless of where in the pattern that is or whether you're using the RTL option,
c
will be case-insensitive whilea
,b
andd
will not (provided they aren't affected by some other preceding or global modifier). That is probably the simplest rule.Second pass: group numbers [unnamed groups]
For this pass you should completely ignore any named groups in the pattern, i.e. those of the form
(?<a>...)
. Note this does not include groups with explicit numbers like(?<2>...)
(which are a thing in .NET).Capturing groups are numbered from left to right. It doesn't matter how complicated your regex is, whether you're using the RTL option or whether you nest dozens of lookbehinds and lookaheads. When you're only using unnamed capturing groups, they are numbered from left to right depending on the position of their opening parenthesis. An example:
This gets a bit trickier when mixing unlabelled groups with explicitly numbered groups. You should still read all of these from left to right, but the rules are a bit trickier. You can determine the number of a group as follows:
(?<1>.)(?<5>.)
is a perfectly valid regex with group number2
to4
unused.Here is an example (without nesting, for simplicity; remember to order them by their opening parentheses when they are nested):
Notice how the explicit group
6
creates a gap, then the group capturingg
takes that unused gap between groups4
and6
, whereas the group capturingh
takes7
because6
is already used. Remember that there might be named groups anywhere in between these, which we're completely ignoring for now.If you're wondering what the purpose of repeated groups like group
1
in this example is, you might want to read about balancing groups.Third pass: group numbers [named groups]
Of course, you can skip this pass entirely if there are no named groups in the regex.
It's a little known feature that named groups also have (implicit) group numbers in .NET, which can be used in backreferences and substitution patterns for
Regex.Replace
. These get their numbers in a separate pass, once all the unnamed groups have been processed. The rules for giving them numbers are as follows:A more complete example with all three types of groups, explicitly showing passes two and three:
Final pass: following the regex engine
Now that we know which modifiers apply to which tokens and which groups have which numbers, we finally get to the part that actually corresponds to the execution of the regex engine, and where we start going back and forth.
.NET's regex engine can process regex and string in two directions: the usual left-to-right mode (LTR) and its unique right-to-left mode (RTL). You can activate RTL mode for the entire regex with
RegexOptions.RightToLeft
. In that case, the engine will start trying to find a match at the end of the string and will go left through the regex and the string. For example, the simple regexWould match a
b
, then it would try to match.*
to the left of that (backtracking as necessary) such that there's ana
somewhere to the left of it. Of course, in this simple example, the result between LTR and RTL mode is identical, but it helps to make a conscious effort to follow the engine in its backtracking. It can make a difference for something as simple as ungreedy modifiers. Consider the regexinstead. We're trying to match
axxbxxb
. In LTR mode, you get the matchaxxb
as expected, because the ungreedy quantifier is satisfied with thexx
. However, in RTL mode, you'd actually match the entire string, since the firstb
is found at the end of the string, but then.*?
needs to match all ofxxbxx
fora
to match.And clearly it also makes a difference for backreferences, as the example in the question and at the top of this answer shows. In LTR mode we use
(.)\1
to match repeated characters and in RTL mode we use\1(.)
, since we need to make sure that the regex engine encounters the capture before it tries to reference it.With that in mind, we can view lookarounds in a new light. When the regex engine encounters a lookbehind, it processes it as follows:
x
in the target string as well as its current processing direction.x
.x
and the original processing direction is restored.While a lookahead seems a lot more innocuous (since we almost never encounter problems like the one in the question with them), its behaviour is actually virtually the same, except that it enforces LTR mode. Of course in most patterns which are LTR only, this is never noticed. But if the regex itself is matched in RTL mode, or we're doing something as crazy as putting a lookahead inside a lookbehind, then the lookahead will change the processing direction just like the lookbehind does.
So how should you actually read a regex that does funny stuff like this? The first step is to split it into separate components, which are usually individual tokens together with their relevant quantifiers. Then depending on whether the regex is LTR or RTL, start going from top to bottom or bottom to top, respectively. Whenever you encounter a lookaround in the process, check which way its facing and skip to the correct end and read the lookaround from there. When you're done with the lookaround, continue with the surrounding pattern.
Of course there's another catch... when you encounter an alternation
(..|..|..)
, the alternatives are always tried from left to right, even during RTL matching. Of course, within each alternative, the engine proceeds from right to left.Here is a somewhat contrived example to show this:
And here is how we can split this up. The numbers on the left show the reading order if the regex is in LTR mode. The numbers on the right show the reading order in RTL mode:
I sincerely hope that you'll never use something as crazy as this in production code, but maybe one day a friendly colleague will leave some crazy write-only regex in your company's code base before being fired, and on that day I hope that this guide might help you figure out what the hell is going on.
Advanced section: balancing groups
For the sake of completeness, this section explains how balancing groups are affected by the directionality of the regex engine. If you don't know what balancing groups are, you can safely ignore this. If you want to know what balancing groups are, I've written about it here, and this section assumes that you know at least that much about them.
There are three types of group syntax that are relevant for balancing groups.
(?<a>...)
or(?<2>...)
(or even implicitly numbered groups), which we've dealt with above.(?<-a>...)
and(?<-2>...)
. These behave as you'd expect them to. When they're encountered (in the correct processing order described above), they simply pop from the corresponding capture stack. It might be worth noting that these don't get implicit group numbers.(?<b-a>...)
which are usually used to capture the string since the last ofb
. Their behaviour gets weird when mixed with right-to-left mode, and that's what this section is about.The takeaway is, the
(?<b-a>...)
feature is effectively unusable with right-to-left mode. However, after a lot of experimentation, the (weird) behaviour actually appears to follow some rules, which I'm outlining here.First, let's look at an example which shows why lookarounds complicate the situation. We're matching the string
abcde...wvxyz
. Consider the following regex:Reading the regex in the order I presented above, we can see that:
fgh
into groupa
..{2}
moves two characters to the left.(?<b-a>.{3})
is the balancing group which pops the capture off groupa
and pushes something onto groupb
. In this case, the group matcheslmn
and we pushijk
onto groupb
as expected.However, it should be clear from this example, that by changing the numerical parameters, we can change the relative position of the substrings matched by the two groups. We can even make those substrings intersect, or have one contained completely inside the other by making the
3
smaller or larger. In this case it's no longer clear what it means to push everything between the two matched substrings.It turns out that there are three cases to distinguish.
Case 1:
(?<a>...)
matches left of(?<b-a>...)
This is the normal case. The top capture is popped from
a
and everything between the substrings matched by the two groups is pushed ontob
. Consider the following two substrings for the two groups:Which you might get with the regex
Then
mn
would be pushed ontob
.Case 2:
(?<a>...)
and(?<b-a>...)
intersectThis includes the case where the two substrings touch, but don't contain any common characters (only a common boundary between characters). This can happen if one of the groups is inside a lookaround and the other one is not or is inside a different lookaround. In this case the intersection of both subtrings will be pushed onto
b
. This is still true when substring is completely contained inside the other.Here are several examples to show this:
Case 3:
(?<a>...)
matches right of(?<b-a>...)
This case I don't really understand and would consider a bug: when the substring matched by
(?<b-a>...)
is properly left of the substring matched by(?<a>...)
(with at least one character between them, such that they don't share a common boundary), nothing is pushedb
. By that I really mean nothing, not even an empty string — the capture stack itself remains empty. However, matching the group still succeeds, and the corresponding capture is popped off thea
group.What's particularly annoying about this is that this case would likely be a lot more common than case 2, since this is what happens if you try to use balancing groups the way they were meant to be used, but in a plain right-to-left regex.
Update on case 3: After some more testing done by Kobi it turns out that something happens on stack
b
. It appears that nothing is pushed, becausem.Groups["b"].Success
will beFalse
andm.Groups["b"].Captures.Count
will be0
. However, within the regex, the conditional(?(b)true|false)
will now use thetrue
branch. Also in .NET it seems to be possible to do(?<-b>)
afterwards (after which accessingm.Groups["b"]
will throw an exception), whereas Mono throws an exception immediately while matching the regex. Bug indeed.