The Baeza-Yates-Gonnet-Algorithm or Shift-or-Algorithm , also known as Shift-and or Bitap , solves the string matching problem by simulating a nondeterministic automaton . Among other things, a modification of this algorithm is used in the Unix tool grep .
Since the implementation can be traced back to bit operations, the algorithm is already very efficient in terms of its execution. If you combine this with the underlying system (one loop over the pattern in preprocessing, one loop over the text during the search), the result is an extremely efficient algorithm.
basis
The algorithm is based on a set of vectors with the following definition:

![{\ displaystyle R_ {j + 1} [i] = {\ begin {cases} 1, & {\ mbox {falls}} i = 0 \\ 1, & {\ text {falls}} R_ {j} [i -1] = 1 {\ text {and}} \ mathrm {sample characters} _ {i} = \ mathrm {input characters} _ {j + 1} \\ 0 & {\ text {otherwise}} \ end {cases}}}](https://wikimedia.org/api/rest_v1/media/math/render/svg/76c25c32ab90505114df9cc7cb6eea8bac411c28) 
This clearly means that exactly when, after processing characters in the text, the last characters match the first characters of the search pattern.
![{\ displaystyle R_ {j} [i]}](https://wikimedia.org/api/rest_v1/media/math/render/svg/b3b077ae5def6a0e8ede2e8f79b4a63b4c8b65c0)




A hit for a search pattern with length is found, if .

![{\ displaystyle R_ {j} [m] = 1}](https://wikimedia.org/api/rest_v1/media/math/render/svg/ab1c6fdc56d6bf605aa1a2b815aae4d744b69932)
Furthermore, the characteristic vectors are required for all characters that may appear in the text:
![{\ displaystyle s_ {a} [i] = {\ begin {cases} 1, & {\ text {if in the search pattern at position}} i {\ text {the character}} a {\ text {stands}} \\ 0 & {\ text {otherwise}} \ end {cases}}}](https://wikimedia.org/api/rest_v1/media/math/render/svg/df3ead999d53059ee1a8586fa63c46f3d8ff869d) 
Example:
Search pattern , length
 
Characteristic vectors:
 
Sequence (exact matching)
In order to simplify the process, a special bit operation Bitshift or for the vector is introduced first:


 
The algorithm for exact matches can now be reduced to a few simple steps:
- Initialize the vector with 0 (for all positions) and begin with the first character of the text to be searched 
- Move all bits in one position to the right using a bit shift . 
- Carry out a combination of and the characteristic vector of the current text character.  
- Go to the next text character. If end, cancel, otherwise go to (2)
On closer inspection, steps (2) and (3) carry out exactly the calculation rule for : By shifting the "old" character is created in place of the place (corresponds in combination with the condition ). The characteristic vector of the current text character has at the point just then , if patterns and text match here. This links both conditions.





![{\ displaystyle R_ {j} [i] = 1}](https://wikimedia.org/api/rest_v1/media/math/render/svg/2dcd04f1ea8ea1d6781839a2e0ca0f4cf97393b6)



Example (exact matching)
Pattern: ,
 
Text:  
 
There is a hit at (position - pattern length + correction for first character).
![{\ displaystyle R_ {8} [5] = 1}](https://wikimedia.org/api/rest_v1/media/math/render/svg/c33f0453b66ec02075ec72ec960b122af87b2096)

Extension (approximate matching)
The algorithm can carry out a fault-tolerant search with slight modifications. For this, the vector is divided:

- 
![{\ displaystyle R_ {j} ^ {0} [i]}](https://wikimedia.org/api/rest_v1/media/math/render/svg/e68c6ce9b3c5a0a88bc0e8f26aaf0c1023cd742a) : corresponds to the previous one ; The index stands for the number of errors that have occurred. : corresponds to the previous one ; The index stands for the number of errors that have occurred.![{\ displaystyle R_ {j} [i]}](https://wikimedia.org/api/rest_v1/media/math/render/svg/b3b077ae5def6a0e8ede2e8f79b4a63b4c8b65c0)  
- 
![{\ displaystyle R_ {j} ^ {1} [i]}](https://wikimedia.org/api/rest_v1/media/math/render/svg/44d890d48e8bb4249f5b1ae13b1c4ff5085ca679) : Designates a vector that is aimed at hits with at most one error. : Designates a vector that is aimed at hits with at most one error. 
- ...
- 
![{\ displaystyle R_ {j} ^ {k} [i]}](https://wikimedia.org/api/rest_v1/media/math/render/svg/11a99e48049dc58b92814df300f96aff522d3432) : Designates a vector that is aimed at hits with a maximum of errors. : Designates a vector that is aimed at hits with a maximum of errors.  
Caution: In the case of the error-prone vectors, the above interpretation with “after j characters the last i match the first i of the pattern” is difficult and no longer necessarily plausible.
The calculation rule for remains unchanged. For error vectors , a distinction is made according to the causing action:
![{\ displaystyle R_ {j} ^ {0} [i]}](https://wikimedia.org/api/rest_v1/media/math/render/svg/e68c6ce9b3c5a0a88bc0e8f26aaf0c1023cd742a)

Insert a character in the search pattern
 
Explanation: The first part of the expression describes when there are already errors, but the current character of the text and pattern match. The second part describes the error case: So far (in ) there were only errors, so the current character can be inserted into the pattern.



Interpretation: if after the input of the last characters at least characters match the search pattern and the rest can be matched by inserting the missing characters.
![{\ displaystyle R_ {j} ^ {k} [i] = 1}](https://wikimedia.org/api/rest_v1/media/math/render/svg/404e765d3b74741c5a7d8d9c13ea09ec3ec0b63b)



Deleting a character from the search pattern
 
Explanation: The first part of the expression describes when there are already errors, but the current character of the text and pattern match. The second part describes the error case: If you do not look at the first characters of the text , but only the first (in the vector the position above), the pattern matches except for errors. The mark of the pattern is then simply deleted.






Replace a character in the pattern
 
Explanation: The first part of the expression describes when there are already errors, but the current character of the text and pattern match. The second part describes the error case: According to characters, the last characters matched . If you now replace the character in the pattern with the character in the text, the last characters also match the first characters of the "new" pattern after characters .








The variants can be linked by bit or as required.
literature
Web links
- 
StringSearch - high-performance pattern matching algorithms in Java (implementations of the Shift-Or algorithm in Java; English)
- 
BNDM algorithm (PDF; 289 kB)