How to loop all the elements that match the regex?
Asked Answered
T

9

72

Here is the case: I want to find the elements which match the regex...

targetText = "SomeT1extSomeT2extSomeT3extSomeT4extSomeT5extSomeT6ext"

and I use the regex in javascript like this

reg = new RegExp(/e(.*?)e/g);   
var result = reg.exec(targetText);

and I only get the first one, but not the follow.... I can get the T1 only, but not T2, T3 ... ...

Tades answered 3/8, 2009 at 12:22 Comment(1)
Can anyone suggest a for loop way to do it?Tades
G
91

function doSomethingWith(result) {
  console.log(result)
}

const targetText = "SomeT1extSomeT2extSomeT3extSomeT4extSomeT5extSomeT6ext"
const reg = /e(.*?)e/g;
let result;
while ((result = reg.exec(targetText)) !== null) {
  doSomethingWith(result);
}
Generable answered 3/8, 2009 at 12:27 Comment(8)
Note that the regex must be assigned to an object before using in this manner: /e(.*?)e/.exec(...) will give an infinite loop.Adiell
I got infinite loop for document.body.innerHTMLNonmetallic
@RaviParekh: did you add the /g modifier?Generable
@chaos, Thanks, I had not added /g global. which led to infinite.Nonmetallic
For posterity's sake, the RegExp needs to be declared outside of the while loop; otherwise you'll still get an infinite loop.Ingeborgingelbert
You can also shorten the while expression: while(result = reg.exec(targetText)) { leveraging the "false" resolve.Ingeborgingelbert
IMPORTANT: You need to set the read/write property of the regex reg.lastIndex = result.index + 1; at the end of the loop in order to move to the next match.Vitriol
@Vitriol You must only increment lastIndex manually (reg.lastIndex = result.index + 1;) if you are not using the g (global) flag in your regex. The global flag will automatically move to the next match, however, the disadvantage is that if you want to re-use a global regex, you must reset reg.lastIndex = 0; before you match a new string.Christianson
I
71

Three approaches depending on what you want to do with it:

  • Loop through each match: .match

    targetText.match(/e(.*?)e/g).forEach((element) => {
       // Do something with each element
    });
    
  • Loop through and replace each match on the fly: .replace

    const newTargetText = targetText.replace(/e(.*?)e/g, (match, $1) => {
      // Return the replacement leveraging the parameters.
    });
    
  • Loop through and do something on the fly: .exec

    const regex = /e(.*?)e/g;  // Must be declared outside the while expression, 
                               // and must include the global "g" flag.
    
    let result;
    while(result = regex.exec(targetText)) {
      // Do something with result[0].
    } 
    
Ingeborgingelbert answered 13/3, 2018 at 17:32 Comment(5)
Ironically my Lint wants to put == in place of = in the while expression.Omnipotent
To make eslint accept the single = the assignment can be surrounded by parentheses like so: while ((result = regex.exec(targetText)))Orchestral
In the second example there, is match a key word or a variable? And if it's a variable what is it holding?Luxuriance
@ColePerry match is the whole RegExp match, and $1 is the first capture group (), in that example, only one.Ingeborgingelbert
Here, only replace will work, because in Node.js the global match/exec returns only total match without groups.Keitel
S
13

Try using match() on the string instead of exec(), though you could loop with exec as well. Match should give you the all the matches at one go. I think you can omit the global specifier as well.

reg = new RegExp(/e(.*?)e/);   
var matches = targetText.match(reg);
Selfrighteous answered 3/8, 2009 at 12:31 Comment(6)
Don't link to w3schools. They are unaffiliated with W3C and have a lot of wrong information. See w3fools.com.Joashus
@MindVirus - if you'd bother to check, they've cleaned up a lot of the stuff that w3fools.com pointed out. I still use them as a quick, though not definitive, resource. Much like one might use a cheat sheet over K&R C to look up something simple.Selfrighteous
As an intermediate dev, I've already encountered a good half dozen clear errors at w3schools. They're off my party list.Lazo
MDN reference for String.prototype.matchdeveloper.mozilla.org/en/docs/Web/JavaScript/Reference/…Strychnine
To clarify the answer, the global specifier is understood by match.Kendrakendrah
the difference is that you don't get the captured groups with String.match, while RegExp.exec does.Undercut
F
4

I kept getting infinite loops while following the advice above, for example:

var reg = /e(.*?)e/g;
var result;
while((result = reg.exec(targetText)) !== null) {
    doSomethingWith(result);
}

The object that was assigned to result each time was:

["", "", index: 50, input: "target text", groups: undefined]

So in my case I edited the above code to:

const reg = /e(.*?)e/g;
let result = reg.exec(targetText);
while(result[0] !== "") {
    doSomethingWith(result);
    result = reg.exec(targetText);
}
Facelifting answered 31/8, 2018 at 15:3 Comment(1)
that's because of the regex. you have let to accept empty strings by putting a question markSaragossa
A
2
targetText = "SomeT1extSomeT2extSomeT3extSomeT4extSomeT5extSomeT6ext"    
reg = new RegExp(/e(.*?)e/g);   
var result;
while (result = reg.exec(targetText))
{
    ...
}
Acupuncture answered 3/8, 2009 at 12:26 Comment(1)
It doesn't work. In JS regex cannot extract groups with global flag, so the user will receive only total match.Keitel
S
1

You could also use the String.replace method to loop through all elements.

result = [];
 // Just get all numbers
"SomeT1extSomeT2extSomeT3ext".replace(/(\d+?)/g, function(wholeMatch, num) {
  // act here or after the loop...
  console.log(result.push(num));
  return wholeMatch;
});
console.log(result); // ['1', '2', '3']

Greetings

Sparker answered 19/2, 2019 at 10:57 Comment(0)
E
1

I did this in a console session (Chrome).

> let reg = /[aeiou]/g;
undefined
> let text = "autoeciously";
undefined
> matches = text.matches(reg);
(7) ["a", "u", "o", "e", "i", "o", "u"]
> matches.forEach(x =>console.log(x));
a
u
o
e
i
o
u
Equitation answered 2/10, 2020 at 19:11 Comment(0)
R
0

Although @tvanfosson suggested in his answer, I'm adding the modified version for someone who only needs to get all matchings.

var targetText = "SomeT1extSomeT2extSomeT3extSomeT4extSomeT5extSomeT6ext";
var reg = /e(.*?)e/g;
var ss = targetText.match(reg); // ss = "eT1e,eT2e,eT3e,eT4e,eT5e,eT6e"
Ricciardi answered 11/8, 2022 at 12:9 Comment(0)
R
-1

I was actually dealing with this issue. I prefer Lambda functions for about everything.

reg = /e(.*?)e/gm;   
targetText.match(reg).forEach(element => console.log(element));
Recompense answered 20/2, 2020 at 20:22 Comment(1)
this only found the first match.Equitation

© 2022 - 2024 — McMap. All rights reserved.