热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

如何写优雅的代码(3)——合理选择函数形参

TITLE:

    //========================================================================
    //TITLE:
    //    如何写优雅的代码(3)——合理选择函数形参
    //AUTHOR:
    //    norains
    //DATE:
    //    Tuesday  21-July-2009
    //Environment:
    //    WINCE5.0 + VS2005
    //========================================================================
   
    函数形参的确定,其实没有太多的讲究,一般而言,想怎么用就怎么用。但从C升级到C++的时候,偏偏给我们多出个引用。俗语道,当方法只有一种时,那是最幸福的。当我们需要改变形参的数值时,究竟是选择指针还是应用,这是我们追求优雅不得不考虑的幸福烦恼。
       
    以经典的交换两个int数值的代码做样例。
   
    采用指针的写法:

 

    采用引用的写法:

 

 

  调用的话,也是很简单:

 

  从功能的角度而言,无论是指针还是引用,都能实现同样的功能。很显然,我们不能通过能否实现某个功能来确定是采用引用还是指针。
  
  我们再来看看这两个函数,最大的区别在于指针还需要判断是否为NULL——对于指针而言,对为NULL值的指针提领,铁定会出现异常。故在比较之前,一定要判断指针的数值。也正是因为这个原因,以指针作为形参的Swap函数还有失败的可能性。但对于引用而言,则没有这个弊端,因为引用不能指向“无物”。由此,上述的采用引用的Swap其实根本不必要返回值,因为一定能够做到交换成功。
  
  精简后采用引用作为传递的Swap函数如下:

 

  是不是比采用指针的更为清晰明了?
  
  
  是否可以下定论,用指针传递的都应该改用引用?当然答案是否定的。一般而言,当传递NULL可以相对改变函数意义时,我们可以也应该选择指针。
  
  我们在日常编程中会经常会碰到这么一种情形,给某一个函数传入一个缓冲区,然后往缓冲区复制数据。因为每次的数据大小都是不一致的,所以我们的缓冲区必须要以new来创建;而以new来创建缓冲,那么首先要确定数据的大小。这时候,我们就可以传入NULL值,告诉函数这次我们不需要拷贝数据,仅仅是想知道所需的大小而已。
  
  根据该情形,我们可以写出类似函数的模型:

 

  调用的时候:

 

  这样,我们就能够在不增加函数,不增加形参的情形下,只是通过NULL为标志确定缓冲区大小,进而分配相应的空间进行数据拷贝。
  
  细心的朋友看到这里,可能会说,这用引用也可以实现啊!因为对于这个函数而言,当传入的参数为0时,意味缓冲没意义,这个也可以作为标志啊!
  
  这当然也可以,所以CopyBuff就可以这样改:

 

  看起来是不是要比使用指针清爽?但问题不出在CopyBuffer函数,而在于调用上。
  
  因为CopyBuffer为引用,所以我们不能想这样调用:

 

   这段代码会引发编译器会发生抱怨:error C2664: 'CopyBuffer' : cannot convert parameter 1 from 'BYTE *' to 'BYTE &'
  
   为了能让编译器安静,我们只能这样:

 

   编译器是高兴了,但运行就郁闷了。因为pBuf为NULL,所以*pBuf一定会异常。
  
   为避免这异常发生,我们在调用前只能先预分配一段缓冲区,故调用代码很可能如下类似:

 

  相对之前采用指针作为形参的CopyBuffer,这样的调用显得更为累赘和繁琐。
  
  如上总结:作为函数的形参,只有当NULL有意义的时候,才选择指针;除此以外,都应该选择引用。
  
  这个结论,对于从C过渡到C++的朋友可能有点难以让人忍受。在C中,如果要改变形参的数值,就采用指针的方式,这样在调用的时候,能够在一定程度上了解该形参是否会被改变。

  如:

 

 

  而采用引用的方式,无论是否会改变形参值,在调用上都不会有任何蛛丝马迹:

 

  也许最让郁闷的是,windows的api函数,凡是用来改变形参的,都无一例外是采用指针的形式。在这方面微软有充足的理由:让C程序员也能编写windows程序。只不过,我们已经上了C++这条贼船,为什么我们不用C++的方式呢?
  
  
  
  在很多情形下,以结构体作为形参,我们往往会带有const和引用的修饰:

 

  const标明我们的态度,不对rcPos的数值进行变更;引用则是告诉编译器,不必为创建临时变量。
  
  也许这样说有点让人糊涂,还是让我们用实例来说明。当我们去掉const,去掉引用,则函数变成:

 

  如果形参不是结构体类型,而是诸如DWORD,int等,那则无不妥;但如果是像例子中的结构体,那么编译器就要忙活了:创建一个对象,这就意味着要调用构造函数,使用完毕后还要析构函数清除。我们为了让编译器不那么辛苦,所以我们给形参加上引用的符号;又因为这个数值我们实在没打算更改,所以再加上const的修饰。于是,这函数就变成了之前我们所见到的形式。
  
  末尾,稍作总结:当传递结构体作为形参,不妨考虑一下const+引用的方式。


推荐阅读
  • 使用C++编写程序实现增加或删除桌面的右键列表项
    本文介绍了使用C++编写程序实现增加或删除桌面的右键列表项的方法。首先通过操作注册表来实现增加或删除右键列表项的目的,然后使用管理注册表的函数来编写程序。文章详细介绍了使用的五种函数:RegCreateKey、RegSetValueEx、RegOpenKeyEx、RegDeleteKey和RegCloseKey,并给出了增加一项的函数写法。通过本文的方法,可以方便地自定义桌面的右键列表项。 ... [详细]
  • 本文介绍了在Windows环境下如何配置php+apache环境,包括下载php7和apache2.4、安装vc2015运行时环境、启动php7和apache2.4等步骤。希望对需要搭建php7环境的读者有一定的参考价值。摘要长度为169字。 ... [详细]
  • C++字符字符串处理及字符集编码方案
    本文介绍了C++中字符字符串处理的问题,并详细解释了字符集编码方案,包括UNICODE、Windows apps采用的UTF-16编码、ASCII、SBCS和DBCS编码方案。同时说明了ANSI C标准和Windows中的字符/字符串数据类型实现。文章还提到了在编译时需要定义UNICODE宏以支持unicode编码,否则将使用windows code page编译。最后,给出了相关的头文件和数据类型定义。 ... [详细]
  • 本文介绍了如何使用C#制作Java+Mysql+Tomcat环境安装程序,实现一键式安装。通过将JDK、Mysql、Tomcat三者制作成一个安装包,解决了客户在安装软件时的复杂配置和繁琐问题,便于管理软件版本和系统集成。具体步骤包括配置JDK环境变量和安装Mysql服务,其中使用了MySQL Server 5.5社区版和my.ini文件。安装方法为通过命令行将目录转到mysql的bin目录下,执行mysqld --install MySQL5命令。 ... [详细]
  • 本文介绍了深入浅出Linux设备驱动编程的重要性,以及两种加载和删除Linux内核模块的方法。通过一个内核模块的例子,展示了模块的编译和加载过程,并讨论了模块对内核大小的控制。深入理解Linux设备驱动编程对于开发者来说非常重要。 ... [详细]
  • 本文介绍了Swing组件的用法,重点讲解了图标接口的定义和创建方法。图标接口用来将图标与各种组件相关联,可以是简单的绘画或使用磁盘上的GIF格式图像。文章详细介绍了图标接口的属性和绘制方法,并给出了一个菱形图标的实现示例。该示例可以配置图标的尺寸、颜色和填充状态。 ... [详细]
  • 本文讨论了在VMWARE5.1的虚拟服务器Windows Server 2008R2上安装oracle 10g客户端时出现的问题,并提供了解决方法。错误日志显示了异常访问违例,通过分析日志中的问题帧,找到了解决问题的线索。文章详细介绍了解决方法,帮助读者顺利安装oracle 10g客户端。 ... [详细]
  • DSP中cmd文件的命令文件组成及其作用
    本文介绍了DSP中cmd文件的命令文件的组成和作用,包括链接器配置文件的存放链接器配置信息、命令文件的组成、MEMORY和SECTIONS两个伪指令的使用、CMD分配ROM和RAM空间的目的以及MEMORY指定芯片的ROM和RAM大小和划分区间的方法。同时强调了根据不同芯片进行修改的必要性,以适应不同芯片的存储用户程序的需求。 ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • Android源码深入理解JNI技术的概述和应用
    本文介绍了Android源码中的JNI技术,包括概述和应用。JNI是Java Native Interface的缩写,是一种技术,可以实现Java程序调用Native语言写的函数,以及Native程序调用Java层的函数。在Android平台上,JNI充当了连接Java世界和Native世界的桥梁。本文通过分析Android源码中的相关文件和位置,深入探讨了JNI技术在Android开发中的重要性和应用场景。 ... [详细]
  • Redis底层数据结构之压缩列表的介绍及实现原理
    本文介绍了Redis底层数据结构之压缩列表的概念、实现原理以及使用场景。压缩列表是Redis为了节约内存而开发的一种顺序数据结构,由特殊编码的连续内存块组成。文章详细解释了压缩列表的构成和各个属性的含义,以及如何通过指针来计算表尾节点的地址。压缩列表适用于列表键和哈希键中只包含少量小整数值和短字符串的情况。通过使用压缩列表,可以有效减少内存占用,提升Redis的性能。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • 在Oracle11g以前版本中的的DataGuard物理备用数据库,可以以只读的方式打开数据库,但此时MediaRecovery利用日志进行数据同步的过 ... [详细]
  • ***byte(字节)根据长度转成kb(千字节)和mb(兆字节)**parambytes*return*publicstaticStringbytes2kb(longbytes){ ... [详细]
  • 本文总结和分析了JDK核心源码(2)中lang包下的基础知识,包括常用的对象类型包和异常类型包。在对象类型包中,介绍了Object类、String类、StringBuilder类、StringBuffer类和基本元素的包装类。在异常类型包中,介绍了Throwable类、Error类型和Exception类型。这些基础知识对于理解和使用JDK核心源码具有重要意义。 ... [详细]
author-avatar
yuhao
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有