学用C++的体会
我在大学的时候,花了一个月时间学了C语言,后来,基本上做什么,都不用翻书了
后来看C++,觉得挺简单,学了一周,想不就是在C语言的基础上,加了一个class吗,工作了一年看的书多了,用的多了,才真正发现自己是井底之蛙。以前,看了所有的VC的书,其实,现在发现VC!=C++,vc只是一个编程环境啊。C++是一门语言。C++的精髓之区在面向对象的思想,为什么可以定义一个类,如何继承一个。说到底是面向对象分析,面向对象设计,也就是OOA/OOD,C++的复杂在于它的范围很广,因为,我们工作往往只能用到它的一部分啊。比如我们很少用模板,很少用那些高级的应用,其实,说到底,是我们的水平不够,大家看看loki库也许就知道,template有多大的威力。我个人认为,template,跟虚函数一样,属于多态的,不过,虚函数是通过指针,实现,在运行时候,进行解析,而template是编译期解析,我想它的实现跟宏定义差不多,但是,更加高级。
还有一个问题,C++的编译器其实为我们的代码做了很多的事情。如果,我们不知道的话,那么,就会常常出现错误,我们应该始终记住这么一句话:"通用的往往是不好的,不适合工程的"。比如说:
函数为什么能够重载?
其实 ,C++,是这样实现函数的名称对应的
比如:
class A
{
show(int a,int b);
show(float a);
};
其实,它在内部是这样对应的:伪代码:
show_INT_INT(int a,int b, void *this);
show_FLOAT(float a,void *this);
(大家想想为什么会有一个this指针)
这样,自然,show()重载多少个都没有问题,从这也可以看出,内部实现,只是把函数的参数类型编进了函数名来实现重载,而没有把返回值编入,因此,函数名一样,参数个数一样,类型对应一样的,返回值不同的两个函数不能构成重载,会出现编译错误。(我们可以想想为什么不把返回值编入函数名里,如果,只有两个函数返回值不同,比如如下:
class B
{
public:
int print(int a);
void print(int a);
}
如果,把返回值也编入函数名的话,那么它们应该可以构成重载。
当你这样如下调用时候,
B a;
a.print(20);
编译器怎么知道它是调用那个函数。
)
2,函数参数传入传出的规则
在函数调用的时候,有三种情况,第一种参数为数值,第二种为传指针,第三种传引用,此外,还有,为const,和non-const的区别。
什么情况下该传数值,什么情况下不该传呢
我觉得如果参数是基本类型的话,没有关系,如果参数是对象的话,就不能传数值,因为,它会调用对应的构造函数。(知道为什么吗)
解释:函数参数的传入,在函数的内部会产生几个临时变量,来拷贝函数的参数数值。
比如 fun(X a)
{
}
那么,在函数对应的堆上,会为a变量开辟空间(m_a),并调用X类的拷贝构造函数来初始化化m_a,如下:X,m_a(a);类似这样。有时候,一个类包含很多变量,构造函数比较复杂,那么,这样的调用fun()会浪费很多系统时间。因此,极力反对用传值调用对象参数的函数。
还有一个,大家经常遇到的问题,就似const 修饰参数与没有const的区别。
有了const的修饰,在函数内部就不会为参数产生临时变量了,这样,自然,不会调用什么参数类的构造函数了,也不会为参数分配空间了。同时,这样的参数也不能在函数中改变它的数值。
3,拷贝构造函数
我们往往忘了写这样一个函数,也许不需要。其实,我们没有写的时候,不过,编译器自动为我们生成一个。
比如:
class A{
private:
int x;
char *str_name;
public :
A(int x,char *str);
};
在这里,我们没有定义一个拷贝构造函数,如果出现这样的例子
fun(A x)
{
}
从上边的例子,我们知道,调用fun()函数的时候,会要调用A类的拷贝构造函数的 这样,如果,没有定义的话,那么,系统产生一个拷贝构造函数(按照位拷贝
变量的值),这样,我们知道,str_name只是一个指针,当临时变量的str_name成员也指向参数的str_name的位置时候,这样,我们就踏上了,一个地雷阵了。为什么,我们想啊,当执行完,fun()函数的时候,临时变量,在某个不确定的时刻就要free()掉,这样,临时变量str_name也会delete,而我们知道,这个str_name跟参数的str_name指向的同一个内存块,那么,当free一次后,那么它的内容就没有了这样,破坏了参数数据内容。
很多东西,一下子说不完,下次再说。