Escaping metacharacters
This chapter will show how to match metacharacters literally. Examples will be discussed for both manually as well as programmatically constructed patterns. You'll also learn about escape sequences supported by the re
module.
Escaping with backslash
You have seen a few metacharacters and escape sequences that help to compose a RE. To match the metacharacters literally, i.e. to remove their special meaning, prefix those characters with a \
(backslash) character. To indicate a literal \
character, use \\
. This assumes you are using raw strings and not normal strings.
# even though ^ is not being used as anchor, it won't be matched literally
>>> bool(re.search(r'b^2', 'a^2 + b^2 - C*3'))
False
# escaping will work
>>> bool(re.search(r'b\^2', 'a^2 + b^2 - C*3'))
True
# match ( or ) literally
>>> re.sub(r'\(|\)', '', '(a*b) + c')
'a*b + c'
# note that the input string is also a raw string here
>>> re.sub(r'\\', '/', r'\learn\by\example')
'/learn/by/example'
As emphasized earlier, regular expressions is just another tool to process text. Some examples and exercises presented in this book can be solved using normal string methods as well. It is a good practice to reason out whether regular expressions is needed for a given problem.
>>> eqn = 'f*(a^b) - 3*(a^b)'
# straightforward search and replace, no need RE shenanigans
>>> eqn.replace('(a^b)', 'c')
'f*c - 3*c'
re.escape()
Okay, what if you have a string variable that must be used to construct a RE — how to escape all the metacharacters? Relax, the re.escape()
function has got you covered. No need to manually take care of all the metacharacters or worry about changes in future versions.
>>> expr = '(a^b)'
# print used here to show results similar to raw string
>>> print(re.escape(expr))
\(a\^b\)
# replace only at the end of string
>>> eqn = 'f*(a^b) - 3*(a^b)'
>>> re.sub(re.escape(expr) + r'\Z', 'c', eqn)
'f*(a^b) - 3*c'
Recall that in the Alternation section, join
was used to dynamically construct RE pattern from an iterable of strings. However, that didn't handle metacharacters. Here are some examples on how you can use re.escape()
so that the resulting pattern will match the strings from the input iterable literally.
# iterable of strings, assume alternation precedence sorting isn't needed
>>> terms = ['a_42', '(a^b)', '2|3']
# using 're.escape' and 'join' to construct the pattern
>>> pat1 = re.compile('|'.join(re.escape(s) for s in terms))
# using only 'join' to construct the pattern
>>> pat2 = re.compile('|'.join(terms))
>>> print(pat1.pattern)
a_42|\(a\^b\)|2\|3
>>> print(pat2.pattern)
a_42|(a^b)|2|3
>>> s = 'ba_423 (a^b)c 2|3 a^b'
>>> pat1.sub('X', s)
'bX3 Xc X a^b'
>>> pat2.sub('X', s)
'bXX (a^b)c X|X a^b'
Escape sequences
Certain characters like tab and newline can be expressed using escape sequences as \t
and \n
respectively. These are similar to how they are treated in normal string literals. However, \b
is for word boundaries as seen earlier, whereas it stands for the backspace character in normal string literals.
The full list is mentioned at the end of docs.python: Regular Expression Syntax section as \a \b \f \n \N \r \t \u \U \v \x \\
. Do read the documentation for details as well as how it differs for byte data.
>>> re.sub(r'\t', ':', 'a\tb\tc')
'a:b:c'
>>> re.sub(r'\n', ' ', '1\n2\n3')
'1 2 3'
If an escape sequence is not defined, you'll get an error.
>>> re.search(r'\e', 'hello') re.error: bad escape \e at position 0
You can also represent a character using hexadecimal escape of the format \xNN
where NN
are exactly two hexadecimal characters. If you represent a metacharacter using escapes, it will be treated literally instead of its metacharacter feature.
# \x20 is space character
>>> re.sub(r'\x20', '', 'h e l l o')
'hello'
# \x7c is '|' character
>>> re.sub(r'2\x7c3', '5', '12|30')
'150'
>>> re.sub(r'2|3', '5', '12|30')
'15|50'
See ASCII code table for a handy cheatsheet with all the ASCII characters and their hexadecimal representations.
Octal escapes will be discussed in the Backreference section. The Codepoints and Unicode escapes section will discuss escapes for unicode characters using \u
and \U
.
Cheatsheet and Summary
Note | Description |
---|---|
\ | prefix metacharacters with \ to match them literally |
\\ | to match \ literally |
re.escape() | automatically escape all metacharacters |
ex: '|'.join(re.escape(s) for s in iterable) | |
\t | escape sequences like those supported in string literals |
\b | word boundary in RE but backspace in string literals |
\e | undefined escapes will result in an error |
\xNN | represent a character using hexadecimal value |
\x7c | will match | literally |
This short chapter discussed how to match metacharacters literally. re.escape()
helps if you are using input strings sourced from elsewhere to build the final RE. You also saw how to use escape sequences to represent characters and how they differ from normal string literals.
Exercises
a) Transform the given input strings to the expected output using the same logic on both strings.
>>> str1 = '(9-2)*5+qty/3-(9-2)*7'
>>> str2 = '(qty+4)/2-(9-2)*5+pq/4'
##### add your solution here for str1
'35+qty/3-(9-2)*7'
##### add your solution here for str2
'(qty+4)/2-35+pq/4'
b) Replace (4)\|
with 2
only at the start or end of the given input strings.
>>> s1 = r'2.3/(4)\|6 foo 5.3-(4)\|'
>>> s2 = r'(4)\|42 - (4)\|3'
>>> s3 = 'two - (4)\\|\n'
>>> pat = re.compile() ##### add your solution here
>>> pat.sub('2', s1)
'2.3/(4)\\|6 foo 5.3-2'
>>> pat.sub('2', s2)
'242 - (4)\\|3'
>>> pat.sub('2', s3)
'two - (4)\\|\n'
c) Replace any matching element from the list items
with X
for given the input strings. Match the elements from items
literally. Assume no two elements of items
will result in any matching conflict.
>>> items = ['a.b', '3+n', r'x\y\z', 'qty||price', '{n}']
>>> pat = re.compile() ##### add your solution here
>>> pat.sub('X', '0a.bcd')
'0Xcd'
>>> pat.sub('X', 'E{n}AMPLE')
'EXAMPLE'
>>> pat.sub('X', r'43+n2 ax\y\ze')
'4X2 aXe'
d) Replace the backspace character \b
with a single space character for the given input string.
>>> ip = '123\b456'
>>> ip
'123\x08456'
>>> print(ip)
12456
>>> re.sub() ##### add your solution here
'123 456'
e) Replace all occurrences of \e
with e
.
>>> ip = r'th\er\e ar\e common asp\ects among th\e alt\ernations'
>>> re.sub() ##### add your solution here
'there are common aspects among the alternations'
f) Replace any matching item from the list eqns
with X
for given the string ip
. Match the items from eqns
literally.
>>> ip = '3-(a^b)+2*(a^b)-(a/b)+3'
>>> eqns = ['(a^b)', '(a/b)', '(a^b)+2']
##### add your solution here
>>> pat.sub('X', ip)
'3-X*X-X+3'