Regex exec возвращает только первое совпадение
Я пытаюсь реализовать следующий поиск по регулярному выражению, найденный на странице синтаксиса в поле для гольфа.
var ptrn = /[a-zA-Z_][a-zA-Z0-9_]*|'(?:\\.|[^'])*'?|"(?:\\.|[^"])*"?|-?[0-9]+|#[^\n\r]*|./mg;
input = ptrn.exec(input);
Ввод - это только первое совпадение регулярного выражения. например:
"hello" "world"
должен возвращать ["hello", "world"]
, но возвращает только ["hello"]
.
Ответы
Ответ 1
RegExp.exec может возвращать сразу один результат совпадения. Чтобы получить несколько совпадений, вам нужно несколько раз запустить exec
в объекте выражения. Например, используя простой цикл while:
var ptrn = /[a-zA-Z_][a-zA-Z0-9_]*|'(?:\\.|[^'])*'?|"(?:\\.|[^"])*"?|-?[0-9]+|#[^\n\r]*|./mg;
var match;
while ((match = ptrn.exec(input)) != null) {
console.log(match);
}
Откроется все совпадения.
Обратите внимание, что для выполнения этой работы вам необходимо убедиться, что регулярное выражение имеет флаг g
(global). Этот флаг гарантирует, что после выполнения определенных методов в выражении обновляется свойство lastIndex
, поэтому дальнейшие вызовы начнутся после предыдущего результата.
Ответ 2
В цепочке можно вызвать метод match
, чтобы получить весь набор совпадений:
var ptrn = /[a-zA-Z_][a-zA-Z0-9_]*|'(?:\\.|[^'])*'?|"(?:\\.|[^"])*"?|-?[0-9]+|#[^\n\r]*|./mg;
var results = "hello world".match(ptrn);
results
(согласно регулярному выражению):
["hello", " ", "world"]
match
spec здесь
Ответ 3
Я не понял, что подразумевается под "hello" "world"
в вашем вопросе, это пользовательский ввод или регулярное выражение, но мне сказали, что у объекта RegExp есть состояние - его lastIndex
позиция, с которой начинается поиск. Он не возвращает сразу все результаты. Он возвращает только первое совпадение, и вам нужно возобновить .exec
, чтобы получить остальные результаты, начиная с позиции lastIndex:
const re1 = /^\s*(\w+)/mg; // find all first words in every line
const text1 = "capture discard\n me but_not_me" // two lines of text
for (let match; (match = re1.exec(text1)) !== null;)
console.log(match, "next search at", re1.lastIndex);
печатает
["capture", "capture"] "next search at" 7
[" me", "me"] "next search at" 19
Функциональный способ JS6 для создания итератора для ваших результатов находится здесь
RegExp.prototype.execAllGen = function*(input) {
for (let match; (match = this.exec(input)) !== null;)
yield match;
} ; RegExp.prototype.execAll = function(input) {
return [...this.execAllGen(input)]}
Также обратите внимание, как, в отличие от poke, гораздо приятнее я использовал переменную match
, заключенную в for
-loop.
Теперь вы можете легко записать свои матчи в одной строке
const matches = re1.execAll(text1)
log("captured strings:", matches.map(m=>m[1]))
log(matches.map(m=> [m[1],m.index]))
for (const match of matches) log(match[1], "found at",match.index)
который печатает
"captured strings:" ["capture", "me"]
[["capture", 0], ["me", 16]]
"capture" "found at" 0
"me" "found at" 16