0

0

匿名函数是一把双刃剑,有利也有弊

零下一度

零下一度

发布时间:2017-06-30 09:29:37

|

1688人浏览过

|

来源于php中文网

原创

作为函数式编程语言,函数在javascript中的重要地位和巨大作用你尽可想象。但在其提供巨大便利的同时,也不可避免的带来巨大的问题。

匿名函数则更是一把双刃剑,它让函数式编程语言更加完美,也让代码更加难于阅读。你应该知道匿名函数是以牺牲语义化为巨大代价的。

如果一个函数没有名字,它可能无关紧要,在大部分场景中它都将失去意义。函数的名字跟你的名字,你朋友的名字,你家小宠物的名字一样,是重要关键的,否则你写它干嘛。

 

即使在看起来最没有必要的位置使用命名函数也有巨大价值

你可能随手就能枚举很多场景来证明匿名函数的便利性,不可否认有些场景无疑是有一些道理的,但大多数人以此为起点,以善小而不为。比如

Array.some,  Array.forEach,  String.replace

这样的例子能举出不少,我们也可以理直气壮的说他们不需要使用命名函数,使用匿名函数更方便,并且大家都是这么做的。但是别忘了,some,forEach,replace本身已经具有宽泛的语义了。但仍然可以更近一步:

1.Array.some, 广泛的语义是找出数组中是否有一些,但究竟有一些什么呢?

2.Array.forEach, 广泛的语义是遍历,Array提供无数的函数用于遍历,你为什么选择forEach, 而不是map,every等等?

3.String.replace, 广泛的语义是替换,但究竟是将什么替换成什么呢?

代码是写给人看的,

1.能不能不要让我去读你的代码猜或者推理出来你要从数组中找出一些什么,是不是有整数,是不是有空值?

2.能不能直接通过函数名告诉我你遍历这个数组是想干什么?

3.能不能直接通过函数名告诉我你想将什么替换成什么?

举个简单的例子,找出价格数组中是否存在整数价格。isInteger可能是已经有的公共函数,如果没有的话,经过你就有了。简洁易读,在阅读主流程的时候,有些代码是不需要阅读的,isInteger就是这样一些代码,万一有错呢?isInteger是可测易测的,如果你不放心,对它执行单元测试。你可能已经注意到,一个小小的改变,有一部分代码已经具有可测性了。即使这里不是一个公用函数,写成命名函数也更整洁更可测。

Yodayo
Yodayo

一个专为动漫迷和vTuber打造的AI艺术创作平台、交流社区

下载

var isAnyInteger = priceArr.some(Common.isInteger.bind(Common));

这些都是非常极端的被认为是可以直接写匿名函数的例子,但很明显可以看到,他们也可以作为命名函数的边界处理,即都写命名函数百利而无一害,只会更好。

 

再简单的代码也要区分架构和实现

另外一个重灾区是 then 函数,匿名函数的代码不能更丑陋,即使大家都这么写,你也应该明白,你不能这么写,正确的姿势应该写成这样, 以展示订单为例:

/*
 * name        : getOrder
 * description : 获取订单数据 */function getOrder()
{//{{{var url = 'https://www.qunar.com/getOrder';//假如收集参数比较费劲,应该用一个函数专门去收集参数var params = getOrderParams();//假如参数体比较庞大,应该先将其赋予一个变量var params = {
        orderNo:'248663058'};//无论如何,现将参数赋予变量你都将获得在这里打印变量方便调试的便利console.log('getOrder url & params:', url, params);return $.post(url, params);
}//}}}/*
 * name        : renderOrder
 * description : 拿订单,拿到就在页面上展示出来,拿不到就告诉用户为什么没拿到 */function renderOrder()
{//{{{//高级函数,只安排工作,不自己实现//getOrder() 对该函数来说是不可见的,它要的只是订单数据,偷得抢的都可以    getOrder().then(render, remindUser);
}//}}}function render()
{//{{{}//}}}function remindUser()
{//{{{}//}}}

当然,你可能觉得 renderOrder 就是鸡肋,跟只拿工资不干活的领导简直一毛一样,把then写在 $.post 后面不就行了?

不行,如果有一天获取订单既可以从本地获取,又可以从本地缓存获取,那么getOrder就升级为次高级函数,它管理两个函数 getOrderFromServer, getOrderFromCache

如果在获取订单之前或者之后还要做一些事情,那之后renderOrder能够从容应付。

想都不要想过度设计这个词,大部分人没有这个能力,不必杞人忧天。

 

你从一开始就是高级架构师只不过兼职写代码

你可以清楚的看到,大量使用命名函数让代码结构非常清晰,任何人都能容易的独懂主流程,任何人都可以容易的去实现每一个命名函数,要实现什么已经清楚的写在了函数名上了。

这些命名函数就像房屋的骨架,再往上堆叠就是一栋楼的骨架,你见过建筑设计师自己砌墙浇水泥的吗?写代码也一样,用函数堆叠成骨架,至于每个函数怎么实现,请帮我实现(在你初学的时候就是请你自己帮你自己实现)。

 

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

38

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

83

2026.03.09

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

97

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

223

2026.03.05

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

458

2026.03.04

AI安装教程大全
AI安装教程大全

2026最全AI工具安装教程专题:包含各版本AI绘图、AI视频、智能办公软件的本地化部署手册。全篇零基础友好,附带最新模型下载地址、一键安装脚本及常见报错修复方案。每日更新,收藏这一篇就够了,让AI安装不再报错!

169

2026.03.04

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

246

2026.03.03

C++高性能网络编程与Reactor模型实践
C++高性能网络编程与Reactor模型实践

本专题围绕 C++ 在高性能网络服务开发中的应用展开,深入讲解 Socket 编程、多路复用机制、Reactor 模型设计原理以及线程池协作策略。内容涵盖 epoll 实现机制、内存管理优化、连接管理策略与高并发场景下的性能调优方法。通过构建高并发网络服务器实战案例,帮助开发者掌握 C++ 在底层系统与网络通信领域的核心技术。

34

2026.03.03

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Excel 教程
Excel 教程

共162课时 | 21.2万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.9万人学习

NumPy 教程
NumPy 教程

共44课时 | 3.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号