程序函数和变量的命名

本文探讨了Microsoft在编程中使用的匈牙利命名法,该方法通过小写字母开头的前缀来指定数据类型,提高代码可读性。然而,这种方法在实际应用中可能过于繁琐,对于非标准类型的变量处理不够灵活,导致一些开发者对其效用提出质疑。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在编程的时候,特别是比较大的程序,最头疼的往往不是算法设计和数据结构的设计,而是如何给类、函数、变量命名。
当然,如果你不追求完美,大可不必为此操心。
至今知道比较系统的命名法则是所谓的“匈牙利命名法”。
以下是在网上找的资料:
 
匈牙利命名法来源

 

MFC、句柄、控件及结构的命名规范

  在编程时,变量、函数的命名是一个极其重要的问题。好的命名方法使变量易于记忆且程序可读性大大提高。Microsoft采用匈牙利命名法来命名Windows API函数和变量。匈牙利命名法是由Microsoft的著名开发人员、Excel的主要设计者查尔斯·西蒙尼在他的博士论文中提出来的,由于西蒙尼的国籍是匈牙利,所以这种命名法叫匈牙利命名法。

    匈牙利命名法为C标识符的命名定义了一种非常标准化的方式,这种命名方式是以两条规则为基础:

    1.标识符的名字以一个或者多个小写字母开头,用这些字母来指定数据类型。

    2.在标识符内,前缀以后就是一个或者多个第一个字母大写的单词,这些单词清楚地指出了源代码内那个对象的用途。比如,m_szStudentName表示一个学生名字的类成员变量,数据类型是字符串型。

a       Array
b       Boolean
by     Byte
c       Char //有符号型字符
cb     Char Byte //无符号型字符(没多大用处)
cr      ColorRef //颜色参考值
cx,cy  Length of x,y (ShortInt) //坐标差(长度)
dw     Double Word
fn      Function
h       Handle
i        Integer
m_    Member of a class
n      Short Integer
np    Near Pointer
p      Pointer lp Long Pointer
s      String
sz     String with Zero End //以字符'/0'结尾的字符串
tm    Text //文本内容
w     Word
x,y    Coordinate //坐标

类名一般没有说明字符,例如theApp. 用在其他类中加m_就行。要注意的是:某些类也有
类似于匈牙利命名法的缩写。例如:CStatusBar m_wndStatusBar;这里的wnd表示窗口类
,但是这种命名法不是标准的匈牙利命名法的一部分。

可是

在实际使用的时候。这种命名方法往往显得太麻烦。而且其作用也没有那么明显。因为这种匈牙利命名法只规定了标准类型的缩写。而实际上我们要用到的对象的类型千变万化,标准类型的变量只是少数,很难给每个类都定义一个对应的缩写。

果然,在网上同样有人抱怨这种命名法不甚好用。

下面是网上的资料:

 

抨击匈牙利命名法

匈牙利命名法是一种编程时的命名规范。命名规范是程序书写规范中最重要也是最富争议的地方,自古乃兵家必争之地。命名规范有何用?四个字:名正言顺。用二分法,命名规范分为好的命名规范和坏的命名规范,也就是说名正言顺的命名规范和名不正言不顺的命名规范。好的舞鞋是让舞者感觉不到其存在的舞鞋,坏的舞鞋是让舞者带着镣铐起舞。一个坏的命名规范具有的破坏力比一个好的命名规范具有的创造力要大得多。

本文要证明的是:匈牙利命名法是一个坏的命名规范。本文的作用范围为静态强类型编程语言。本文的分析范本为C语言和C++语言。下文中的匈法为匈牙利命名法的简称。

一 匈牙利命名法的成本

匈法的表现形式为给变量名附加上类型名前缀,例如:nFoo,szFoo,pFoo,cpFoo分别表示整型变量,字符串型变量,指针型变量和常指针型变量。可以看出,匈法将变量的类型信息从单一地点(声明变量处)复制到了多个地点(使用变量处),这是冗余法。冗余法的成本之一是要维护副本的一致性。这个成本在编写和维护代码的过程中需要改变变量的类型时付出。冗余法的成本之二是占用了额外的空间。一个优秀的书写者会自觉地遵从一个法则:代码最小组织单位的长度以30个自然行以下为宜,如果超过50行就应该重新组织。一个变量的书写空间会给这一法则添加不必要的难度。

二 匈牙利命名法的收益

这里要证明匈牙利命名法的收益是含糊的,无法预期的。

范本1:strcpy(pstrFoo,pcstrFoo2) Vs strcpy(foo,foo2)
匈法在这里有什么收益呢?我看不到。没有一个程序员会承认自己不知道strcpy函数的参数类型吧。

范本2:unknown_function(nFoo) Vs unknown_function(foo)
匈法在这里有什么收益呢?我看不到。对于一个不知道确定类型的函数,程序员应该去查看该函数的文档,这是一种成本。使用匈法的唯一好处是看代码的人知道这个函数要求一个整型参数,这又有什么用处呢?函数是一种接口,参数的类型仅仅是接口中的一小部分。诸如函数的功能、出口信息、线程安全性、异常安全性、参数合法性等重要信息还是必须查阅文档。

范本3:nFoo=nBar Vs foo=bar
匈法在这里有什么收益呢?我看不到。使用匈法的唯一好处是看代码的人知道这里发生了一个整型变量的复制动作,听起来没什么问题,可以安心睡大觉了。如果他看到的是nFoo=szBar,可能会从美梦中惊醒。且慢,事情真的会是这样吗?我想首先被惊醒的应该是编译器。另一方面,nFoo=nBar只是在语法上合法而已,看代码的人真正关心的是语义的合法性,匈法对此毫无帮助。另一方面,一个优秀的书写者会自觉地遵从一个法则:代码最小组织单位中的临时变量以一两个为宜,如果超过三个就应该重新组织。结合前述第一个法则,可以得出这样的结论:易于理解的代码本身就应该是易于理解的,这是代码的内建高质量。好的命名规范对内建高质量的助益相当有限,而坏的命名规范对内建高质量的损害比人们想象的要大。

三 匈牙利命名法的实施

这里要证明匈牙利命名法在C语言是难以实施的,在C++语言中是无法实施的。从逻辑上讲,对匈法的收益做出否定的结论以后,再来论证匈法的可行性,是画蛇添足。不过有鉴于小马哥曾让已射杀之敌死灰复燃,我还是再踏上一支脚为妙。

前面讲过,匈法是类型系统的冗余,所以实施匈法的关键是我们是否能够精确地对类型系统进行复制。这取决于类型系统的复杂性。

先来看看C语言:

1.内置类型:int,char,float,double 复制为 n,ch,f,d?好像没有什么问题。不过谁来告诉我void应该怎么表示?
2.组合类型:array,union,enum,struct 复制为 a,u,e,s?好象比较别扭。
这里的难点不是为主类型取名,而是为副类型取名。an表示整型数组?sfoo,sbar表示结构foo,结构bar?ausfoo表示联合结构foo数组?累不累啊。
3.特殊类型:pointer。pointer在理论上应该是组合类型,但是在C语言中可以认为是内置类型,因为C语言并没有非常严格地区分不同的指针类型。下面开始表演:pausfoo表示联合结构foo数组指针?ppp表示指针的指针的指针?

噩梦还没有结束,再来看看类型系统更阿为丰富的C++语言:

1.class:如果说C语言中的struct还可以用stru搪塞过去的话,不要梦想用cls来搪塞C++中的class。严格地讲,class根本就并不是一个类型,而是创造类型的工具,在C++中,语言内置类型的数量和class创造的用户自定义类型的数量相比完全可以忽略不计。stdvectorFoo表示标准库向量类型变量Foo?疯狂的念头。
2.命名空间:boostfilesystemiteratorFoo,表示boost空间filesystem子空间遍历目录类型变量Foo?程序员要崩溃了。
3.模板:你记得std::map<std::string,std::string>类型的确切名字吗?我是记不得了,好像超过255个字符,还是饶了我吧。
4.模板参数:template <class T, class BinaryPredicate>const T& max(const T& a, const T& b, BinaryPredicate comp) 聪明的你,请用匈法为T命名。上帝在发笑。
5.类型修饰:static,extern,mutable,register,volatile,const,short,long,unsigned 噩梦加上修饰是什么?还是噩梦。

你愿意做镣铐上的舞者吗?

 

除此之外,国外也有不少人讨论这个问题。请看这里

### 函数变量命名规范与最佳实践 在编程中,函数变量命名规范对于代码的可读性、可维护性协作效率至关重要。以下是关于函数变量命名的一些重要规则最佳实践: #### 变量命名规范 变量命名应当清晰明了,便于理解其用途。以下是一些常见的变量命名规则[^1]: - **描述性强**:变量名称应该能够清楚表达其所代表的数据或对象的意义。 - **大小写约定**:不同语言有不同的惯例。例如,在 Python 中推荐使用小写字母加下划线的方式(snake_case),而在 JavaScript 或 Java 中则更倾向于驼峰式命名法(camelCase)。 - **避免歧义**:不要使用容易引起误解的名字,比如单字母变量除非是在循环计数等非常短且明确的情况下。 #### 函数命名规范 函数作为程序逻辑的核心组件之一,其命名同样需要遵循一定的标准来提升代码质量。具体如下[^3][^4]: - **动词开头**:为了体现动作或者行为,建议以动词开始定义函数名,这样可以帮助开发者快速识别该方法的功能是什么。 - **全小写并用下划线分割单词**:这符合 PEP8 对于 Python 的指导原则,有助于保持一致性。 - **避开关键字冲突**:确保所选名称不会与任何已存在的类库成员或其他内置功能发生重叠。 - **意义明确**:如同变量一样,好的函数名字可以让读者无需查看内部实现就能大致猜到它的作用范围及其返回值类型。 下面给出一段基于上述准则编写的小型 Python 程序片段展示如何应用这些概念: ```python def calculate_total_price(items, tax_rate): total = sum(item['price'] * item['quantity'] for item in items) taxed_amount = total * (tax_rate / 100) final_total = total + taxed_amount return round(final_total, 2) shopping_cart = [ {"product": "apple", "price": 0.99, "quantity": 5}, {"product": "banana", "price": 0.75, "quantity": 3} ] sales_tax_percentage = 8.25 total_cost = calculate_total_price(shopping_cart, sales_tax_percentage) print(f"The total cost including taxes is ${total_cost}") ``` 此示例展示了恰当运用有意义的变量函数命名可以使整个脚本更加直观易懂。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值