zoukankan      html  css  js  c++  java
  • 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)。唯一需要好好设计的就是构造函数。

      

  • 相关阅读:
    枚举类 --单例模式
    模板设计模式
    动态代理
    反射应用--修改属性值
    通过反射绕过泛型
    java反射
    网络编程练习
    TCP编程
    GUI 聊天界面
    UDP传输多线程
  • 原文地址:https://www.cnblogs.com/ironstark/p/5658548.html
Copyright © 2011-2022 走看看