给函数取一个“好”的名字
来源:Matrix海子(ID:matrix_haizi)
早在2013年,国外有个程序员做了一个有意思的投票统计,该投票是让程序员从以下几个选项中选出平时在工作中自己认为最难做的事情:
做项目方案设计
编写测试用例
撰写设计文档
向别人解释我们在做什么事情
实现你不认同的feature
在别人写的代码基础上做改造
与人沟通
给函数、变量命名
进行工作量估时
也许在大家的印象中,撰写设计文档和在别人写的代码基础上做改造应该是最难的事情。但是最终的投票结果确让大家意想不到,这次投票一起有4522名程序员参与了投票,排在第一位的是给函数、变量命名:
大概一半的人投票给了给函数、变量命名,从这次投票结果我们足可以看出:给函数、变量命名虽然是一件再普通不过的事情,但是要想把这件事做好绝非易事。那么今天,我们就来聊一聊如何给函数取一个好的名字。
一. 常见函数命名风格
目前来说,最常见的函数命名主要有两种风格:驼峰命名和帕斯卡命名。
驼峰命名:多个单词组成一个名称时,第一个单词全部小写,后面单词首字母大写;如:
public void setUserName(String userName);
帕斯卡命名:多个单词组成一个名称时,每个单词的首字母大写;
public void SetUserName(String userName);
两种命名风格都是ok的,但要保证一点,对于一个团队或者一个项目,需要根据语言本身的推荐命名方式做好约定。比如java一般都采取驼峰命名,C#采取帕斯卡命名。
二. 函数命名最高境界
我们通常说:天下武功,唯快不破。那么对于函数命名来说最高境界是什么呢?我认为是:见字如面,顾名思义,就是看到函数的名字就知道这个函数具体做了哪些事情。
比如上面的函数:
public void setUserName(String userName);
但是下面这个函数命名就不是一个好的命名:
public String addCharacter(String originString, char ch);
这个函数,一咋看,还不错,从函数字面意思看是给某个字符串添加一个字符。但是到底是在原有字符串首部添加,还是在原有字符串末尾追加呢?亦或是在某个固定位置插入呢?从函数名字完全看不出来这个函数的真正意图,只能继续往下读这个函数的具体实现才知道。
而下面这几个名字就比上面要好得多:
public String appendCharacter(String originString, char ch); // 追加到末尾
public String insertCharacter(String originString, char ch, int insertPosition); // 插入指定位置
三. 函数命名最佳实践
1)要领1:动词选取要精准
通常来说,动词决定了一个函数要采取什么"动作"。动词取的好,一个函数名字已经成功了80%。
常用动词表:
类别 |
单词 |
---|---|
添加/插入/创建/初始化/加载 |
add、append、insert、create、initialize、load |
删除/销毁 |
delete、remove、destroy、drop |
打开/开始/启动 |
open、start |
关闭/停止 |
close、stop |
获取/读取/查找/查询 |
get、fetch、acquire、read、search、find、query |
设置/重置/放入/写入/释放/刷新 |
set、reset、put、write、release、refresh |
发送/推送 |
send、push |
接收/拉取 |
receive、pull |
提交/撤销/取消 |
submit、cancel |
收集/采集/选取/选择 |
collect、pick、select |
提取/解析 |
sub、extract、parse |
编码/解码 |
encode、decode |
填充/打包/压缩 |
fill、pack、compress |
清空/拆包/解压 |
flush、clear、unpack、decompress |
增加/减少 |
increase、decrease、reduce |
分隔/拼接 |
split、join、concat |
过滤/校验/检测 |
filter、valid、check |
2)要领2:名词使用领域词汇
动词决定了函数的具体动作,而名词决定了函数具体的操作对象,对于名词,尽量使用领域词汇,不要使用生僻或者大家很少使用的词语。
举个例子:集合的容量通常用capacity、集合实际元素个数用size、字符串长度用length,这种就遵循大家的使用习惯,不要用size去形如字符串的长度。
再比如,假如使用到建造者模式,那么通常会用build作为函数名字,这个时候就不要另辟蹊径,用create来作为函数名字,使用大家约定俗成的命名习惯更容易让你的代码被别人读懂。
常用名词表:
类别 |
单词 |
---|---|
容量/大小/长度 |
capacity、size、length |
实例/上下文 |
instance、context |
配置 |
config、settings |
头部/前面/前一个/第一个 |
header、front、previous、first |
尾部/后面/后一个/最后一个 |
tail、back、next、last |
区间/区域/某一部分/范围/规模 |
range、interval、region、area、section、scope、scale |
缓存/缓冲/会话 |
cache、buffer、session |
本地/局部/全局 |
local、global |
成员/元素 |
member、element |
菜单/列表 |
menu、list |
源/目标 |
source、destination、target |
3)要领3:函数取名最忌讳"名不副实"
函数取名最忌讳的是"名不副实",举个例子,假如有个Cache类,里面有个函数判断key是否过期:
public boolean isExpired(String key) {
// 当前时间戳
long curTimestamp = DateUtils.nowUnixTime();
// 获取key的存入时间戳
long storeTimestamp = getStoreTimestamp(key);
if (curTimestamp - storeTimestamp > MAX_EXPIRE_SECONDS) {
// 注意这个地方的delete是个隐藏逻辑
delete(key);
return true;
}
return false;
}
上面这个函数从函数字面意思看是判断key是否过期,但是!!它居然在函数里面隐藏了一段特殊逻辑:如果过期则删除掉key。这个就是典型的"名不副实",这个是最忌讳的,会给后续的开发人员留下"巨坑"。
有两种方式去优化这段代码:
方式一:将隐藏逻辑去掉
public boolean isExpired(String key) {
// 当前时间戳
long curTimestamp = DateUtils.nowUnixTime();
// 获取key的存入时间戳
long storeTimestamp = getStoreTimestamp(key);
if (curTimestamp - storeTimestamp > MAX_EXPIRE_SECONDS) {
return true;
}
return false;
}
方式二:改变函数名字
public int deleteIfExpired(String key) {
// 当前时间戳
long curTimestamp = DateUtils.nowUnixTime();
// 获取key的存入时间戳
long storeTimestamp = getStoreTimestamp(key);
if (curTimestamp - storeTimestamp > MAX_EXPIRE_SECONDS) {
return delete(key);
}
return 0;
}
4)要领4:多查询条件的函数名字谨慎使用介词by
我们平时在写查询接口时,假如有多个查询参数怎么办?每个通过by一起连接依赖?No!这绝对不是明智的方式。假如一开始产品的需求是通过学生姓名查询学生信息,写出来的可能是这样的函数:
public List<Student> getByName(String name);
然后突然又有一天产品提出了新的需求,希望同时可以通过姓名和电话号码来查询学生信息,那么函数可能变成这样了:
public List<Student> getByNameAndMobile(String name, String mobile);
接着,没过多久,产品又希望根据学生年龄来查询学生信息,那么函数可能变成这样了:
public List<Student> getByNameAndMobileAndAge(String name, String mobile, int age);
如果这样来给函数命名,那么你的噩梦大门即将打开。
通常比较好的做法是:
如果是通过主键id来查询,那么可以通过by来连接查询信息,比如:
public Student getByStudentId(long studentId);
如果是通过其他属性来查询,并且未来会存在多个组合查询的可能性,建议进行封装,比如:
public List<Student> getStudents(StudentSearchParam searchParam);
最后,建议大家平时在写代码过程中,不要怕在函数命名上耗费时间,一个好的函数命名在后期会大大减少你代码重构的成本,争取对函数命名做到"见字如面"。
(完)
Java团长
专注于Java干货分享
扫描上方二维码获取更多Java干货
给函数取一个“好”的名字相关推荐
- 工程实践:如何给变量取一个好的名字
工程实践:如何给变量取一个好的名字 在上一篇文章中跟大家分享了关于函数命名的一些实践心得,今天我们继续命名这个话题,来讲一讲如何对变量命名. 以下是本文的目录大纲: 一. 变量命名风格 二. 变量命名 ...
- wpf中使用控件时,最好给控件取一个好的名字
wpf中使用控件时,最好给控件取一个好的名字, 比如按钮A可以 取名为 btnA 这样使用会是编程清晰.
- 给数据库表列取一个好的名字
做父母的总希望给自己的子女取一个响亮的.富有含义的名字.有时候家长为了找到一个合适的名字,会拿起平时碰都不碰的汉语大辞典翻个好几天.天下父母心呀.其实在数据库中创建对象时,管理员也要对其进行命名. 其 ...
- 给函数取一个“好”名字
参考资料:https://www.cnblogs.com/dolphin0520/p/10567879.html 一.常见的函数命名风格 目前,函数最常见的两种命名风格: (1)驼峰命名法: 多个单词 ...
- 【C#】如何给变量取一个好的名字
讲一讲如何对变量命名. 以下是本文的目录大纲: 一. 变量命名风格 二. 变量命名最高境界 三. 变量命名最佳实践 一.变量命名风格 变量命名风格通常会根据不同的变量类型来区分,以Java语言为例,根 ...
- 转载--给函数取个好名
工程实践:给函数取一个"好"的名字 - Matrix海子 - 博客园 (cnblogs.com) 我发现,我给函数起名简直就是一个老大难的问题,并不是不知道这个函数是 ...
- 教你如何给孩子取一个好名字!
古语有云:"赐子千金,不如教子一艺:教子一艺,不如赐子好名."可谓道出了姓名对人的重要性.那么,作为父母如何给孩子起一个大富大贵.提升运势的名字呢? 首先,须以孩子的生辰八字取用神 ...
- 给函数和变量取一个有意义的名字
为什么要给函数和变量取一个有意义的名字? 1. 代码写出来是给以后的自己和别人看的,易于理解的变量名可以更好的阅读,并且可以看出编码者的个人修养和编程功底: 2. 好的变量名字有利于提高自己英语水平( ...
- 用人工智能取一个超酷的名字(一)
用人工智能取一个超酷的名字(一) 在我年轻的时候,我总是讨厌别人叫我春阳.这主要是因为我觉得这个名字有点土,听起来不够有格调.但是没有办法,按照我们中国人的传统,我这辈按照族谱中的规定,第二个字是春, ...
- 为了取一个花名,我爬下了中草药网所有的名字!
很酷哦! 不过,对我这个选择恐惧症来说,也很纠结- 我们先看一下有哪些要求吧? 中草药名?人参?西洋参?还有啥??? 作为一个不怎么吃药的非医学生,这题真的超纲了呀.我只能求助度娘. 但是有个问题,我 ...
最新文章
- SpringSecurity分布式整合之common工具模块创建
- nginx模块开发—HTTP初始化之listen
- Linux内核分析作业第八周
- python中与0xf2相等的是_python中__str__与__repr__
- 预见2021-罗兰贝格中国行业趋势报告
- numpy+matplotlib绘制正弦曲线
- 解决keil5不支持Samsung 2440A等较旧型号Soc的方法
- java如何打JAR包
- 深度学习中的优化算法之Adadelta
- iTerm2使用zmodem协议上传下载文件
- 【Windows远程桌面】RDP Wrapper 监听器状态为 Not listening [not supported] 的解决方法
- 上传图片到淘宝接口调用展示
- Golang hijack 劫持
- 实现多余文字显示省略号
- IDEA设置按键提示 Ctrl+p
- python下tkinter模块和mysql构建图书管理系统实验
- ArrayMap源码注释
- 解决ubuntu14 restart mysql提示stop Unknown job:mysql问题
- [Qt]的Layout边缘空白调整
- ESP8266开发之旅 基础篇⑥ Ticker——ESP8266定时库
热门文章
- 从更高到更好 揭开2021阿里双11背后的技术亮点
- 电话线配线架是110配线架吗?电话线配线架的接法详细介绍
- 布局中颜色搭配怎么看最舒服之白色的最佳10种颜色搭配
- 内存测试拷机软件,烤机软件 OCCT 更新 9.0.0 版本,新增 CPU/内存跑分测试
- MQTT Qos详解(一)
- [EXtJS5学习笔记]第一节 Sencha Cmd 学习笔记 简介 Sencha Cmd是什么
- 做你的大玩具——轩小样儿的六一
- r730 raid5 linux 驱动,Dell power edge R730 raid卡安装配置
- 罗永浩“卖艺”还债:所有命运馈赠的礼物,都早已在暗中标好了价格
- Docker镜像安装的一般步骤