分类: C/C++
2008-04-14 23:49:04
来源:vcer.net |
摘要
C++技术固然是很时髦的,许多C用户都想在尽可能短的时间内为自己贴上C++的标签。介绍C++的书很多,但只有那些已经侥幸入门的用户才偶尔去翻翻,仍有不少在C++门口徘徊的流浪汉。
本文只针对C用户,最好是一位很不错的老用户(譬如他在遇到最简单的问题时都尝试着使用指针),通过一些C和更好的C++(本文用的是Borland C++3.1版本)例程介绍有关C++的一些知识,让读者朋友们“浅入深出”,轻轻松松C to C++!
一、标签!标签!
快快为你的程序贴上C++的标签,让你看起来很像个合格的C++用户……
1.注释(comment) Hello-world!
如果你尝试着在test0l. exe中找到这些高级的注释,很简单,它们不会在那里的。
2. cincout
你可能从test0l中嗅出什么味儿来了,在C++中,其次的贵族是cout,而不是很老土的printf ( )。左移操作符’<<’的含义被重写,称作“输出操作符”或“插入操作符”。你可以使用’<<’将一大堆的数据像糖葫芦一样串起来,然后再用cout输出:
如何来输出一个地址的值呢?在C中可以通过格式控制符”%p”来实现,如: 类似地,C++也是这样: 但对字符串就不同啦!因为: 只会输出String的内容。但方法还是有的,如采取强制类型转换: cin采取的操作符是’>>’,称作“输入操作符”或“提取操作符”。在头文件iostream.h中有cin cout的原型定义,cin语句的书写格式与cout的完全一样: 看到了?别再傻傻地送一个scanf()常用的’&’地址符给它。
C++另外提供了一个操纵算子endl,它的功能和’\n’完全一样,如test0l中的cout语句可改版为: 而C++抛弃这点可读性,允许用户采取更自由的书写形式:
即时声明常见于for循环语句中:
这种形式允许在语句段中任点声明新的变量并不失时机地使用它(而不必在所有的声明结束之后)。
特别地,C++强化了数据类型的类概念,对于以上出现的”int i=1 j=2;”完全可以写成: 再如:
这不属于“即时声明”的范畴,但这些特性足以让你的代码与先前愚昧的C产品区别开来。
4.作用域(scope)及其存取操作符(scope qualifier operator) 作用域对应于某一变量的生存周期,它通常表现为以下五种:
块作用域 函数作用域 函数原型作用域 文件作用域 类作用域 具有不同作用域的变量可以同名,如test02:
编译器并未给出错误信息。
作用域与可见性并不是同一概念,具有作用域不一定具有可见性,而具有可见性一定具有作用域。
在test02中,float i的使用使全局int i失去可见性,这种情形被称作隐藏(override)。但这并不意味着int i失去了作用域,在main()函数运行过程中,int i始终存在。
有一种办法来引用这丢了名份的全局i,即使用C++提供的作用域存取操作符::,它表示引用的变量具有文件作用域,如下例程:
在上例中,通过::操作符,第8行语句偷偷地改写了i所属的性别。更妙的是,::之前还可以加上某些类的名称,它表示引用的变量是该类的成员。
5. new delete C++允许用户使用这些函数,但它同时也提供了两个类似的操作符new和delete,它们分别用来分配和释放内存,形式如下: 因此以上的cp操作可改版为: new delete操作符同样亦可作用于C中的结构变量,如: 当不能成功地分配所需要的内存时,new将返回NULL。对于字符型变量可以如下初始化: 对应地,new可以同时对变量的值进行初始化,如: new不需要用户再使用sizeof运算符来提供尺寸,它能自动识别操作数的类型及尺寸大小,这虽然比malloc)函数聪明不了多少,但起码使用起来会比它方便得多。当然,正如calloc()函数,new也可以用于数组,形式如下: 对应的内存释放形式: 同理首例中ip操作可以改版为: 用new分配多维数组的形式为: 从来没有太快活的事情,例如以下使用非法: C++最多只允许数组第一维的尺寸(即c0)是个变量,而其它的都应该为确定的编译时期常量。使用new分配数组时,也不能再提供初始值: 6.引用(reference)
(1)函数参数引用 在上例中,参数的调用仍采取的是拷贝传值方式,Swap()拷贝一份实参的值(其中的内容即a b的地址),但这并不表明vapvbp与实参apbp占据同一内存单元。
对实际数据操作时,传统的拷贝方式并不值得欢迎,C++为此提出了引用方式,它允许函数使用实参本身(其它一些高级语言,如BASIC FORTRAN即采取这种方式)。以下是相应的程序:
很明显,a b与vavb的地址完全重合。
对int&的写法别把眼睛瞪得太大,你顶多只能撇撇嘴,然后不动声色地说:“就这么回事!加上&就表明引用方式呗!”
(2)简单变量引用
简单变量引用可以为同一变量取不同的名字,以下是个例子: 这样定义之后,Rat就是Mouse(用中文说就是:老鼠就是老鼠),这两个名字指向同一内存单元,如: 一种更浅显的理解是把引用看成伪装的指针,例如,Mouse很可能被编译器解释成:*(& Rat),这种理解可能是正确的。
由于引用严格来说不是对象(?!),在使用时应该注意到以下几点: 当初始值是一个常量或是一个使用const修饰的变量,或者引用类型与变量类型不一致时,编译器则为之建立一个临时变量,然后对该临时变量进行引用。例如:
(3)函数返回引用
函数可以返回一个引用。观察程序test07:
这个程序利用函数返回引用写出了诸如Value (String 2) ='d’这样令人费解的语句。在这种情况下,函数允许用在赋值运算符的左边。 7.缺省参数(default value)
从事过DOS环境下图形设计的朋友(至少我在这个陷阱里曾经摸了两年时间)肯定熟悉initgraph()函数,它的原型为: 也许你会为它再定做一个函数:
一段时间下来,你肯定有了你最钟情的调用方式,例如你就喜欢使用640 * 480 * 16这种工作模式。
既然如此,你完全可以将函数InitGraph ( )声明成具有缺省的图形模式参数,如下: 这样,每次你只需简单地使用语句InitGraph ();即可进入你所喜爱的那种模式。当然,当你使用InitGraph (CGA CGAHI );机器也会毫不犹豫地切入到指定的CGAHI模式,而与正常的函数没有两样。
这就是缺省参数的用法!为了提供更丰富的功能,一些函数要求用户提供更多的参数(注意到许多Windows程序员的烟灰缸旁边都有一本很厚很厚的Windows函数接口手册),而实际上,这些参数中的某几项常常是被固定引用的,因此,就可以将它们设定为缺省参数,例如以下函数: 将可能在((x y)处以Color颜色、Mode模式画一个点,缺省情况下,颜色为黑色,写点模式为覆盖方式。
以下对函数的调用合法: 而以下调用形式并不合法: 前两种形式缺少参数,因为x、y值并没有缺省值;第三种形式则天真地以为编译器会将其处理成: 并且不会产生任何二义性问题,不幸的是,C++并不赞成这样做。
作为一条经验,缺省参数序列中最容易改变其调用值的应尽量写在前面,最可能使用其缺省值的(即最稳定的)置于后端。如果将以上函数原型声明成如下形式: 包括你自己,也不会喜欢它。 |