• 第五章:表达式


    new 和 delete 表达式:

    耗尽内存:

    自由存储区总有可能耗尽,new表达式就有可能失败,无法获取需要的内存空间,系统将抛出名为 bad_alloc 的异常。

    1.动态创建和释放数组。

    2.创建和释放单个对象:

    int i;

    int *pi=new int; // 返回新创建对象地址

    3.初始化变量方式实现初始化:

    int i(1024);

    int *pi=new int(1024);   // 撤销 delete pi;

    string s(10,'9');

    string *ps=new string(10,'9');

    4.默认初始化:

    string *ps=nw string;  // initialized to empty string

    int *pi=new int;  // pi points to an uninitialized int

    做值初始化:

    string *ps=new string(); // points to empty string

    int *pi=new int(); // points to 0

    5.C++保证:删除0值指针式安全的。

    int *p=0;

    delete p; //但是这么做没有意义

    悬垂指针和野指针:http://www.cnblogs.com/submarine/archive/2013/03/02/2940169.html

    5.const 对象的动态分配和回收

    const int *pci=new const int(1024);

    delete pci;

    const string *pcs=new const string; // 调用默认构造函数

    类型转换 

     之命名的强制类型转换:

    http://www.cnblogs.com/ider/archive/2011/07/22/cpp_cast_operator_part2.html

    C++提供了四个转换运算符:

    • const_cast <new_type> (expression)
    • static_cast <new_type> (expression)
    • reinterpret_cast <new_type> (expression)
    • dynamic_cast <new_type> (expression)

    const_cast (expression)

    const_cast转换符是用来移除变量的const或volatile限定符。对于后者,我不是太清楚,因为它涉及到了多线程的设计,而我在这方面没有什么了解。所以我只来说const方面的内容。

    用const_cast来去除const限定

    对于const变量,我们不能修改它的值,这是这个限定符最直接的表现。但是我们就是想违背它的限定希望修改其内容怎么办呢?

    下边的代码显然是达不到目的的: const int constant = 10;
    int modifier = constant;

    因为对modifier的修改并不会影响到constant,这暗示了一点:const_cast转换符也不该用在对象数据上,因为这样的转换得到的两个变量/对象并没有相关性。

    只有用指针或者引用,让变量指向同一个地址才是解决方案,可惜下边的代码在C++中也是编译不过的: const int constant = 21;
    int* modifier = &constant
    // Error: invalid conversion from 'const int*' to 'int*'

    (上边的代码在C中是可以编译的,最多会得到一个warning,所在在C中上一步就可以开始对constant里面的数据胡作非为了)

    把constant交给非const的引用也是不行的。 const int constant = 21;
    int& modifier = constant;
    // Error: invalid initialization of reference of type 'int&' from expression of type 'const int'

    于是const_cast就出来消灭const,以求引起程序世界的混乱。

    下边的代码就顺利编译功过了: const int constant = 21;
    const int* const_p = &constant;
    int* modifier = const_cast<int*>(const_p);
    *modifier = 7;

    传统转换方式实现const_cast运算符

    我说过标:准转换运算符是可以用传统转换方式实现的。const_cast实现原因就在于C++对于指针的转换是任意的,它不会检查类型,任何指针之间都可以进行互相转换,因此const_cast就可以直接使用显示转换(int*)来代替: const int constant = 21;
    const int* const_p = &constant;
    int* modifier = (int*)(const_p);

    或者我们还可以把他们合成一个语句,跳过中间变量,用 const int constant = 21;
    int* modifier = (int*)(&constant);

    替代 const int constant = 21;
    int* modifier = const_cast<int*>(&constant);

    为何要去除const限定

    从前面代码中已经看到,我们不能对constant进行修改,但是我们可以对modifier进行重新赋值。

    但是但是,程序世界真的混乱了吗?我们真的通过modifier修改了constatn的值了吗?修改const变量的数据真的是C++去const的目的吗?

    如果我们把结果打印出来: cout << "constant: "<< constant <<endl;
    cout << "const_p: "<< *const_p <<endl;
    cout << "modifier: "<< *modifier <<endl;
    /**
    constant: 21
    const_p: 7
    modifier: 7
    **/

    constant还是保留了它原来的值。

    可是它们的确指向了同一个地址呀:
    cout << "constant: "<< &constant <<endl;
    cout << "const_p: "<< const_p <<endl;
    cout << "modifier: "<< modifier <<endl;

    /**
    constant: 0x7fff5fbff72c
    const_p: 0x7fff5fbff72c
    modifier: 0x7fff5fbff72c
    **/

    这真是一件奇怪的事情,但是这是件好事:说明C++里是const,就是const,外界千变万变,我就不变。不然真的会乱套了,const也没有存在的意义了。

    IBM的C++指南称呼“*modifier = 7;”为“未定义行为(Undefined Behavior)”。所谓未定义,是说这个语句在标准C++中没有明确的规定,由编译器来决定如何处理。

    位运算的左移操作也可算一种未定义行为,因为我们不确定是逻辑左移,还是算数左移。

    再比如下边的语句:v[i] = i++; 也是一种未定义行为,因为我们不知道是先做自增,还是先用来找数组中的位置。

    对于未定义行为,我们所能做的所要做的就是避免出现这样的语句。对于const数据我们更要这样保证:绝对不对const数据进行重新赋值。

    如果我们不想修改const变量的值,那我们又为什么要去const呢?

    原因是,我们可能调用了一个参数不是const的函数,而我们要传进去的实际参数确实const的,但是我们知道这个函数是不会对参数做修改的。于是我们就需要使用const_cast去除const限定,以便函数能够接受这个实际参数。

    
    #include <iostream>
    using namespace std;
    
    void Printer (int* val,string seperator = "\n")
    {
    	cout << val<< seperator;
    }
    
    int main(void) 
    {	
    	const int consatant = 20;
    	//Printer(consatant);//Error: invalid conversion from 'int' to 'int*'
    	Printer(const_cast<int *>(&consatant));
    	
    	return 0;
    }

    出现这种情况的原因,可能是我们所调用的方法是别人写的。还有一种我能想到的原因,是出现在const对象想调用自身的非const方法的时候,因 为在类定义中,const也可以作为函数重载的一个标示符。有机会,我会专门回顾一下我所知道const的用法,C++的const真的有太多可以说的 了。

    IBM的C++指南中还提到了另一种可能需要去const的情况:

    #include <iostream>
    using namespace std;
    
    int main(void) {
    	int variable = 21;
    	int* const_p = &variable;
    	int* modifier = const_cast<int*>(const_p);
    	
    	*modifier = 7
    	cout << "variable:" << variable << endl;
    	
    	return 0;
    } 
    /**
    variable:7
    **/

    我们定义了一个非const的变量,但用带const限定的指针去指向它,在某一处我们突然又想修改了,可是我们手上只有指针,这时候我们可以去const来修改了。上边的代码结果也证实我们修改成功了。

    不过我觉得这并不是一个好的设计,还是应该遵从这样的原则:使用const_cast去除const限定的目的绝对不是为了修改它的内容,只是出于无奈。(如果真像我说是种无奈,似乎const_cast就不太有用到的时候了,但的确我也很少用到它)

  • 相关阅读:
    javascript设计模式——链式模式学习
    浏览器debug常用技巧
    前端到底要不要学后台
    坑爹的JS闭包,怎么去理解才是正确的
    如何更加简单的理解JS中的原型原型链概念
    前端那么多框架,我们到底学哪一个
    大前端之——数据交互
    随便写一点自己对前端的感受
    如何手动使用webpack搭建一个react项目
    浅谈 CSS 预处理器: 为什么要使用预处理器?
  • 原文地址:https://www.cnblogs.com/liyangguang1988/p/3061589.html
Copyright © 2020-2023  润新知