Як отримати n-ту зустріч у рядку?


104

Я хотів би отримати вихідну позицію 2ndвиникнення ABCз подібним чином:

var string = "XYZ 123 ABC 456 ABC 789 ABC";
getPosition(string, 'ABC', 2) // --> 16

Як би ти це зробив?


Друге виникнення чи остання? :)
Ja͢ck

Вибачте за плутанину Я не шукаю останній індекс. Я шукаю вихідну позицію nthвиникнення, в даному випадку другу.
Адам

Відповіді:


158

const string = "XYZ 123 ABC 456 ABC 789 ABC";

function getPosition(string, subString, index) {
  return string.split(subString, index).join(subString).length;
}

console.log(
  getPosition(string, 'ABC', 2) // --> 16
)


26
Мені справді не подобається ця відповідь. З огляду на необмежену довжину, вона без потреби створює масив необмеженої довжини, а потім викидає більшу частину її. Було б швидше і ефективніше просто ітеративно використовувати fromIndexаргумент доString.indexOf
Альнітак

3
function getPosition(str, m, i) { return str.split(m, i).join(m).length; }
копія

9
Було б добре, якби ви вказали, що означає кожен параметр.
Раніше

1
@Якщо я просто реалізував функцію, визначену ОП
Denys Séguret

5
Це дасть вам довжину рядка, якщо є < iвходження m. Тобто getPosition("aaaa","a",5)дає 4, як робить getPosition("aaaa","a",72)! Я думаю, ти хочеш -1 у тих випадках. var ret = str.split(m, i).join(m).length; return ret >= str.length ? -1 : ret;Ви також можете зловити i <= 0зreturn ret >= str.length || i <= 0 ? -1 : ret;
Ruffin

70

Ви також можете використовувати рядок indexOf, не створюючи масивів.

Другий параметр - індекс, щоб почати шукати наступний збіг.

function nthIndex(str, pat, n){
    var L= str.length, i= -1;
    while(n-- && i++<L){
        i= str.indexOf(pat, i);
        if (i < 0) break;
    }
    return i;
}

var s= "XYZ 123 ABC 456 ABC 789 ABC";

nthIndex(s,'ABC',3)

/*  returned value: (Number)
24
*/

Мені подобається ця версія через кешування довжини і не розширення прототипу String.
Крістоф Руссі

8
Згідно з jsperf, цей спосіб набагато швидший, ніж прийнята відповідь
boop

Приріст iможе бути менш заплутаним:var i; for (i = 0; n > 0 && i !== -1; n -= 1) { i = str.indexOf(pat, /* fromIndex */ i ? (i + 1) : i); } return i;
hlfcoding

1
Я віддаю перевагу цій відповіді прийнятій відповіді, як коли я перевіряв другий екземпляр, який не існував, інша відповідь повертала довжину першого рядка, де ця поверталася -1. Підвищення голосу і дякую.
Іван

2
Абсурдно, що це не вбудована особливість JS.
Зловісна борода

20

Опрацьовуючи відповідь kennebec, я створив функцію прототипу, яка поверне -1, якщо n-й виникнення не буде знайдено, а не 0.

String.prototype.nthIndexOf = function(pattern, n) {
    var i = -1;

    while (n-- && i++ < this.length) {
        i = this.indexOf(pattern, i);
        if (i < 0) break;
    }

    return i;
}

2
Ніколи не використовуйте camelCase, оскільки можлива адаптація функцій споконвічно може бути ненавмисно перезаписана цим прототипом. В цьому випадку я б рекомендував все рядкові і підкреслення (риски для URL - адреси): String.prototype.nth_index_of. Навіть якщо ви вважаєте, що ваше ім'я унікальне і досить божевільне, світ доведе, що він може і буде божевільнішим.
Іван

Тим більше, що при прототипуванні. Звичайно, ніхто не може використовувати саме таку назву методу, хоча дозволяючи собі це робити, ви створюєте шкідливу звичку. Інший, хоча критичний приклад: завжди вкладайте дані під час роботи SQL, INSERTоскільки mysqli_real_escape_stringце не захищає від хак-одиничних цитат. Значна частина професійного кодування - це не просто мати хороші звички, але також розуміти, чому такі звички важливі. :-)
Іван

1
Не подовжуйте прототип протоколу.

4

Бо рекурсія - це завжди відповідь.

function getPosition(input, search, nth, curr, cnt) {
    curr = curr || 0;
    cnt = cnt || 0;
    var index = input.indexOf(search);
    if (curr === nth) {
        if (~index) {
            return cnt;
        }
        else {
            return -1;
        }
    }
    else {
        if (~index) {
            return getPosition(input.slice(index + search.length),
              search,
              nth,
              ++curr,
              cnt + index + search.length);
        }
        else {
            return -1;
        }
    }
}

1
@RenanCoelho The tilde ( ~) є побітним НЕ оператором JavaScript: developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/…
Sébastien

2

Ось моє рішення, яке просто перебирає рядок, поки nне будуть знайдені збіги:

String.prototype.nthIndexOf = function(searchElement, n, fromElement) {
    n = n || 0;
    fromElement = fromElement || 0;
    while (n > 0) {
        fromElement = this.indexOf(searchElement, fromElement);
        if (fromElement < 0) {
            return -1;
        }
        --n;
        ++fromElement;
    }
    return fromElement - 1;
};

var string = "XYZ 123 ABC 456 ABC 789 ABC";
console.log(string.nthIndexOf('ABC', 2));

>> 16

2

Цей метод створює функцію, яка вимагає індексу n-х подій, що зберігаються в масиві

function nthIndexOf(search, n) { 
    var myArray = []; 
    for(var i = 0; i < myString.length; i++) { //loop thru string to check for occurrences
        if(myStr.slice(i, i + search.length) === search) { //if match found...
            myArray.push(i); //store index of each occurrence           
        }
    } 
    return myArray[n - 1]; //first occurrence stored in index 0 
}

Я не думаю, що ви вказали myString у наведеному вище коді, і не впевнені, чи myStr === myString?
Сет-Едем

1

Коротший шлях і я думаю простіше, не створюючи зайвих рядків.

const findNthOccurence = (string, nth, char) => {
  let index = 0
  for (let i = 0; i < nth; i += 1) {
    if (index !== -1) index = string.indexOf(char, index + 1)
  }
  return index
}

0

Використання indexOfта рекурсія :

Спочатку перевірте, чи передана n-та позиція перевищує загальну кількість входів підрядків. Якщо це було передано, рекурсивно проходьте кожен індекс, поки не буде знайдений n-й.

var getNthPosition = function(str, sub, n) {
    if (n > str.split(sub).length - 1) return -1;
    var recursePosition = function(n) {
        if (n === 0) return str.indexOf(sub);
        return str.indexOf(sub, recursePosition(n - 1) + 1);
    };
    return recursePosition(n);
};

0

Використання [String.indexOf][1]

var stringToMatch = "XYZ 123 ABC 456 ABC 789 ABC";

function yetAnotherGetNthOccurance(string, seek, occurance) {
    var index = 0, i = 1;

    while (index !== -1) {
        index = string.indexOf(seek, index + 1);
        if (occurance === i) {
           break;
        }
        i++;
    }
    if (index !== -1) {
        console.log('Occurance found in ' + index + ' position');
    }
    else if (index === -1 && i !== occurance) {
        console.log('Occurance not found in ' + occurance + ' position');
    }
    else {
        console.log('Occurance not found');
    }
}

yetAnotherGetNthOccurance(stringToMatch, 'ABC', 2);

// Output: Occurance found in 16 position

yetAnotherGetNthOccurance(stringToMatch, 'ABC', 20);

// Output: Occurance not found in 20 position

yetAnotherGetNthOccurance(stringToMatch, 'ZAB', 1)

// Output: Occurance not found

0
function getStringReminder(str, substr, occ) {
   let index = str.indexOf(substr);
   let preindex = '';
   let i = 1;
   while (index !== -1) {
      preIndex = index;
      if (occ == i) {
        break;
      }
      index = str.indexOf(substr, index + 1)
      i++;
   }
   return preIndex;
}
console.log(getStringReminder('bcdefgbcdbcd', 'bcd', 3));

-2

Я розігрувався з наступним кодом для іншого питання про StackOverflow і подумав, що це може бути доречним тут. Функція printList2 дозволяє використовувати регулярний вираз і перераховує всі події в порядку. (printList був спробою більш раннього рішення, але в ряді випадків не вдався.)

<html>
<head>
<title>Checking regex</title>
<script>
var string1 = "123xxx5yyy1234ABCxxxabc";
var search1 = /\d+/;
var search2 = /\d/;
var search3 = /abc/;
function printList(search) {
   document.writeln("<p>Searching using regex: " + search + " (printList)</p>");
   var list = string1.match(search);
   if (list == null) {
      document.writeln("<p>No matches</p>");
      return;
   }
   // document.writeln("<p>" + list.toString() + "</p>");
   // document.writeln("<p>" + typeof(list1) + "</p>");
   // document.writeln("<p>" + Array.isArray(list1) + "</p>");
   // document.writeln("<p>" + list1 + "</p>");
   var count = list.length;
   document.writeln("<ul>");
   for (i = 0; i < count; i++) {
      document.writeln("<li>" +  "  " + list[i] + "   length=" + list[i].length + 
          " first position=" + string1.indexOf(list[i]) + "</li>");
   }
   document.writeln("</ul>");
}
function printList2(search) {
   document.writeln("<p>Searching using regex: " + search + " (printList2)</p>");
   var index = 0;
   var partial = string1;
   document.writeln("<ol>");
   for (j = 0; j < 100; j++) {
       var found = partial.match(search);
       if (found == null) {
          // document.writeln("<p>not found</p>");
          break;
       }
       var size = found[0].length;
       var loc = partial.search(search);
       var actloc = loc + index;
       document.writeln("<li>" + found[0] + "  length=" + size + "  first position=" + actloc);
       // document.writeln("  " + partial + "  " + loc);
       partial = partial.substring(loc + size);
       index = index + loc + size;
       document.writeln("</li>");
   }
   document.writeln("</ol>");

}
</script>
</head>
<body>
<p>Original string is <script>document.writeln(string1);</script></p>
<script>
   printList(/\d+/g);
   printList2(/\d+/);
   printList(/\d/g);
   printList2(/\d/);
   printList(/abc/g);
   printList2(/abc/);
   printList(/ABC/gi);
   printList2(/ABC/i);
</script>
</body>
</html>

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.