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

品牌网址seo和sem哪个工资高

品牌网址,seo和sem哪个工资高,免费英文b2c网站建设,网络策划是什么意思目录 一、设计一个类,不能被拷贝二、设计一个类,只能在堆上创建对象三、设计一个类,只能从栈上创建对象四、设计一个类,不能被继承五、设计一个类,只能创建一个对象(单例模式)5.1 饿汉模式5.2 懒…

目录

  • 一、设计一个类,不能被拷贝
  • 二、设计一个类,只能在堆上创建对象
  • 三、设计一个类,只能从栈上创建对象
  • 四、设计一个类,不能被继承
  • 五、设计一个类,只能创建一个对象(单例模式)
    • 5.1 饿汉模式
    • 5.2 懒汉模式

一、设计一个类,不能被拷贝

//1、请设计一个类,不能被拷贝
// 
// 拷贝一个类,要么调用拷贝构造函数,要么调用赋值重载函数,所以要令一个类不能
// 被拷贝,只需要让该类不能调用拷贝构造和赋值重载函数就可以了。
// 
// 在C++98语法下,只需要把该类的拷贝构造函数和赋值重载函数声明为私有,
// 并且不定义即可
// 
//原因:拷贝构造函数和赋值重载函数只要我们声明了,编译器就不会再默认生成,但是
//我们又不定义它,所以这两个函数就不能被调用,如果定义了,反而在类内部会被调用
// 拷贝构造函数和赋值重载函数,不符合题意;并且我们声明为私有函数,别人想要
//从类外面自己定义也是做不到的,所以这样的类是不能被拷贝的class A
{
public:
private:A(const A& a);A& operator=(const A& a);
};// C++11语法下:
// 
// C++11扩展delete的用法,delete除了释放new申请的资源外,如果在默认成员函数后跟上 = delete,
// 表示让编译器删除掉该默认成员函数。
// 所以直接用关键字delete把拷贝构造函数和赋值重载函数删除掉即可
class A
{
private:A(const A& a) = delete;A& operator=(const A& a) = delete;
};

二、设计一个类,只能在堆上创建对象

//2、请设计一个类,只能在堆上创建对象
// 
//方法一:要想设计一个类只能在堆上创建对象,只需要将析构函数私有即可//原因:因为只要定义出对象的,对象是自定义类型,必然需要调用构造
// 函数和析构函数,而现在把析构函数私有,则定义的对象在销毁的时候
// 无法调用析构函数,就一定会报错
//但是如果是在堆上创建对象,返回值是一个对象的指针,指针是内置类型,
//不会调用构造函数和析构函数,所以即使析构函数是私有的,也可以通过
//new在堆上创建对象
class HeapOnly
{
public:void func(){cout << "func" << endl;}void Destroy(){//这里delete掉this指针等于是释放了调用//该函数的类的指针,这种写法是正确的delete this;cout << "~HeapOnly()" << endl;}private://析构函数私有~HeapOnly(){}private:int _b;
};//方法二:把构造函数私有,然后提供一个CreateObj的函数,
// 这个函数内部用new创建一个对象,然后返回对象的指针即可,
// 但是要注意把拷贝构造和赋值重载函数delete掉,防止别人
// 在栈上构造对象
//
class HeapOnly
{
public:static HeapOnly* CreateObj(){HeapOnly* ptr = new HeapOnly;return ptr;}
private:HeapOnly(){}HeapOnly(const HeapOnly& ho) = delete;HeapOnly& operator=(const HeapOnly& ho) = delete;
};int main()
{//static B b;HeapOnly* pb = new HeapOnly;pb->func();//HeapOnly b(*pb);pb->Destroy();return 0;
}//int main()
//{
//	HeapOnly* ho = HeapOnly::CreateObj();
//
//	return 0;
//}

三、设计一个类,只能从栈上创建对象

//3、请设计一个类,只能在栈上创建对象
//不能完全设计出只在栈上创建对象的类;
//沿用2的设计思路,把构造函数私有,然后提供一个CreateObj的函数,
//该函数返回一个在栈上创建的对象。
//因为这个返回的是临时对象,所以不能把拷贝构造函数和赋值函数delete掉,
//因为传值返回对象需要被拷贝。
class StackOnly
{
public:static StackOnly CreateObj(){return StackOnly();}void func(){cout << "StackOnly()" << endl;}private:StackOnly(){}// 对一个类实现专属operator new,这句代码的意思是把用new//创建该类对象的方法delete掉,外部不能再利用new来创建对象了void* operator new(size_t size) = delete;};int main()
{StackOnly so = StackOnly::CreateObj();cout << &so << endl;int a = 0;cout << &a << endl;so.func();//因为用户可能用new调用拷贝构造函数在堆上创建对象,所以需要// 把类的专属的operator new给delete掉,禁止用new在堆上创建对象//StackOnly* pso = new StackOnly(so);//这个类唯一不能禁止的就是在静态区创建对象,因为要在栈上创建对象就//一定要有拷贝构造函数,有拷贝构造函数就可以在静态区创建静态对象static StackOnly so1= StackOnly::CreateObj();static int b = 0;cout << &so1 << endl;cout << &b << endl;return 0;
}

四、设计一个类,不能被继承

//4. 请设计一个类,不能被继承
// 
//C++98语法
//构造函数私有,然后提供一个CreateObj函数创建对象
// 原因:构造函数私有,也就意味着构造函数不能被显式地调用,
// 因为在继承体系中,子类成员中父类的成员必须调用父类的构造函数
// 初始化父类那一部分成员的,所以如果把父类的构造函数私有,子类
// 就没有办法调用父类的构造函数初始化父类那一部分成员,所以就继承不了
//
class Final
{
public:
private:Final(int f):_f(f){}int _f = 0;
};class A:public Final
{
public://A的构造函数无法调用Final的构造函数,所以Final类不能被继承A():Final(2),_a(1){}private:int _a;
};
//C++11语法:
//C++11提供了一个关键字final,即直接在类后面加上final表示该类不能被继承
class Final final
{
public:Final(int f):_f(f){}private:int _f = 10;
};这里就会报错了,因为Final是不可被继承的
//class F :public Final
//{
//
//};

五、设计一个类,只能创建一个对象(单例模式)

设计模式:
设计模式(Design Pattern)是一套被反复使用、多数人知晓的、经过分类的、代码设计经验的总结。
使用设计模式的目的:为了代码可重用性、让代码更容易被他人理解、保证代码可靠性。 设计模式使代码编写真正工程化;设计模式是软件工程的基石脉络,如同大楼的结构一样。

单例模式:
一个类只能创建一个对象,即单例模式,该模式可以保证系统中该类只有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。比如在某个服务器程序中,该服务器的配置
信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息,这种方式简化了在复杂环境下的配置管理。

5.1 饿汉模式

//1、饿汉模式:一开始(main函数之前)就创建单例对象
//优点:简单
//缺点:
// (1) 如果单例对象要初始化的内容很多,启动速度慢。
// (2) 如果A单例对象的创建依赖B单例对象,要求B对象先创建好,但是我们无法保证先让B单例对象创建好。
// (3) 如果单例对象很大,占用资源很多,但是单例对象创建出来之后不是立刻使用,
//     会占用着大量的内存,导致其它需要内存的地方获取不到内存。
namespace hungry
{class Singalton{public:static Singalton& GetInstance(){return _Inst;}private://构造函数私有,防止别人随意创建对象Singalton(){}//把拷贝构造函数和赋值重载函数删除掉,防止拷贝Singalton(const Singalton& sg) = delete;Singalton& operator=(const Singalton& sg) = delete;//因为全局只有唯一的一个对象,如何保证我们每次获取到的都是同一个对象呢?// 因为静态的全局变量在main函数之前就会定义好的,所以静态的全局变量是全局唯一的。// 所以这里一般都是在类里面声明一个静态的Singalton对象(类外定义),// 在Singalton类里面可以声明Singalton的静态对象吗?// 声明静态对象static Singalton是可以的,// 但是声明普通对象Singalton就不行,为什么呢?// 因为声明普通对象的话就会出现无限套娃的情况了,对象里面又会套一个对象;但// 是声明静态的Singalton对象为什么就可以了呢?// 因为静态的Singalton对象本身并不存在于Singalton类的空间里面,而是存在于// 静态区中,属于所有对象共有的,所以不存在套娃的情况的,所以可以声明静态的Singalton对象的。// // 同时要注意,这里只是声明,普通静态对象必须在类外面定义,但是const static对象比较特殊,// 可以在类内定义这样每一次调用GetInstance的时候就返回这个静态的对象就可以保证每次返回的都是// 同一个_Inst了static Singalton _Inst;// 在C++中,允许在类内部声明一个本类的静态对象的原因是为了方便和灵活性,//当声明一个类的静态成员时,编译器只需要知道该成员的类型和名称,而不需要//知道成员的具体定义和大小。这样做的好处是可以避免一些循环依赖的问题。如// 果不允许在类内部声明本类的静态对象,那么在类定义之前就无法实例化包含静// 态成员的对象,因为在类定义之前,编译器还不知道该类的完整定义。//通过在类内部声明本类的静态对象,可以为程序提供更高的灵活性。在类定义之后// 的任何位置,都可以在需要的时候进行定义和初始化这个静态对象。这样,程序员// 可以根据需要更具灵活性地控制对象的创建和使用。//需要注意的是,在定义之前使用这个静态对象可能会导致未定义的行为,因此在类// 定义之后的某个地方,一定要进行静态对象的定义和初始化,以确保它的正确使用。};//定义,在main函数之前就已经创建好了单例对象Singalton Singalton::_Inst;
}int main()
{//每次调用GetInstance获取到的对象都是同一个cout << &hungry::Singalton::GetInstance() << endl;cout << &hungry::Singalton::GetInstance() << endl;cout << &hungry::Singalton::GetInstance() << endl;return 0;
}

如果这个单例对象在多线程高并发环境下频繁使用,性能要求较高,那么显然使用饿汉模式来避免资源(例如锁)竞争,提高响应速度更好。

5.2 懒汉模式

如果单例对象构造十分耗时或者占用很多资源,比如加载插件, 初始化网络连接,读取文件等等操作时,并且有可能该对象程序运行时不会用到,那么也要在程序一开始就进行初始化,就会导致程序启动时非常的缓慢。 所以这种情况使用懒汉模式(延迟加载)更好。

//2、懒汉模式
//比较懒,到有人定义对象的时候才创建对象
//(1) 优点:第一次使用实例对象时,创建对象。进程启动无负载。多个单例实例启动顺序自由控制。
//(2) 缺点:复杂。
namespace lazy
{class Singalton{public:static Singalton* GetInstance(){//这里要使用双判断加锁的方式处理,这样才能很好地提高效率//这个if是判断第一次调用实例对象时需要先创建对象if (_pInst == nullptr){//加锁避免线程安全的问题,两个线程同时进入了这里,需要先竞争锁_pmtx->lock();//这个if是判断如果两个进程同时来到了这里,竞争到锁的进程//先来到这个判断条件,如果_pInst还是nullptr,说明这是第一次调用实例对象//此时创建一个Singalton,后竞争到锁的进程在再一次if判断时_pInst就不再是nullptr//了,此时说明单例对象已经存在了,就不会再创建了if (_pInst == nullptr){_pInst = new Singalton;}_pmtx->unlock();}return _pInst;}//一般情况下,单例对象不需要释放,因为程序正常结束就释放了,// 并且单例对象一般也不大,所以可以把析构函数设置成私有//// 但是有些特殊场景:// 1、中途需要显式释放;//2、程序结束时需要做一些持久化(把数据写入到文件中);// 所以需要提供一个显式调用的DelInstance函数(里面封装析构函数)static void DelInstance(){cout << "DelInstance()" << endl;if (_pInst != nullptr){//自定义对象,调用_pInst对象的析构函数delete _pInst;delete _pmtx;}}void Add(const pair<string, string>& val){_um.insert(val);}private://构造函数Singalton(){}//析构函数~Singalton(){//显式调用析构函数书写日志或者持久化(数据写入文件)cout << "~Singalton()" << endl;FILE* fp = fopen("test.txt", "w");for (const auto& e : _um){fputs(e.first.c_str(), fp);fputs(":", fp);fputs(e.second.c_str(), fp);fputs("\n", fp);}}//拷贝构造函数,单例模式防拷贝Singalton(const Singalton& sg) = delete;//赋值重载函数Singalton& operator=(const Singalton& sg) = delete;//相当于一个垃圾回收类class GC{public://用Gc的析构函数管理单例类的析构函数,因为在Singalton中声明了一个Gc的静态的//成员变量,所以在进程结束的时候会调用析构函数,而Gc析构函数又管理着Singalton的// 析构函数DelInstance,所以无论如何进程结束的时候都会自动调用Singalton的析构// 函数的,所以就不存在内存泄漏的隐患了//~GC(){DelInstance();}};private://如何保证每次调用GetInstance函数的时候获取到的_Inst都是同一个呢?因为这里是运行时//才创建对象的,所以不能用静态的对象,而这里要获取到同一个对象,所以只能在堆上开辟;//但是这里为什么是用静态的呢?因为GetInstance是静态的,没有this指针,但是GetInstance//函数中需要用到_Inst,所以需要这个_Inst也要设置为静态的,这样GetInstance函数才能访问_Inst//同理_mtx也要是静态的static Singalton* _pInst;static mutex* _pmtx;unordered_map<string, string> _um;//在单例类中声明一个Gc类型的静态的成员变量,在类外定义;该对象在进入main函数前就已经创建好了,//到进程结束时才调用析构函数销毁,说明这个变量是整个进程都有效的static GC gc;};//必须在类外定义类内的静态成员变量Singalton* Singalton::_pInst = nullptr;mutex* Singalton::_pmtx = new mutex;Singalton::GC Singalton::gc;
}//GC也可以这样写,这样写更容易理解,用GC创建一个全局对象,进程结束时GC调用析构函数,
// 进而调用lazy::Singalton::DelInstance()释放单例对象,但是遇到多个单例类就把它们
// 全部放到~GC函数中即可,上面那种写法就是把GC定义到单例类内部,每一个GC对象管理一个单例类
//class GC
//{
//public:
//	~GC()
//	{
//		lazy::Singalton::DelInstance();
//	}
//};
//GC gc;int main()
{//lazy::Singalton* p1 = lazy::Singalton::GetInstance();//lazy::Singalton* p2 = lazy::Singalton::GetInstance();//lazy::Singalton* p3 = lazy::Singalton::GetInstance();//cout << p1 << endl;//cout << p2 << endl;//cout << p3 << endl;lazy::Singalton* p1 = lazy::Singalton::GetInstance();p1->Add(make_pair("string", "字符串"));p1->Add(make_pair("left", "左边"));p1->Add(make_pair("right", "右边"));return 0;
}

以上就是常见的特殊类的设计,你学会了吗?今天的分享就到这里啦,如果你感觉到有所收获,那么就点点小心心点点关注呗,后期还会持续更新C++的相关知识哦,我们下期见!!!

http://www.dinnco.com/news/24391.html

相关文章:

  • 自己做网站怎么能被访问网站怎么制作教程
  • 公司微网站怎么建设推广一次多少钱
  • 北海网站开发页面优化的方法有哪些
  • 什么网站从做系统电商推广和网络推广的策略
  • 保健品网站制作百度一下官网首页百度一下
  • 扶风网站开发长沙sem培训
  • 合肥网站建设方案朝阳区seo搜索引擎优化怎么样
  • 济南网站建设 力推聚搜网络全网营销推广方式
  • 郑州哪有做网站的公司南宁做网站公司
  • 阿里去可以做几个网站企业网站推广渠道有哪些
  • 百度景安空间网站东莞网络公司排行榜
  • 网站美工做确认取消对话框西安seo经理
  • 做外贸首先要做网站百度推广一般要多少钱
  • kangle搭建wordpress深圳seo优化方案
  • 免费电子商务网站建设搜索引擎推广试题
  • wordpress附加字段镇江seo
  • 南京网站制作希丁哥广东seo教程
  • 专业app网站建设哪家好东莞seo计费管理
  • 南京专业网站制作多少钱seo职位
  • 网站开发学什么编程语言为什么打开网址都是站长工具
  • 竹妃怎么在公众号里做网站网络营销推广专家
  • 南阳市宛城区疫情最新消息郑州seo课程
  • 哈尔滨网站开发公司电话培训计划模板
  • 类似优酷的网站开发企业网站建设目标
  • 音乐网站开发与设计小学生班级优化大师
  • 网站建设属于什么发票seo中国是什么
  • 杭州做网站套餐2023全民核酸又开始了
  • 做相册集什么网站青岛招聘seo
  • 新疆生产建设兵团第七师门户网站白山seo
  • 网站建设管理与维护网站关键词排名软件推荐