C++学习 —— 住着魔鬼的细节

  13周的C++课程转眼就学完了5周,C++的标准基本上已经覆盖到了。再加上coding了上百行,总算是对C++有了一个基本的了解。接下来的学习会是关于STL的,所以在此对目前所学做一个小的总结。

  the devil is in the detail这是我最大最大的体会。由其是对于这种称之为标准的内容而言。更是如此。

1、赋值,引用,指针

  这三个概念是很多编程语言里面共有的,为了实现c++的高效率,这3板斧更是被发挥的淋漓尽致。总结一下,可以分为两种情况:【传】【返回】。

  【传】赋值的意思就是对变量进行复制  fun(int a), fun(5)。 matlab 里全部都是这样的调用,matlab 并不能传引用。这样做的好处是函数不会修改传入值,坏处是需要对传入的参数进行复制,如果参数很大(e10的点云),那么复制会有很大开销。

  【传】引用的意思是将变量的引用传入,比如 fun(int & a) , int b; fun(b)。这样的做法可以在函数中改变b的值。好处是直接传了地址,不需要复制开销,坏处是不利于变量保护。

  【传】指针和传引用类似,一般情况下,指针是一种很丑的参数,但是有的时候又不得不以指针作为参数。比如点云,往往点云的规模是动态的,内存也是动态分配的,所以很多点云处理函数的参数都是PointCloud<PointXYZ>::ptr,传指针是会改变指针所指向的内容的。

 

  【返回】值的意思是函数返回一个值,往往是 a = fun(b),被返回的值只能做右值。

  【返回】引用的意思是返回一个变量,这个变量可以有值,也可以未被初始化。它是可以作为左值的,比如 a.getobj() = createobj(A). 或者 ostream & operator <<(ostream & o){return o;  } cout<<MY_OBJ<<endl; 

  【返回】指针,返回一个指针也是无奈之举,比如输入的是一个点云,返回它的分割子集。

  总而言之,返回引用多半是为了做左值,返回指多半是因为有开销上的顾虑。

 

2、const 与数据保护

  const 有三种用法,分别是和数结合,和指针结合,和函数结合。

  【和数结合】和数结合的情况非常单纯,意思就是这个变量不能改了。 int c, const int a =5; const int & b = c; 这样的情况下a = 1, b= 1都会报错,如果想要b = 1,可以令c = 1;

  【和指针结合】和指针结合的情况有点点复杂,分两种 const int * p = & c; 表示常int,这是意思是c的值不能通过p 来修改,和上面的常引用是一样的意思!! int const *p = & c 则表示常*p,意思就是p不能指向其他数据。

  【和函数结合】和函数结合要更复杂那么一点点。分两种情况,fun(const int & a),表示a 不能被修改。fun(int a) const{},const 修饰{}——常成员函数,表示成员函数不允许修改成员变量。常对象只能调用这种常成员函数。

 

3、构造与析构

  构造与析构是面向对象的路子。主要是构造函数有一点复杂,分缺省构造,普通构造,拷贝构造。

  【缺省构造】如果我们不写构造函数,那么会自动生成缺省构造函数。但是一旦写了,那么就没有缺省构造函数了。如果你想要没参数的构造函数,要么用缺省值伪装,要么自己写上 myclass(){}。建议后者,有时候不记得搞了缺省值会出事的。

  【普通构造】构造函数的作用是初始化成员变量,如果成员变量是私有的,那很可能只有这么一次机会初始化。

  【拷贝构造】拷贝构造是最啰嗦的,它有两种调用形式:1、myclass b ;  myclass a(b); 2、myclass a = b。 第二种看似用了操作符 = ,实际不是的,它调了拷贝构造函数,而不是操作符重载。如果你重载了=,最好别有二义性。

  【析构函数】析构函数最重要的是,如果用new 申请了变量,那么一定要在析构函数里delete []。当然,如果用boost 的 智能指针,可以不用手动释放。

 

4、操作符重载

  操作符重载还是比较好理解的,记好一个细节,左值要返回引用。

 

5、继承

  继承是 is 的操作,如果一个类继承了其夫类,那么子类的对象,在内存里会有父类的所有成员变量!并且自动获得父类所有的成员函数。不过获得的成员函数只能操作内存里父类的那些成员变量了。

  设计继承最重要的是子类必须获得父类所有的性质。如果父类有一个行为是子类不需要的,那么就算设计的不好了。

  子类中比较有难度的是构造函数的设计,因为要对内存里父类的那些成员变量赋值,所以其构造函数要写成如下:son(int a,int b,int c):father(a,b),c_(c){}。直接用父类的类名,来完成成员变量的赋值。一个简单的继承例子如下:

 

 1 #include <iostream>
 2 #include <cstring>
 3 #include <cstdlib>
 4 using namespace std;
 5 
 6 class MyString:public string
 7 {
 8 public:
 9     MyString(const MyString & s):string(s){};
10     MyString(const string & s):string(s){};
11     MyString(const char * s):string(s){};
12     MyString(){};
13 
14     MyString operator()(int start_index,int length){
15 
16     MyString substr  =  this->substr(start_index,length);
17     return substr;
18 
19 }
20 
21     ~MyString(){};
22 };
23 
24 
25 
26 int CompareString( const void * e1, const void * e2) {
27     MyString * s1 = (MyString * ) e1;
28     MyString * s2 = (MyString * ) e2;
29     if( *s1 < *s2 ) return -1;
30     else if( *s1 == *s2 ) return 0;
31     else if( *s1 > *s2 ) return 1;
32 }
33 
34 int main() {
35     MyString s1("abcd-"),s2,s3("efgh-"),s4(s1);
36     MyString SArray[4] = {"big","me","about","take"};
37     cout << "1. " << s1 << s2 << s3<< s4<< endl;
38     s4 = s3; s3 = s1 + s3;
39     cout << "2. " << s1 << endl;
40     cout << "3. " << s2 << endl;
41     cout << "4. " << s3 << endl;
42     cout << "5. " << s4 << endl;
43     cout << "6. " << s1[2] << endl;
44     s2 = s1; s1 = "ijkl-";
45     s1[2] = 'A' ;
46     cout << "7. " << s2 << endl;
47     cout << "8. " << s1 << endl;
48     s1 += "mnop";
49     cout << "9. " << s1 << endl;
50     s4 = "qrst-" + s2;
51     cout << "10. " << s4 << endl;
52     s1 = s2 + s4 + " uvw " + "xyz";
53     cout << "11. " << s1 << endl;
54     qsort(SArray,4,sizeof(MyString), CompareString);
55     for( int i = 0;i < 4;++i )
56         cout << SArray[i] << endl;
57     //输出s1从下标0开始长度为4的子串
58     cout << s1(0,4) << endl;
59     //输出s1从下标为5开始长度为10的子串
60     cout << s1(5,10) << endl;
61     return 0;
62 }
View Code

  这里,MyString 虽然没说,但是获得了string所有的操作符重载(=,+....)还有所有的函数(substr)。唯一需要好好设计的就是构造函数。

  

posted @ 2016-07-10 21:00  IronStark  阅读(663)  评论(0编辑  收藏  举报