378

I need a regex able to match everything but a string starting with a specific pattern (specifically index.php and what follows, like index.php?id=2342343)

SirPeople
  • 3,611
  • 20
  • 43
pistacchio
  • 50,394
  • 95
  • 256
  • 391
  • And what *specific* pattern do you want to not match? – Dominic Rodger Nov 06 '09 at 13:35
  • 3
    Is there a reason why you can't match against your pattern and not do something if the string matches that? – Thomas Owens Nov 06 '09 at 13:35
  • 2
    Possible duplicate of [Regular expression to match a line that doesn't contain a word?](https://stackoverflow.com/questions/406230/regular-expression-to-match-a-line-that-doesnt-contain-a-word) – 7vujy0f0hy Nov 09 '17 at 09:22
  • @ThomasOwens: It depends. It depends on which part of the expression shall be negated. If the whole expression is to be negated, then you got a point. For example, if you want to code up "if the string doesn't contain 'Bruce' as a substring, then do something", you'd use plainly /Bruce/, and put the negation into the if statement, outside the regex. But it could be that you'd like to negate some subexpression. Say, you're looking for something like firstname lastname, where firstname is Bruce, and lastname is everything except XYZ, where XYZ is the last name of some celebrity called Bruce. – mathheadinclouds Nov 21 '19 at 12:42

7 Answers7

429

Regex: match everything but:

Demo note: the newline \n is used inside negated character classes in demos to avoid match overflow to the neighboring line(s). They are not necessary when testing individual strings.

Anchor note: In many languages, use \A to define the unambiguous start of string, and \z (in Python, it is \Z, in JavaScript, $ is OK) to define the very end of the string.

Dot note: In many flavors (but not POSIX, TRE, TCL), . matches any char but a newline char. Make sure you use a corresponding DOTALL modifier (/s in PCRE/Boost/.NET/Python/Java and /m in Ruby) for the . to match any char including a newline.

Backslash note: In languages where you have to declare patterns with C strings allowing escape sequences (like \n for a newline), you need to double the backslashes escaping special characters so that the engine could treat them as literal characters (e.g. in Java, world\. will be declared as "world\\.", or use a character class: "world[.]"). Use raw string literals (Python r'\bworld\b'), C# verbatim string literals @"world\.", or slashy strings/regex literal notations like /world\./.

Wiktor Stribiżew
  • 484,719
  • 26
  • 302
  • 397
  • Great write up! For the case of "a string (not) equal to some string", with the example of `^(?!foo$)`, why is it that the dollar sign has to be within the parentheses for the expression to work? I was expecting `^(?!foo)$` to give the same results, but it does not. – Grant Humphries Jan 07 '17 at 17:10
  • 3
    @GrantHumphries: When the `$` anchor is inside the lookahead, it is part of the condition, part of that *zero-width assertion*. If it were outside, like in `^(?!foo)$`, it will be part of the *consuming* pattern requiring the end of string right after the start of string, making the negative lookahead irrelevant since it would always return *true* (there cannot be any text after the end of string, let alone `foo`). So, `^(?!foo$)` matches start of a string that is not followed with `foo` that is followed with the string end. `^(?!foo)$` matches an empty string. – Wiktor Stribiżew Jan 07 '17 at 19:12
  • @robots.txt Please remove these comments. You are asking an XY question. Character classes are meant to match single chars, there is no way to define a sequence of chars with them. You should probably just find the substring between the start of a string and the first occurrence of `cot` or `lan`, and remove the match, like [`regex.replace(myString, "^.*?(?:cot|lan)\s*", "")`](https://regex101.com/r/23R4Lu/2). – Wiktor Stribiżew Jul 06 '19 at 20:48
  • Dear Wiktor. You have closed my question however your linked answer fails. I have updated my question https://stackoverflow.com/questions/60004380/how-to-remove-any-given-string-pairs-from-text – MonsterMMORPG Jan 31 '20 at 13:09
  • For example your linked answer fails at this example "ing packages editors now use--> Lorem Ipsum" – MonsterMMORPG Jan 31 '20 at 13:09
  • why (*SKIP)(*FAIL) raise an error in python? how to solve this? – Khayyam Dec 10 '20 at 22:13
  • 1
    @Dotizo Python `re` library is quite different from PCRE. Use [PyPi regex library](https://pypi.org/project/regex/) that supports the `(*SKIP)(*FAIL)` verbs. – Wiktor Stribiżew Dec 10 '20 at 22:22
312

Not a regexp expert, but I think you could use a negative lookahead from the start, e.g. ^(?!foo).*$ shouldn't match anything starting with foo.

Cat Plus Plus
  • 113,388
  • 26
  • 185
  • 215
281

You can put a ^ in the beginning of a character set to match anything but those characters.

[^=]*

will match everything but =

  • 64
    That's true, but it only processes one character at a time. If you want to exclude a sequence of two or more characters, you have to use negative lookahead like the other responders said. – Alan Moore Jul 20 '13 at 10:42
  • perfect solution tu remove any undesirable character ***but*** those in the pattern. thanks – Sirmyself Jan 30 '20 at 22:26
  • @Alan, "...you have to use a negative lookahead..." is incorrect, but we shouldn't be too hard on you because Wiktor didn't post his answer--which shows why--until 2016. – Cary Swoveland Jun 07 '20 at 18:24
5

Just match /^index\.php/ then reject whatever matches it.

5

In python:

>>> import re
>>> p='^(?!index\.php\?[0-9]+).*$'
>>> s1='index.php?12345'
>>> re.match(p,s1)
>>> s2='index.html?12345'
>>> re.match(p,s2)
<_sre.SRE_Match object at 0xb7d65fa8>
AJ.
  • 25,364
  • 15
  • 77
  • 87
0

I need a regex able to match everything but except a string starting with index.php a specific pattern (specifically index.php and what follows, like index.php?id=2342343)

Use method Exec

    let match,
        arr = [],
        myRe = /([\s\S]+?)(?:index\.php\?id.+)/g;

    var str = 'http://regular-viragenia/index.php?id=2342343';

    while ((match = myRe.exec(str)) != null) {
         arr.push(match[1]);
    } 
    
    console.log(arr);

var myRe = /([\s\S]+?)(?:index\.php\?id=.+)/g;
var str = 'http://regular-viragenia/index.php?id=2342343';
var matches_array = myRe.exec(str);
console.log(matches_array[1]);

OR OTHER MATCH

let match,
            arr = [],
            myRe = /index.php\?id=((?:(?!index)[\s\S])*)/g;

        var str = 'http://regular-viragenia/index.php?id=2342343index.php?id=111index.php?id=222';

        while ((match = myRe.exec(str)) != null) {
             arr.push(match[1]);
        } 

        console.log(arr);
-17

How about not using regex:

// In PHP
0 !== strpos($string, 'index.php')
Percutio
  • 991
  • 6
  • 8
  • 12
    The OP specifically requested a regex... I'm not sure this helps! (He may be using `grep` on the command-line, for example, or Perl/Python/any other language, or an "Execute this regex for every line" command in a text editor, etc...) – rinogo Nov 19 '15 at 21:56