欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 新闻 > 会展 > C++面试速通宝典——26

C++面试速通宝典——26

2025/1/7 9:54:37 来源:https://blog.csdn.net/qq_43504141/article/details/142929094  浏览:    关键词:C++面试速通宝典——26

478. 你了解的网络攻击方式有哪些?SYN攻击的防范方法?

网络攻击方式包括:

  1. Dos/DDos攻击:通过大量请求耗尽目标资源。
  2. Main-in-the-Middle攻击:攻击者拦截和篡改双方通信。
  3. Phishing:通过假冒网站或通信骗取用户信息。
  4. SQL注入:在数据库查询中注入恶意SQL代码。
  5. 跨站脚本(XSS)攻击:在网页中注入恶意脚本。
  6. Malware:使用恶意软件进行攻击。

SYN攻击(一种Dos攻击)防范方法包括:

  1. SYN Cookies:仅在确认客户端响应后分配资源。
  2. 包过滤:过滤掉异常的或非法的SYN包。
  3. 连接限制:限制打开SYN请求的速率或数量。
  4. 超时设置:降低半开连接的超时时间。

479. epoll、poll、select的区别?epoll为什么高效?

  1. select:它使用位掩码来表示监视的描述符集合,但最大能处理的描述符数目有限制(通常是1024)。
  2. poll:它通过一个链表来存储网络描述符,解决了最大描述符限制问题,但每次调用poll都需要遍历整个链表,效率较低。
  3. epoll:可以理解为变种的poll,它与poll类似,都是使用链表来存储网络描述符,但每当有事件触发时,它只需要获取已经就绪的事件就可以返回了,因此处理的事件少则效率越高。

480. epoll为什么高效?

  1. 回调机制:epoll使用回调的方式,只关注活动的socket,不必每次都轮询所有的被监听的socket。
  2. scalability:epoll具有较好的扩展性,可以处理的socket描述符数量比select和poll都要大。
  3. 减少用户和内核空间之间复制数据的次数

481. MySQL支持的隔离级别是什么?最常用什么?

四种事务隔离级别:

  1. READ UNCOMMITTED : (读未提交)。
  2. READ COMMITTED : (读已提交)。
  3. REPEATABLE READ : (可重复读)。
  4. SERIALIZABLE : (串行化)。

最常用的隔离级别为 REPEATABLE READ(可重复读)。因为他平衡了性能与一致性,同时也是MySQL的默认隔离级别。

482. MySQL索引是主要通过什么数据结构实现的?为什么用B+树?

MySQL索引主要是通过B+树数据结构实现的。

原因:

  1. 高效的读写:对于数据库读写操作频繁的场景,B+树的平衡性能较好,读写性能稳定。
  2. 范围查询优势:B+树支持范围查询,hash表只支持精确查询,而二叉树效率低下。
  3. 磁盘读写优化:B+树的节点大小通常和磁盘扇区大小相同,这样可以最大化磁盘I/O效率。
  4. 减少磁盘I/O次数:B+树的分支因子大,树的层级较低,可以减少在查询过程中磁盘I/O的次数。

483. 虚函数实现机制

‌‌‌‌  C++中的虚函数通过虚函数表实现。

‌‌‌‌  每一个使用虚函数的类都有一个对应的虚函数表,其中存储了指向类的虚函数的指针。每一个对象实例包含一个指向其类的虚函数表的指针(vptr),当调用一个对象的虚函数时,会通过这个指针在虚函数表中查找对应的函数实现进行调用,从而实现多态性。这种机制允许在运行时根据对象的实际类型来动态绑定方法,而不是在编译时

484. const和define的区别

  1. const定义的常量有数据类型,而define是预处理指令,替换文本,没有数据类型。
  2. const有作用域限制,define一直到文件结束或被#undef 指令取消。
  3. const常量通常存储在程序的数据段中,define是简单替换,不分配存储空间。
  4. const常量在调试中可见,因为它们是符号化的,而define替换后的代码通常难以调试。
  5. const是在编译阶段使用,define发生在预处理阶段,编译器还未参与。

485. struct和class的区别

struct和class的主要区别在于默认的访问权限和继承类型:

  1. 默认访问权限:struct中的成员默认是公开类型,而class中的成员默认是私有的。
  2. 默认继承类型:使用struct继承默认是公开继承,而class继承默认是私有继承。

486. 4层负载和7层负载均衡的区别

‌‌‌‌  4层负载均衡工作在传输层,主要关注IP地址和端口,使用TCP/UDP协议进行流量分发。

‌‌‌‌  7层负载均衡工作在应用层,可以处理更高级的协议如HTTP/HTTPS,并且可以基于URL、cookies等因素来进行智能分发。

总的来说,4层负载均衡更关注网络层面,7层负载均衡则提供更细粒度的流量管理能力。

487. 你了解哪些负载均衡方法

负载均衡算法主要有以下几种:

  1. 轮询:请求按照顺序轮流分配到不同的服务器上。
  2. 加权轮询:类似轮询,但服务器有不同的权重。
  3. 最少连接:请求被发送到连接数最少的服务器。
  4. 加权最少连接:考虑服务器权重和连接数,选择最优服务器。
  5. 源地址哈希:根据请求源地址的哈希结果分配请求。
  6. URL哈希:根据请求的URL哈希结果来分配请求。
  7. 最少响应时间:选择响应时间最短的服务器。
  8. 最少带宽:选择当前带宽消耗最少的服务器。

488. malloc的底层原理了解么?涉及到哪些系统调用?

malloc是标准C库中用于动态内存分配的函数。其底层原理涉及到以下几个系统调用:

  1. brk:改变数据段的末尾地址,用于增加或减少数据段的大小。
  2. sbrk:增加进程的程序断点,实际上是对brk的封装。
  3. mmap:创建一个新的映射区域,用于分配大块内存。

‌‌‌‌  malloc通常是通过上述系统调用来向操作系统请求内存,上层的内存分配器使用这些系统调用来管理内存的分配和释放,以及维护内存的数据结构,如空闲列表或者内存池等,来优化内存的使用和减少内存碎片。

489. mmap的原理

mmap是一种将文件或者设备的内容映射到进程的地址空间的机制。其基本原理如下:

  1. 地址空间映射:mmap通过创建一个文件或设备内容与进程虚拟地址空间之间的对应关系,允许进程直接访问这片内存空间。这意味着对这部分内存的读写直接反应到了被映射文件或设备上,无需通过read或write系统调用。
  2. 延迟加载:mmap采用需求分页的方式,实际内存页只有在访问时才被映射到物理内存中,这代表mmap可以有效管理和使用内存,减少不必要的物理内存占用。
  3. 共享与私有:mmap支持共享内存映射和私有映射。共享映射允许不同进程共享同一片内存区域,对这片区域的修改会反映到所有映射了该区域的进程中。私有映射则为每个进程创建映射的副本,进程间的修改互不影响。
  4. 页表:操作系统利用页表来管理这种映射关系,确保进程可以通过虚拟地址访问到正确的物理内存或文件内容。

490. string的底层实现、vector的底层实现

string的底层实现:

  1. 动态数组:string在底层实际上是一个动态数组,用于存储字符序列。
  2. 自动扩容:当字符串内容超过当前分配的内存时,string会自动进行扩容(通常是倍增策略)并拷贝原有数据到新的内存区域。
  3. 短字符串优化(SSO):对于短字符串,为了减少动态内存分配的开销,string会在对象内预留一小段空间,直接在其中存储字符串数组和结尾的空字符’\0’,避免了堆分配。
  4. 复制写时优化(COW,现代C++标准中已废除):在一些旧的实现中,多个string对象可以共享同一个字符数组,只有在修改时才会进行真正的拷贝,以减少内存使用和提升效率。

vector的底层实现:

  1. 动态数组:vector内部同样使用动态数组来存储数据,它支持随机访问。
  2. 自动扩容:当插入的元素超过vector当前的容量时,它会自动重新分配更大的内存空间,并将原有元素复制或移动到新的地址。
  3. 空间预留:vector提供reserve方法,允许手动指定容量,减少不必要的重新分配和数据拷贝。

491. 结构体内存对齐相关

结构体内存对齐是为了提高CPU访问效率而采取的方式

  1. 对齐规则:结构体的每个成员在内存中的起始地址应该是其类型大小的整数倍(例如,int通常是4字节,他的地址应该是4的整数倍)。
  2. 结构体总大小:结构体的总大小为其最大成员类型大小的整数倍。如果不满足,会在结构体末尾进行填充。
  3. 按序排列:结构体成员变量之间可能存在填充(padding),以保证后续成员的地址对齐。
  4. pragma pack(n):可以通过#pragma pack(n)来指定结构体以n字节对齐。n一般为2的次幂。

492. size of 一个 string 变量的结果

‌‌‌‌  sizeof 一个 string 变量通常返回的是string类型固定的大小,而不是字符串内容的长度。这个大小包括string对象中管理字符串所需的所有成员的大小,如指针、长度和容量等,具体的大小取决于实现(例如具体的编译器和其版本)。在某些实现中可能会有短字符优化(SSO),在这种情况下,小的字符串可以直接存储在这个固定大小的内存之中,而无需动态分配。

493. std::move应用场景,和std::forward区别

std::move应用场景:

  1. 主要用于将对象标记为可移动,使之可以被移动构造或移动赋值函数使用。
  2. **应用于转移资源的所有权,**避免不必要的复制,例如移动大型容器或独占所有权的对象(如std::unique_ptr)。

std:move和std::forward的区别:

  1. std::move无条件地将一个对象转换为右值引用,从而允许资源的移动操作
  2. std::forward用于完美转发,在函数模板中,他可以保持参数的值属性(即左值保持左值,右值保持右值),通常与右值引用模板参数结合使用

494. 移动构造函数和复制构造函数的区别

复制构造函数

  1. 创建一个对象的副本
  2. 完成成员逐个复制(深复制)
  3. 原始对象和新对象都拥有自己的独立资源

移动构造函数

  1. 通过接管另一个对象的资源来创建对象(资源转移)
  2. 执行“浅复制”,只是将资源的所有权从源对象转移。
  3. 源对象不再拥有这些资源或处于有效但未定义的状态。

495. push_back和emplace_back对比

push_back:

  1. 接受一个元素作为参数,并将其复制(或移动)到容器末尾。

emplace_back:

  1. 使用所提供的参数直接在容器末尾构造元素
  2. 避免了额外的复制或移动操作

496. push_back可以传入std::move的变量嘛

‌‌‌‌  push_bakc可以接受通过std::move传递的变量,这样可以将变量的资源移动到容器中,而不是复制

497. push_back一个std::move的变量 与 直接emplace_back 性能对比

‌‌‌‌  push_back一个std::move的变量和直接使用emplace_back的性能是相似的,因为他们都是在进行移动操作,而非复制操作。

498. shared_ptr以引用形式传入参数会有什么问题

将shared_ptr以引用形式传递不会增加对象的引用计数,这可能会导致以下问题:

  1. 若函数内部存储或复制了这个引用,而在函数外部原shared_ptr已经被销毁,那么这个存储的引用将成为悬空引用,导致未定义的行为。
  2. 函数调用者可能会期望传递的shared_ptr对象会因函数调用而共享其所有权,但实际上却没有发生。

499. C++的四种类型转换

C++提供了四种类型转换操作符:

  1. static_cast:用于非多态类型的转换,安全性较高的类型转换。
  2. dynamic_cast:用于多态类型的转换,主要用于基类和派生类之间的安全转换。
  3. const_cost:用于去除或添加const或volatile属性。
  4. reinterpret_cast:用于进行低层次的重新解释类型转换,可能是不安全的。

500. explicit关键字的作用,使用场景

explicit关键字用来阻止C++编译器执行隐式类型转换,要求类型转换必须明确无误。使用explicit的典型场景包括:

  1. 防止构造函数的隐式调用,确保只能显式地进行对象初始化。
  2. 防止具有单个参数的构造函数或转换操作符被编译器自动用作隐式类型转换

501. 协程和线程的区别

  1. 线程由操作系统内核管理,协程一般由用户空间的库管理
  2. 线程切换包括内核空间和用户空间的上下文切换,开销较大;协程切换仅包含用户空间的上下文切换,开销较小。
  3. 线程的调度由操作系统内核控制;协程的调度由程序员或库控制,拥有更细粒度的控制能力。
  4. 大量线程会增加系统负担,而大量协程对系统负担较小,可以实现更高的并发性能

502. 共享内存如何使用,原理

共享内存的使用通常包括以下步骤:

  1. 创建共享内存段:一个进程创建一个特定大小的共享内存段。
  2. 将共享内存段附加到进程的地址空间:需要通信的进程将这个共享内存段映射到他们的地址空间。
  3. 使用共享内存:进程可以通过指针和内存操作读写这个内存段。
  4. 分离共享内存段:进程用完后,将共享内存从自己的地址空间分离。
  5. 删除共享内存段:所有进程都不再需要时,最后一个使用的进程应删除共享内存段。

‌‌‌‌  共享内存的原理是提供一块能被多个进程访问的内存区域。操作系统确保这块内存区域在多个进程间是可见的,从而实现非常快速的数据交换。因为它避免了数据的复制操作,直接在物理内存上进行读写。需要注意的是,由于多个进程可以同时访问,通常需要用同步机制(如信号量)来避免竞争条件。

503. 虚拟内存和物理内存在什么时候进行转换?

‌‌‌‌  虚拟内存和物理内存进行转换的时候称为“页映射”,这个过程由操作系统的内存管理单元(MMU)控制。当一个程序试图访问虚拟内存系统时,如果这个地址已经映射到物理内存,MMU将直接转换访问;如果没有映射,操作系统将会触发一个缺页中断,然后加载相应的数据到物理内存,并更新页表以反映新的映射关系,之后访问可以继续进行。这个转换过程对程序是透明的。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com