JavaScript函数式编程核心是优先使用纯函数并严格管理副作用:纯函数具确定性、无外部依赖与干扰、不改变参数;副作用指函数返回值外导致外部状态可观测变化的行为,需通过隔离、标记和延迟执行来管控。
JavaScript函数式编程不是写一堆function堆砌,而是用一种更可预测、更易测试、更少出错的方式来组织逻辑。核心就两点:**优先使用纯函数,严格管理副作用**。
纯函数就像一台数学计算器——给它2和3,它永远返回5;再按一次,还是5;换台机器,结果也不变。
Math.max(1, 2)、String.prototype.toUpperCase()、自定义的add(a, b) { return a + b; }都是纯的。Date.now()或Math.random()这类“会变”的东西。push、sort直接改它,而是用concat、slice、map等返回新数组。只要函数做了返回值以外、能让外部状态发生可观测变化的事,就算有副作用。这些变化不一定是错的,但会让代码变得难追踪
、难复现、难测试。
count++,下次调用结果就不同了。obj.name = 'new',调用者原对象就被动改了。fetch()、console.log()、document.body.appendChild()、localStorage.setItem()都算。new Date()或Math.random(),哪怕没改外部,每次结果也不同,破坏了确定性。不是为了炫技,而是解决真实痛点:
map、filter、reduce这些纯函数天然支持链式抽象。实际开发中不可能完全消灭副作用——页面总要渲染,用户总要点按钮,数据总要发出去。关键在于:
useEffect、Redux的thunk/saga。saveToServer()比processData()更清楚会有I/O。