Доброго времени суток, друзья!
Прочитал на freeCodeCamp статью про рекурсию.
Для того, чтобы понять рекурсию, нужно сначала понять рекурсию. Чего?
Рекурсия — это когда функция вызывает сама себя. Уже лучше.
Пример из статьи:
const counter = number => {
// условие остановки рекурсии
if (number === 0) {
return
}
console.log(number)
return counter(number - 1)
}
counter(5) // 5 4 3 2 1
Вот оно что… А почему нельзя перебрать элементы в цикле?
Например, так:
const counter2 = num => {
while (num !== 0) {
console.log(num)
num--
}
}
counter2(5) // 5 4 3 2 1
Ответа на данный вопрос статья не содержит. Видимо, так неинтересно.
Более того, в ней говорится о том, что рекурсия, как правило, менее производительна по сравнению с другими способами решения задачи.
Тем не менее, в статье предлагается два упражнения, которые мне захотелось решить. Делюсь с вами результатами и приглашаю внести свою лепту.
Задачи простые, если не считать того, что их надо решить с помощью рекурсии.
Вот мои рекурсивные и нерекурсиные решения. Прежде чем заглядывать под кат, попробуйте свои силы.
Задача номер раз
Реализуйте функцию, принимающую строку и инвертирующую ее. Например, строка «JavaScript» должна стать строкой «tpircSavaJ».
function invert(string) {
const lastIndex = string.length - 1
if (lastIndex > 0) {
return string[lastIndex] + invert(string.slice(0, lastIndex))
} else {
return string
}
}
const str = 'JavaScript'
console.log(invert(str)) // tpircSavaJ
Стрелочная функция + тернарный оператор:
const invert2 = (str, i = str.length - 1) => (i > 0) ? str[i] + invert2(str.slice(0, i)) : str
console.log(invert2(str)) // tpircSavaJ
Стрелочная функция + тернарный оператор + деструктуризация:
const invert3 = ([firstLetter, ...str]) => firstLetter ? invert3(str) + firstLetter : ''
console.log(invert3(str)) // tpircSavaJ
const invert4 = str => str.split('').reverse().join('')
console.log(invert4(str)) // tpircSavaJ
Стрелочная функция + деструктуризация:
const invert5 = str => [...str].reverse().join('')
console.log(invert5(str)) // tpircSavaJ
Задача номер два
Реализуйте функцию, принимающую строку и символ и возвращающую количество символов в строке. Например, если мы передаем «JavaScript» и «a», то должны получить 2 (в строке «JavaScript» две буквы «a»).
Обычная функция:
function count(string, letter) {
if (string.length > 0) {
if (string[0] === letter) {
return 1 + count(string.slice(1), letter)
} else {
return count(string.slice(1), letter)
}
} else {
return 0
}
}
console.log(count(str, 'a')) // 2
Стрелочная функция + тернарный оператор:
const count2 = (
str,
l,
str2 = str.slice(1)
) => (str.length === 0) ? 0 : (str[0] === l) ? 1 + count2(str2, l) : count2(str2, l)
console.log(count2(str, 'a')) // 2
const count3 = (str, l, n = 0) => {
str
// .toLowerCase()
.split('')
.map(lt => {
if (lt === l) {
n++
}
})
return n
}
console.log(count3(str, 'a')) // 2
Еще одна стрелочная функция:
const count4 = (str, l) => str.toLowerCase().split(l).length - 1
console.log(count4(str, 'a')) // 2
Стрелочная функция + деструктуризация:
const count5 = (str, l) => [...str.toLowerCase()].filter(lt => lt === l).length || 0
console.log(count5(str, 'a')) // 2
Стрелочная функция + регулярное выражение:
const count6 = (str, l) => {
const m = str.match(new RegExp(l, 'gi'))
return m ? m.length : 0
}
console.log(count6(str, 'a')) // 2
Немного отсебятины
Можно придумать еще парочку задач со строками, но уже без рекурсии.
Как «капитализировать» строку, т.е. сделать первую букву заглавной?
const capWord = str => `${str[0].toUpperCase()}${str.slice(1)}`
console.log(capWord('hello')) // Hello
Что, если строка состоит из нескольких слов?
const capWords = str =>
str.split(' ')
// используем предыдущую функцию
.map(i => capWord(i))
.join(' ')
console.log(capWords('hello, world')) // Hello, World
Как привести строку к camel case?
const toCamelCase = str => str.split(' ').reduce((acc, cur) => acc + capWord(cur))
console.log(toCamelCase('camel case hello world')) // camelCaseHelloWorld
Как капитализировать слова в строке с помощью регулярного выражения?
const capWithRegex = str =>
str.replace(
// регулярка зависит от строки, для кириллицы - /[а-яё]\S+/gi
/\w+\S+/gi,
txt => `${txt[0].toUpperCase()}${txt.slice(1).toLowerCase()}`
)
console.log(capWithRegex('pRima, alTera, triEra')) // Prima, Altera, Triera
В целом, складывается впечатление, что актуальность рекурсии в JavaScript уменьшается прямо пропорционально увеличению количества встроенных методов. Вероятно, существуют специфические задачи, единственным решением которых является использование рекурсии, но я о таких не слышал.
Рекурсивный образ мышления (назовем его так) показался мне непривычным и неудобным, каким-то, выражаясь словами героев фильма «Довод», инвертированным: ехали медведи на велосипеде, а за ними кот задом наперед. И только я подумал о том, что не вижу объективных причин целенаправленно формировать у себя такое мышление, как вспомнил об алгоритмах, ведь многие из них пропитаны рекурсией. Все-таки придется инвертироваться.
А что вы думаете о рекурсии?
Suvitruf
В JavaScript какая-то особенная рекурсия? Зачем акцентировать внимание на языке?
lostero
В нём легче получить
Call Stack Overflow
в браузерах типа IE (иногда достаточно цепочки из 10 рекурсивных вызовов).В более продвинутых JS движках нужно больше рекурсивных вызовов, чтобы вызвать эту ошибку (иногда задают лимиты от объёма памяти). Некоторые, с недавних пор, могут в оптимизацию рекурсивных вызовов.
Рекурсия — верный путь к проблемам в JS.
Но некоторые вещи без рекурсии сложно реализовать, приходиться ограничивать кол-во рекурсивных вызовов и возвращать индикатор ошибки при выполнении рекурсивной функции.