| 网站首页 | 业界新闻 | 小组 | 威客 | 人才 | 下载频道 | 博客 | 代码贴 | 在线编程 | 编程论坛
欢迎加入我们,一同切磋技术
用户名:   
 
密 码:  
共有 1894 人关注过本帖
标题:C++ 编程技巧
只看楼主 加入收藏
yuyunliuhen
Rank: 6Rank: 6
等 级:贵宾
威 望:20
帖 子:1435
专家分:0
注 册:2005-12-12
结帖率:100%
收藏
 问题点数:0 回复次数:27 
C++ 编程技巧

C++语言是一个面向对象的语言,使用C++编写的代码更加简捷、高效,更具可维护性和可重用性。但是很多人使用了C++语言后却感到C++与C编程没有什么区别。这其实是由于对C++语言的特点和特色理解和使用不够造成的。事实上,没有任何一个程序员使用C语言的编程效率可以超过C++语言的。
一、使用new和delete进行动态内存分配和释放
运算符new和delete是C++新增的过算符,提供了存储的动态分配和释放功能。它的作用相当于C语言的函数malloc()和free(),但是性能更为优越。使用new比使用malloc()有以下的几个优点:
(1)、new自动计算要分配类型的大小,不使用sizeof运算符,比较省事,可以避免错误。
(2、它自动地返回正确的指针类型,不用进行强制指针类型转换。
(3)、可以用new对分配的对象进行初始化。
使用例子:
(1)、int *p;
p=new int[10]; //分配一个含有10个整数的整形数组
delete[] p; //删除这个数组
(2)、int *p;
p=new int (100);//动态分配一个整数并初始化
二、使用inline内连函数替代宏调用
对于频繁使用的函数,C语言建议使用宏调用代替函数调用以加快代码执行,减少调用开销。但是宏调用有许多的弊端,可能引起不期望的副作用。例如宏:
#define abs(a) ((a)<0?(-a):(a)), 当使用abs(I++)时,这个宏就会出错。
所以在C++中应该使用inline内连函数替代宏调用,这样即可达到宏调用的功能,又避免了宏调用的弊端。
使用内连函数只需把inline关键字放在函数返回类型的前面。例如:
inline int Add(int a,int b);//声明Add()为内连函数
这样编译器在遇到Add()函数时,不再进行函数调用,而是直接嵌入函数代码以加快程序的执行。
三、使用函数重载
在C语言中,两个函数的名称不能相同,否则会导致编译错误。而在C++中,函数名相同而参数不同的两个函数被解释为重载。例如:
void PutHz(char *str); //在当前位置输出汉字
void PutHz(int x,int y,char *str); //在x,y处输出汉字
使用函数重载可以帮助程序员应付更多的复杂性,避免了使用诸如intabs()、fabs()、dabs()等繁杂的函数名称;同时在大型程序中,使函数名易于管理和使用,而不必绞尽脑汁地去处理函数名。
四、使用引用(reference)代替指针进行参数传递
在C语言中,如果一个函数需要修改用作参数的变量值的时候 ,参数应该声明为指针类型。例如:
void Add(int *a) {(*a)++;}
但是对于复杂的程序,使用指针容易出错,程序也难以读懂。在C++中,对于上述情况 可以使用引用来代替指针,使程序更加清晰易懂。引用就是对变量取的一个别名,对引用进行操作,就相当于对原有变量进行操作。,例如使用引用的函数定义为:
void Add(int &a) (a++;); //a为一个整数的引用
这个函数与使用指针的上一个函数的功能是一样的,然而代码却更为简洁和清晰易懂。
五、使用缺省参数
在C++中函数可以使用缺省参数,例如:
void PutHzxy(char *str,int x=-1,int y=-1)
{ if (x==-1) x=wherex();
if (y==-1) y=wherey();
moveto(x,y)
PutHz(str);}
可以有两种方式调用函数PutHzxy(),例如:
PutHzxy("C++语言");//使用缺省参数,在当前位置输出
PutHzxy("C++语言",10,10);//没有使用缺省参数
通常的情况下,一个函数应该尽可能地具有更大的灵活性,使用缺省参数为程序员处理更大的复杂性和灵活性提供了有效的方法。所以在C++的代码中都大量的使用了缺省参数。
需要说明的是,所有的缺省参数必须出现在不缺省参数的右边。亦即,一旦开始定义取缺省数值的参数,就不可再说明非缺省的参数。
例如:
void PutHzxy(char *str,int x=-1,int y=-1); //正确
void PutHzxy(int x=-1,int y=-1,char *str);//错误
六、使用“类”对数据进行封状
C语言是模块化的程序语言,通过函数的使用和文件的单独编译实现了一定的数据封装功能。但C++通过使用“类”的强大功能,在数据封装、继承等很多的方面比C做得更好。通过使用“类”把数据和对数据的所有操作集合封装在一起,建立了一个定义良好的接口,使程序员在使用一个类的时候可以只关心它的使用,而不必关心它的实现。
由于函数也可一定程度上实现对数据的封装,在编写C++程序时何时使用函数,何时使用类,对于C++的初学者难以把握。根据笔者的经验,对于函数和类的使用总结出以下的方法:
首先把程序需要完成的功能划分为很多的基本子过程,一个子过程实现一种相对完整的功能。然后根据如下的规则进行划分:
(1)、如果有一些数据被两个以上的子过程同时使用,应该把这些数据和这些子过程使用“类”进行封装。
(2)、如果一些数据只被一个子过程使用,应把这些数据和这个子过程合成一个函数。这些数据声明为这个函数的内部临时数据。
(3)、如果一些数据被一个子过程在不同的时间里几次使用,应把这些数据和这个子过程合成一个函数。这些数据被定义为这个函数的内部静态数据。
(4)、如果一个子过程的功能在以后可能被修改或扩展,应该把这些过程及其使用的数合成一个类,以便以后使用继承的方法对其功能进行修改和扩充。
(5)、当(2)、(3)和(4)矛盾时,以(4)为准。
例如,对于在C++中使用鼠标这一程序含有10多个子过程,诸如MouseOpen(),MouseHide()等等。如果是调用DOS的33H中断来实现,因为在程序中各个子过程之间没有共同使用的数据,所以应该把每个子过程定义为函数。
又如,如果定义了一个表示图形的数据结构,对于这个图形要进行放大、移动、旋转等子过程。因为这些子过程都要使用公共的图形数据,所以应该把这些子过程和这些图形数据定义为一个类。
七、使用模板和BIDS
在Borland C++ 3.1中还引入模板(template)的功能,通过模板Borland C++ 3.1实现了功能强大的BIDS(Borland International Data Structures)。使用BIDS可以不需编程实现可以存储任何数据类型的数组、链表、椎栈、队列等数据结构。下面的例子实现了一个存储整形变量的堆栈:
typedef BI_StackAsVector <int> intstack;
main()
{instack is;//定义一个整形变量的堆栈
for(int I=0;I<10;I++)
is.push(I);//10个数压栈
for(I=0;I<10;I++)
cout<<is.ppop()<<end1; //10个数出栈
}
通过语句is.push(),is.pop()可以对堆栈进行操作。对BIDS的使用可以参照《Borland c++ 3.0程序员指南》。
本文以Borland C++ 3.1为背景,但是适用于大多的C++编译器。
搜索更多相关主题的帖子: 技巧 
2006-12-23 22:23
song4
Rank: 7Rank: 7Rank: 7
等 级:贵宾
威 望:38
帖 子:1533
专家分:4
注 册:2006-3-25
收藏
得分:0 

不管怎么说,支持一下是对的


嵌入式 ARM 单片机 驱动 RT操作系统 J2ME LINUX  Symbian C C++ 数据结构 JAVA Oracle 设计模式 软件工程 JSP
2006-12-24 00:26
song4
Rank: 7Rank: 7Rank: 7
等 级:贵宾
威 望:38
帖 子:1533
专家分:4
注 册:2006-3-25
收藏
得分:0 
希望连载,可以加精

嵌入式 ARM 单片机 驱动 RT操作系统 J2ME LINUX  Symbian C C++ 数据结构 JAVA Oracle 设计模式 软件工程 JSP
2006-12-24 00:27
anlzy
Rank: 1
等 级:新手上路
帖 子:33
专家分:0
注 册:2006-12-16
收藏
得分:0 

我就是不能理解C和C++不一样的地方,怎么说呢,就是那一种好象心里知道了,但一想又更糊涂了!郁闷啊

2006-12-27 21:41
yuyunliuhen
Rank: 6Rank: 6
等 级:贵宾
威 望:20
帖 子:1435
专家分:0
注 册:2005-12-12
收藏
得分:0 

成员函数的三种调用方式

非静态成员函数
非静态成员函数的效率和非成员函数相同,也就是说

float magnitude(const Point3d * _this) {…};

float Point3d::magnitude() const {…};

的效率相同。

成员函数的转化步骤如下(编译器所做的工作):
改写函数原型

//非静态成员函数

Point3d::magnitude(Point3d * const this);

//常量非静态成员函数

Point3d::magnitude(const Point3d * const this);

对非静态数据成员的存取改由this指针完成

{this->xxxxx}

把成员函数改写为一个外部函数,对函数名称进行处理,使它在程序中独一无二


虚成员函数
通过指针调用

Point3d *ptr;

……

ptr->normalize();

在内部将被转化成为(*ptr->vptr[1])(ptr);

通过对象调用

Point3d obj;

Obj.normalize();

由于编译器可以确定调用的肯定是Point3d::normalize(),因此在内部这样的调用将被转化为:

normalize_7Point3dEv(&Obj);//经过编码的函数名称,在整个程序中保持唯一性。

静态成员函数
Point3d *ptr;

……

Point3d obj;

ptr->normalize();

Obj.normalize();

在内部都将被转化为normalize_7Point3dEv();

静态成员函数的主要特征是没有this指针,其次要特征为

1. 不能够直接存取Class中的非静态成员

2. 不能被声明为const、volatile或vitrual

不用经过Class Object才能够调用


Go confidently in the  directions of your dreams,live the life you have imagined!Just do it!
It is no use learning without thinking!
2006-12-27 22:30
yuyunliuhen
Rank: 6Rank: 6
等 级:贵宾
威 望:20
帖 子:1435
专家分:0
注 册:2005-12-12
收藏
得分:0 

C++的成员函数指针简要

C++的成员变量指针和普通的指针没有什么大的区别,int类型成员变量的指针就是和int *, Foo类型的成员变量的指针就是Foo *,但是成员函数就不大一样了,主要是因为非静态成员函数实际上都隐藏这一个调用对象的指针的参数。

静态成员函数指针,和普通的C函数指针没什么两样
class Foo
{
public:
static int foo();
};

int (*fp)() = Foo::foo();
因为静态成员函数的执行和类的对象无关,也没有隐藏的对象指针参数

非静态成员函数,因为有隐藏的对象指针参数,就有点不一样
class Foo
{
int foo1();
int foo2(long);
};
为了代码的简洁,最好用typedef先定义指针类型
typedef int (Foo::*fpType1)(); //Foo类返回类型为int,参数列表为空的成员函数指针类型
typedef int (Foo::*fpType2)(long); //Foo类返回类型为int,参数列表为long的成员函数指针类型
然后就可以声明指向成员函数的指针
fpType1 fp1 = &Foo::foo1;
fpType2 fp2 = &Foo::foo2;
通过成员函数调用的时候注意,因为非静态的成员函数执行实际上是需要一个类对象的
Foo f;
int result = (f.*fp1)();
或者通过对象指针
Foo f2 = &f;
int result2 = (f2->*fp1)();



Go confidently in the  directions of your dreams,live the life you have imagined!Just do it!
It is no use learning without thinking!
2006-12-27 22:35
yuyunliuhen
Rank: 6Rank: 6
等 级:贵宾
威 望:20
帖 子:1435
专家分:0
注 册:2005-12-12
收藏
得分:0 

友元函数
对友元函数新的了解友元函数其实也就是为了访问类中的数据成员. 如果函数要访问两个甚至更多类中的数据成员时,这个函数应该被它所访问所有类数据成员的类声明为友元函数 For example: -------------------------------------------------------------------------------------------------------------------------------

#include <iostream>

using namespace std;

class B; //先声明,因为A类中的友元函数参数引用了B类
class A
{
public:
A(int aa = 0, int bb = 0){a = aa; b = bb;}
friend int Addab(A &Ta, B &Tb); //友元函数,因为Addab函数调用了A中私有成员a和b
private:
int a;
int b;
};

class B
{
public:
B(int ii = 0, int jj = 0){i = ii; j = jj;}
friend int Addab(A &Ta, B &Tb); //友元函数,因为Addab函数调用了B中私有成员i和j

private:
int i;
int j;
};

int Addab(A &Ta, B &Tb)
{
int p = Ta.a + Ta.b;
int k = Tb.i + Tb.j;
int re = p + k;
return re;
}

int main()
{
A a;
B b;
int result = Addab(a, b);
cout << result << endl;
return 0;
}


Go confidently in the  directions of your dreams,live the life you have imagined!Just do it!
It is no use learning without thinking!
2006-12-27 22:39
jishuai
Rank: 1
等 级:新手上路
帖 子:97
专家分:0
注 册:2006-12-15
收藏
得分:0 

呵呵
我还没有开始看类啊


2006-12-27 22:43
高达
Rank: 1
等 级:新手上路
威 望:1
帖 子:261
专家分:0
注 册:2006-10-27
收藏
得分:0 

学习


哎 时间....................
2006-12-28 17:41
一二三四五
Rank: 3Rank: 3
等 级:新手上路
威 望:8
帖 子:856
专家分:0
注 册:2006-11-13
收藏
得分:0 
不错,学习了


hey,di va la
2006-12-28 17:55
快速回复:C++ 编程技巧
数据加载中...
 
   



关于我们 | 广告合作 | 编程中国 | 清除Cookies | TOP | 手机版

编程中国 版权所有,并保留所有权利。
Powered by Discuz, Processed in 0.016289 second(s), 7 queries.
Copyright©2004-2024, BCCN.NET, All Rights Reserved