常见C++设计模式面试题和场景题

面试题补充


今天有同学问到豆芽一个问题:为什么微信不能像qq一样打开多个。豆芽突然想到看样子还得补充一些面试题。豆芽尽我所能帮助到大家。

对于C++软件求职要求确实是比较高的,《蒋豆芽的秋招打怪之旅》专刊包含近400道高频面试题,今天再补充一些设计模式面试题和场景题。


设计模式面试题


  1. 说说面对对象中的设计原则

    SRP(Single Responsibility Principle):单一职责原则,就是说一个类只提供一种功能和仅有一个引起它变化的因素。

    OCP(Open Close Principle):开放封闭原则,就是对一个类来说,对它的内部修改是封闭的,对它的扩展是开放的。

    DIP(Dependence Inversion Principle):依赖倒置原则,就是程序依赖于抽象,而不依赖于实现,它的主要目的是为了降低耦合性,它一般通过反射和配置文件来实现的。

    LSP(Liskov Substitution Principle):里氏替换原则,就是基类出现的地方,通过它的子类也完全可以实现这个功能

    ISP(Interface Segregation Principle):接口隔离原则,建立单一接口,不要建立庞大臃肿的接口,尽量细化接口,接口中的方法尽量少。也就是说,我们要为各个类建立专用的接口,而不要试图去建立一个很庞大的接口供所有依赖它的类去调用。

    CRP(Composite Reuse Principle):合成复用原则,多用组合设计类,少用继承。

  2. 单一职责原则和接口隔离原则的区别

    1. 单一职责原则注重的是职责;而接口隔离原则注重对接口依赖的隔离。
    2. 单一职责原则主要是约束类,其次才是接口和方法,它针对的是程序中的实现和细节; 而接口隔离原则主要约束接口,主要针对抽象,针对程序整体框架的构建。
  3. 说说你了解的设计模式

    1. 单例模式(要求学会手撕)

      单例模式只允许创建一个活动的对象(实例),提供了对唯一实例的受控访问。

      比如Windows的任务管理器,就是一个很典型的单例模式实现。

      单例实现原理是,将能够创建对象的函数都设置为private,通过静态成员返回一个实例。

      有两种方式,一个是懒汉式,一个是饿汉式。懒汉式需要考虑加锁。

      实现代码如下:

      #include <iostream>
      #include <pthread.h>
      using namespace std;
      
      class singleInstance{
      public:
      	static singleInstance* GetsingleInstance(){
      		if (instance == NULL){
      			pthread_mutex_lock(&mutex);//mlock.lock();
      			if (instance == NULL){
      				instance = new singleInstance();
      			}
      			pthread_mutex_unlock(&mutex);//mlock.unlock();
      		}
      		return instance;
      	};
      	~singleInstance(){};
      	static pthread_mutex_t mutex;//mutex mlock; 加锁互斥
      private:// 涉及创建对象的函数都设置为private
      	singleInstance(){};
      	singleInstance(const singleInstance& other){};
      	singleInstance& operator=(const singleInstance& other){ return *this; };
      	static singleInstance* instance;
      };
      
      //懒汉式,静态变量需要定义
      singleInstance* singleInstance::instance = nullptr;
      pthread_mutex_t singleInstance::mutex;
      
      int main(){
      	// 因为没有办法创建对象,就得采用静态成员函数的方法返回静态成员变量
      	singleInstance *s = singleInstance::GetsingleInstance();
      	//singleInstance *s1 = new singleInstance(); // 报错
          cout << "Hello World";
          delete s;  // 防止内存泄露
          return 0;
      }
      

      下面是饿汉式:

      #include <iostream>
      #include <pthread.h>
      using namespace std;
      
      class singleInstance{
      public:
      	static singleInstance* GetsingleInstance(){ // 饿汉式,直接创建一个对象,不需要加锁
      		static singleInstance instance;
      		return &instance;
      	};
      	~singleInstance(){};
      private:// 涉及创建对象的函数都设置为private
      	singleInstance(){};
      	singleInstance(const singleInstance& other){};
      	singleInstance& operator=(const singleInstance& other){ return *this; };
      };
      
      int main(){
      	// 因为没有办法创建对象,就得采用静态成员函数的方法返回
      	singleInstance *s = singleInstance::GetsingleInstance();
      	//singleInstance *s1 = new singleInstance(); // 报错
          cout << "Hello World";
          return 0;
      }
      
    2. 工厂模式(要求学会手撕)

      就是建立一个工厂类,对实现了同一接口的一些类进行实例的创建。简单工厂模式的实质是由一个工厂类根据传入的参数,动态决定应该创建哪一个产品类(这些产品类继承自一个父类或接口)的实例。

      #include <iostream>
      #include <pthread.h>
      using namespace std;
      
      //产品类(抽象类,不能实例化)
      class Product{
      public:
          Product(){};
          virtual void show()=0;  //纯虚函数
          virtual ~Product(){};
      };
      
      class productA:public Product{
      public:
          productA(){};
          void show(){ cout << "product A create!" << endl; };
          ~productA(){ cout << "product A delete!" << endl; };
      };
      
      class productB:public Product{
      public:
          productB(){};
          void show(){ cout << "product B create!" << endl; };
          ~productB(){ cout << "product B delete!" << endl; };
      };
      
      class simpleFactory{ // 工厂类
      
      public:
          simpleFactory(){};
          Product* product(const string str){
              if (str == "productA")
                  return (new productA());
              if (str == "productB")
                  return (new productB());
              return NULL;
          };
          ~simpleFactory(){};
      };
      
      int main(){
          simpleFactory obj; // 创建工厂
          Product* pro; // 创建产品
      
          pro = obj.product("productA");
          pro->show(); // product A create!
          delete pro;
      
          pro = obj.product("productB");
          pro->show(); // product B create!
          delete pro;
      
          return 0;
      }
      

      工厂模式为的就是代码解耦,如果我们不采用工厂模式,如果要创建产品A、B,我们通常做法是不是用switch...case语句?那麻烦了,代码耦合程度高,后期添加更多的产品进来,我们不是要添加更多的case吗?这样就太麻烦了,而且不符合设计模式中的开放封闭原则

      为了进一步解耦,在简单工厂的基础上发展出了抽象工厂模式,即连工厂都抽象出来,实现了进一步代码解耦。代码如下:

      #include <iostream>
      #include <pthread.h>
      using namespace std;
      
      //产品类(抽象类,不能实例化)
      class Product{
      public:
      	Product(){}
      	virtual void show()=0;  //纯虚函数
      	virtual ~Product(){}
      };
      
      //产品A
      class ProductA:public Product{
      public:
      	ProductA(){}
      	void show(){ cout<<"product A create!"<<endl; };
      	~ProductA(){ cout<<"product A delete!"<<endl; };
      };
      
      //产品B
      class ProductB:public Product{
      public:
      	ProductB(){}
      	void show(){ cout<<"product B create!"<<endl; };
      	~ProductB(){ cout<<"Product B delete!"<<endl; };
      };
      
      class Factory{//抽象类
      public:
      	virtual Product* CreateProduct()=0;
      	~Factory(){}
      };
      
      class FactorA:public Factory{//工厂类A,只生产A产品
      public:
      	Product* CreateProduct(){
      		return (new ProductA());
      	}
      };
      
      class FactorB:public Factory{//工厂类B,只生产B产品
      public:
      	Product* CreateProduct(){
      		return (new ProductB());
      	}
      };
      
      int main(){
      	
      	Product* _Product = nullptr;
      	auto MyFactoryA = new FactorA();
      	_Product = MyFactoryA->CreateProduct();// 调用产品A的工厂来生产A产品
      	_Product->show();
      	delete _Product;
      
      	auto MyFactoryB=new FactorB();
      	_Product=MyFactoryB->CreateProduct();// 调用产品B的工厂来生产B产品
      	_Product->show();
      	delete _Product;
      
      	getchar();
      	return 0;
      }
      
    3. 观察者模式

      观察者模式的作用是:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。

      比如拍卖的时候,拍卖师观察最高标价,然后通知给其他竞价者竞价。

      代码如下:

      #include <iostream>
      #include <list>
      using namespace std;
       
      class Observer{  // 观察者抽象
      public:
          virtual void Update(int) = 0;
      };
       
      class Subject{  // 被观察者抽象
      public:
          virtual void Attach(Observer *) = 0;
          virtual void Detach(Observer *) = 0;
          virtual void Notify() = 0;
      };
       
      class ConcreteObserver1:public Observer{  // 第一个观察者
      public:
          ConcreteObserver1(Subject *pSubject):m_pSubject(pSubject){}
          void Update(int value){
              cout << "ConcreteObserver1 get the update. New State:" << value << endl;
          }
      private:
          Subject *m_pSubject;
      };
       
      class ConcreteObserver2 : public Observer{  // 第二个观察者
      public:
          ConcreteObserver2(Subject *pSubject):m_pSubject(pSubject){}
          void Update(int value){
              cout << "ConcreteObserver2 get the update. New State:" << value << endl;
          }
      private:
          Subject *m_pSubject;
      };
       
      class ConcreteSubject:public Subject{  // 被观察者
      public:
          void Attach(Observer *pObserver);
          void Detach(Observer *pObserver);
          void Notify();
          void SetState(int state){
              m_iState = state;
          }
      private:
          std::list<Observer *> m_ObserverList;
          int m_iState;
      };
       
      void ConcreteSubject::Attach(Observer *pObserver){ // 添加观察者
          m_ObserverList.push_back(pObserver);
      }
       
      void ConcreteSubject::Detach(Observer *pObserver){ // 删除观察者
          m_ObserverList.remove(pObserver);
      }
       
      void ConcreteSubject::Notify(){  // 通知观察者
          std::list<Observer *>::iterator it = m_ObserverList.begin();
          while (it != m_ObserverList.e

剩余60%内容,订阅专栏后可继续查看/也可单篇购买

- 本专栏适合于C/C++已经入门的学生或人士,有一定的编程基础。 - 本专栏特点: 本专刊囊括了C语言、C++、操作系统、计算机网络、嵌入式、算法与数据结构、数据库等一系列知识点,总结出了高频面试考点(附有答案)共计309道,事半功倍,为大家春秋招助力。 - 本专栏内容分为七章:共计309道高频面试题(附有答案)

全部评论
我将所有题目都汇总到了一篇文章中,这样方便大家学习。https://blog.nowcoder.net/n/ee923114ff7b4df980e99747493ed054
点赞 回复 分享
发布于 2021-04-07 11:35
针对于非科班和基础不太好的同学,建议学习本人专刊文章《蒋豆芽的秋招打怪之旅》,该专刊文章对每一个知识点进行了详细解析。本专刊采用讲故事的形式为大家串联、呈现每一个知识点,做到有趣有干货、通俗易懂,春秋招本是痛苦难捱的过程,我希望通过自己讲故事能缓解大家的压力与焦虑,并将知识轻松学到手。适合C++软件开发和嵌入式软件开发的同学们。 网址: https://www.nowcoder.com/tutorial/10078/index
点赞 回复 分享
发布于 2021-04-07 11:36
单例模式中的懒汉模式,有两个小问题。(1)锁的问题,锁是一种竞争资源,既然是竞争那么大家都应该知晓它的存在,C语言一般喜欢用全局变量--pthread_mutex_t来处理,博主的锁是对象私有--以至于每个对象都有自己的锁,这违背了锁的基本概念。C++不太推荐全局变量,因为类中static也可以达到共享目的,所以应该把pthread_mutex_t定义成类中的static成员变量(2)单例模式不允许外部使用new来创建对象,这里使用的是指针,需要使用delete,不然内存泄漏,但是用的static修饰的啊?其实混淆了基本概念,指针指向一片内存,我们关心的是指向的内存,delete可以把指向的那片内存释放,至于指针本身--它在栈区、全局区os会自动清理
点赞 回复 分享
发布于 2021-05-16 16:15
简单工厂模式中有两个问题。(1)抽象基类中需要虚析构函数,否则多态代码无法正确析构对象,默认的析构函数不是虚的(2)按照工厂想法--工厂是为了创建实例,可能是不同实例,没必要用私有变量m保存它,将工厂函数设为static,每做一项product都需要用新的指针指向,不然前面就泄漏了
点赞 回复 分享
发布于 2021-05-16 19:42
单例模式-懒汉模式的那个锁有问题,是一个局部锁,不能起到作用
点赞 回复 分享
发布于 2021-07-25 23:22
请问工厂模式中的一些类,像Product、simpleFactory类中,没有定义析构函数是因为什么呢
点赞 回复 分享
发布于 2021-08-23 11:57
单例模式中的饿汉模式是不是有问题呀,博主是用的magic static实现的懒汉模式吧。还有这个是没法析构创建的实例的,建议用智能指针完善一下。
点赞 回复 分享
发布于 2022-04-25 15:03
Marmark
点赞 回复 分享
发布于 2023-09-01 07:50 安徽
请问观察者模式中,在被观察者的SetState()方法中调用Notify()方法是不是更符合观察者模式的特点?因为这样一更新就必须通知,而不是自己选择是否通知
点赞 回复 分享
发布于 2023-10-31 09:53 广东

相关推荐

2024-12-23 11:36
中南大学 Java
点赞 评论 收藏
分享
01-17 15:55
点赞 评论 收藏
分享
评论
8
12
分享

创作者周榜

更多
牛客网
牛客企业服务