当前位置: 首页 > news >正文

C++异常与智能指针

异常

关于异常/错误的处理,C语言时期的方法是通过错误码的形式来处理错误。错误码类似于一个map,C语言内核会为每一个错误码附带一个错误信息,发生异常后便会给出错误码,接着我们需要拿着错误码去找错误信息,比较麻烦。在C++中异常处理机制得到了改善,使用的是异常的抛出与捕获方法。

当程序出现问题的时候,我们通过抛出(throw)一个对象来触发一个异常,接着由catch判断类型是否符合。每一个try都至少会匹配一个catch,否则就会报错。异常会优先找离自己最近的那个类型相同的catch,当发生了throw,程序就不会执行throw以后的程序,则是直接去找catch了,类似于return,只不过return是返回到栈的上一层,catch会先在本函数里面找catch,如果没找到符合的,会沿着调用链往上寻找,当他开始往上寻找,沿着调用链创建的对象都将被销毁。如果一直到main函数都没找到匹配的catch子句,程序会调用标准库的terminate函数。

#include <iostream>
using namespace std;
float Divide(int a, int b)
{try {if (b == 0){string s("除0了!!!!");throw s;}else{return (double)a / (double)b;}}catch (int id){cout << "id" << endl;}}void Func()
{int a, b;cin >> a >> b;try {cout << Divide(a, b) << endl;}catch (int errid){cout << errid<< endl;}
}int main()
{while (1){try{Func();}catch (string& error){cout << error << endl;}}return 0;
}

一般来说,抛出对象和catch类型是完全一致的,但是这个类型也允许一些例外,例如允许从非常量转为常量,数组转换为指向数组元素类型的指针,函数被转换为指向函数的函数指针,允许派生类向基类的转换,最后一点非常实用,在实际的工作流中都是用这个方法进行设计了。

class Exception
{public :Exception(const string& errmsg, int id): _errmsg(errmsg), _id(id){}virtual string what() const{return _errmsg;} int getid() const{return _id;}
protected:string _errmsg;int _id;
};
class SqlException : public Exception
{public :SqlException(const string& errmsg, int id, const string& sql): Exception(errmsg, id), _sql(sql){}virtual string what() const{string str = "SqlException:";str += _errmsg;str += "->";str += _sql;return str;}
private:const string _sql;
};
class CacheException : public Exception
{ 
public :
CacheException(const string & errmsg, int id)
: Exception(errmsg, id)
{
}
virtual string what() const
{
string str = "CacheException:";
str += _errmsg;
return str;
}
};
class HttpException : public Exception
{public :HttpException(const string& errmsg, int id, const string& type): Exception(errmsg, id), _type(type){}virtual string what() const{string str = "HttpException:";str += _type;str += ":";str += _errmsg;return str;}
private:const string _type;
};
void SQLMgr()
{if (rand() % 7 == 0){throw SqlException("权限不⾜", 100, "select * from name = '张三'");} else{cout << "SQLMgr 调⽤成功" << endl;}
}
void CacheMgr()
{if (rand() % 5 == 0){throw CacheException("权限不⾜", 100);} else if (rand() % 6 == 0){throw CacheException("数据不存在", 101);} else{cout << "CacheMgr 调⽤成功" << endl;} SQLMgr();
} 
void HttpServer()
{if (rand() % 3 == 0){throw HttpException("请求资源不存在", 100, "get");} else if (rand() % 4 == 0){throw HttpException("权限不⾜", 101, "post");} else{cout << "HttpServer调⽤成功" << endl;} CacheMgr();
} int main()
{srand(time(0));while (1){this_thread::sleep_for(chrono::seconds(1));try{HttpServer();} catch(const Exception & e) // 这⾥捕获基类,基类对象和派⽣类对象都可以被捕获{cout << e.what() << endl;} catch(...){cout << "Unkown Exception" << endl;}} return 0;
}

异常抛出后,后面的代码就不会执行了,会逐步的把在局部域内的对象给销毁,如果前面申请了资源,但是释放在throw后面,就会造成了内存泄漏的问题。如果要用逻辑程序来解决这个问题会很麻烦,因此就要用到我们下一章会提到的RAII方式。catch(...)表示的是啥都捕获。throw表示的是catch里面捕获到啥抛出啥,去寻找下一个符合类型的catch.

double Divide(int a, int b)
{// 当b == 0时抛出异常if (b == 0){throw "Division by zero condition!";} return(double)a / (double)b;
} 
void Func()
{// 这⾥可以看到如果发⽣除0错误抛出异常,另外下⾯的array没有得到释放。// 所以这⾥捕获异常后并不处理异常,异常还是交给外层处理,这⾥捕获了再// 重新抛出去。int* array = new int[10];try{int len, time;cin >> len >> time;cout << Divide(len, time) << endl;} catch(...){// 捕获异常释放内存cout << "delete []" << array << endl;delete[] array;throw; // 异常重新抛出,捕获到什么抛出什么} cout << "delete []" << array << endl;delete[] array;
} 
int main()
{try{Func();} catch(const char* errmsg){cout << errmsg << endl;} catch(const exception & e){cout << e.what() << endl;} catch(...){cout << "Unkown Exception" << endl;} return 0;
}

对于用户和编译器来说,如果能提前知道某个程序能不能抛出异常,我们就能对应的设计程序来简化代码,提高效率,在C++11中加入noexcept,函数参数列表后面加上noexcept表示不会抛出异常,啥都不加的话表示可能抛异常。一个函数如果用noexcept修饰了,但是里面还是有throw相关内容,编译还是会通过,但是如果这个函数触发了异常,程序会调用terminate终止程序,而非常见的catch捕捉机制。noexcept(expression)还可以用来检测一个表达式是否会抛异常,可能返回false(0),不会返回true(1)。noexcept不会检查内部的具体函数的

/ C++98
// 这⾥表⽰这个函数只会抛出bad_alloc的异常
void* operator new (std::size_t size) throw (std::bad_alloc);
// 这⾥表⽰这个函数不会抛出异常
void* operator delete (std::size_t size, void* ptr) throw();
// C++11
size_type size() const noexcept;
iterator begin() noexcept;
const_iterator begin() const noexcept;
double Divide(int a, int b) noexcept
{
// 当b == 0时抛出异常
if (b == 0)
{
throw "Division by zero condition!";
} r
eturn (double)a / (double)b;
} i
nt main()
{
try
{
int len, time;
cin >> len >> time;
cout << Divide(len, time) << endl;
} c
atch (const char* errmsg)
{
cout << errmsg << endl;
} c
atch (...)
{
cout << "Unkown Exception" << endl;
} i
nt i = 0;
cout << noexcept(Divide(1,2)) << endl;
cout << noexcept(Divide(1,0)) << endl;
cout << noexcept(++i) << endl;
return 0;
}

智能指针

以前,当我们在new的时候,我们得匹配delete,new[]的时候,我们需要delete[]来进行匹配。但是此时如果抛异常了,或者提前return了,那么我们可能没有相应的可以释放内存的程序,就会导致内存泄漏。此时,我们使用智能指针,就能解决这个问题。

官方的智能指针包括auto_ptr,unique_ptr,shared_ptr,weak_ptr。其中auto_ptr完全不建议用,有指针悬空的问题。unique_ptr不允许拷贝,只允许移动。shared_ptr泛用性比较强,可以多个对象管理同一块空间。weak_ptr是为了解决shared_ptr可能的循环引用问题。

RAII与智能指针的设计思路

RAII是Resource Accqusition Initiallization的缩写,意为是资源获取初始化,本质上是将获取的动态资源利用对象生命周期来进行管理,这里的资源可以是内存,文件指针,网络连接等等,RAII获取资源后将其委托给一个对象,接着控制对资源的访问,资源在对象的声明周期时有效,对象析构的时候会释放资源,这样我们就能避免资源泄露。一般来说,智能指针还会重载 operator*/operator->/operator[] 等运算符。下面是关于shared_ptr的部分重新实现。

#include <functional>
namespace yu
{template <class T>class shared_ptr{public:shared_ptr(T* ptr): _ptr(ptr),_ptr_count(new int(1)){}template <class D>shared_ptr(T* ptr, D del): _ptr(ptr),_ptr_count(new int(1)),_del(del){}shared_ptr(const shared_ptr<T>& sp):_ptr(sp._ptr),_ptr_count(sp._ptr_count),_del(sp._del){(*_ptr_count)++;}void release(){if (--(*_ptr_count) == 0){delete _ptr;delete _ptr_count;}}~shared_ptr(){release();}shared_ptr& operator=(const shared_ptr& sp){if (_ptr != sp._ptr){release();_ptr == sp._ptr;_del = sp._del;_ptr_count = sp._ptr_count;(*_ptr_count)++;}return *this;}T& operator*(){return *_ptr;//返回解引用的内容}T* operator->(){return _ptr;//返回地址}int use_count() const{return *_ptr_count;}private:T* _ptr;int* _ptr_count;function<void(T*)> _del = [](T* ptr) {delete ptr;};//用包装器进行包装,void是返回类型,T*是参数};
}
int main()
{int* a = new(int);string* s = new(string);*a = 1;cout << *a << endl;yu::shared_ptr<int> sp1 = a;cout << *a << endl;//yu::shared_ptr<int> sp2 = a;报错!因为这样构造出来的对象未拥有同一个_ptr_count,虽然指向同一块空间,但是计数紊乱。后面析构的时候也会多次析构!智能通过sp1来构造yu::shared_ptr<int> sp2 = sp1;*sp2 = 4;cout << *sp2 << endl;return 0;
}

shared_ptr⼤多数情况下管理资源⾮常合适,⽀持RAII,也⽀持拷⻉。但是在循环引⽤的场景下会
导致资源没得到释放内存泄漏,所以我们要认识循环引⽤的场景和资源没释放的原因,并且学会使
⽤weak_ptr解决这种问题。 
• 如下图所述场景,n1和n2析构后,管理两个节点的引⽤计数减到1 
1. 右边的节点什么时候释放呢,左边节点中的_next管着呢,_next析构后,右边的节点就释放了。 
2. _next什么时候析构呢,_next是左边节点的的成员,左边节点释放,_next就析构了。 
比特就业课3. 左边节点什么时候释放呢,左边节点由右边节点中的_prev管着呢,_prev析构后,左边的节点就释
放了。
4. _prev什么时候析构呢,_prev是右边节点的成员,右边节点释放,_prev就析构了。 
• ⾄此逻辑上成功形成回旋镖似的循环引⽤,谁都不会释放就形成了循环引⽤,导致内存泄漏
• 把ListNode结构体中的_next和_prev改成weak_ptr,weak_ptr绑定到shared_ptr时不会增加它的
引⽤计数,_next和_prev不参与资源释放管理逻辑,就成功打破了循环引⽤,解决了这⾥的问题

weak_ptr不支持RAII,weak_ptr也没有重载operator*和operator->等,因为他不参与资源管理,那么如果他绑定的shared_ptr已经释放了资源,那么他去访问资源就是很危险的。weak_ptr⽀持expired检查指向的资源是否过期,use_count也可获取shared_ptr的引⽤计数,weak_ptr想访问资源时,可以调⽤lock返回⼀个管理资源的shared_ptr,如果资源已经被释放,返回的shared_ptr是⼀个空对象,如果资源没有释放,则通过返回的shared_ptr访问资源是安全的。 

http://www.dtcms.com/a/483641.html

相关文章:

  • 网站后台信息怎么更新忽悠别人做商城网站
  • NVMe高速传输之摆脱XDMA设计53: 选择测试环境软件平台
  • 品牌网站建设磐石网络优等网站如何不被百度搜到
  • 如何让Agent停止输出,使用strands agents sdk
  • Json数据字段类型兼容性处理
  • IvorySQL 亮相第 27 届中国国际软件博览会:开源创新,共筑软件新生态
  • 景安一个空间怎么做多个网站wampserver装wordpress
  • Electron技术深度解析:跨平台桌面开发的利器与挑战
  • FocusAny 发布v1.1.0 插件搜索过滤,FAD文件优化,插件显示MCP服务
  • AI硬件的“窄门”:未来智能创始人马啸的成功与启示
  • 潍坊信息网网站建设手机网页尺寸
  • 做羞羞的事的网站东莞最新招聘
  • 上海网站设计外包微信小程序登录入口官网
  • 活动 网站 源码网站开发 手机 电脑
  • 学而思编程网站重庆荣昌网站建设价格
  • 个人的网站备案多少钱福州鼓楼区建设局网站
  • python如何使用nacos
  • 网站建设建站南京哪里做网站
  • 有谁知道网站优化怎么做大型门户网站核心技术
  • 网站开发过程中的功能需求分析wordpress小插件下载
  • 网站运营推广该如何做上海做网站公司有哪些
  • 网站建设方案范文2000字扁平化网站配色
  • 【计算机视觉】Harris角点检测
  • 由于openssl升级导致重启php时提示libssl.so.1.0.0不存在
  • 用wordpress做网站青海省教育厅门户网站登录
  • 网站信息化建设建议和意见自媒体运营从入门到精通
  • CSS中常用的几种定位。
  • 郑州网站优化推广培训江苏网站建设效果
  • 潜江建设网站十堰秦楚网
  • 做视频网站用哪个软件好wordpress怎么添加关键词