Rabin-Karp Algorithm
It is designed to address the multiple pattern string matching problem.
The Rabin-Karp algorithm, also known as the Karp-Rabin algorithm, was introduced by Richard M. Karp and Michael O. Rabin in 1987. It is designed to address the multiple pattern string matching problem.
Its implementation is somewhat unconventional. It begins by computing the hash values of two strings and then determines whether there is a match by comparing these hash values.
Algorithm Analysis and Implementation
Choosing an appropriate hash function is crucial. Assuming the text string is , and the pattern string is , where , let represent the hash value of the substring .
When , it is natural to compare . In this process, if we recalculate the hash value for the substring , it would require time complexity, which is not cost-effective. Observing that there are m-1 overlapping characters between the substrings and , we can use a rolling hash function. This reduces the time complexity of recalculation to .
The rolling hash function used in the Rabin-Karp algorithm primarily leverages the concept of Rabin fingerprint. For example, the formula to calculate the hash value of the substring is as follows:
Here, is a constant. In Rabin-Karp, it is generally set to 256 because the maximum value of a character does not exceed 255. The formula above has an issue - hash values could overflow. To address this, we take the modulus, and the value should be as large as possible and preferably a prime number. Here, we take 101.
The formula to calculate the hash value of the substring is then:
The complete code is as follows:
#include <iostream>
#include <string.h>
using namespace std;
#define BASE 256
#define MODULUS 101
void RabinKarp(char t[], char p[])
{
int t_len = strlen(t);
int p_len = strlen(p);
// For rolling hash
int h = 1;
for (int i = 0; i < p_len - 1; i++)
h = (h * BASE) % MODULUS;
int t_hash = 0;
int p_hash = 0;
for (int i = 0; i < p_len; i++)
{
t_hash = (BASE * t_hash + t[i]) % MODULUS;
p_hash = (BASE * p_hash + p[i]) % MODULUS;
}
int i = 0;
while (i <= t_len - p_len)
{
// Considering the possibility of hash collisions, we use memcmp for additional verification
if (t_hash == p_hash && memcmp(p, t + i, p_len) == 0)
cout << p << " is found at index " << i << endl;
// Rolling hash
t_hash = (BASE * (t_hash - t[i] * h) + t[i + p_len]) % MODULUS;
// Avoiding negative values
if (t_hash < 0)
t_hash = t_hash + MODULUS;
i++;
}
}
int main()
{
char t[100] = "It is a test, but not just a test";
char p[10] = "test";
RabinKarp(t, p);
return 0;
}
The output is as follows:
test is found at index 8
test is found at index 29
Complexity Analysis
Let's examine the space complexity first, which is easily determined: .
Now, consider the time complexity. Let the length of the text string be n and the pattern string be m. Preprocessing requires , and during matching, in the best case where there are no hash collisions, . In the worst case, where there is a collision every time, . In practical scenarios, n is often much larger than m, so the final complexity table is:
Application Analysis
The primary application of the Rabin-Karp algorithm is in plagiarism detection for articles, such as the detection system used by Semantic Scholar.
However, from the complexity data above, the Rabin-Karp algorithm does not seem to have a significant advantage. Is it practical for detecting text plagiarism? Feedback from actual usage results indicates that the time complexity for plagiarism detection is only . I believe this is mainly due to the following two points:
- In real-life articles, the text data does not often exhibit as many hash collisions as we might imagine.
- The original content in a submitted article is likely to be much larger than the plagiarized content. In other words, successful matches do not occur as frequently as we might imagine.