unity客户端面试高频2(自用)

标题是我

1.构造函数为什么不能为虚函数?析构函数为什么要虚函数?

构造函数不能定义为虚函数的原因是因为虚函数是用于实现动态多态性的机制,而构造函数的调用是在对象创建的过程中完成的。在对象创建时,其类型是已知的,不需要通过动态绑定来确定调用哪个构造函数。因此,构造函数不需要被定义为虚函数。

析构函数需要定义为虚函数的主要原因是为了在使用基类指针指向派生类对象并通过该指针删除对象时,可以正确地调用派生类对象的析构函数,以防止内存泄漏。如果不将析构函数定义为虚函数,当使用基类指针指向派生类对象并通过该指针删除对象时,只会调用基类的析构函数,而不会调用派生类的析构函数。这会导致派生类对象的资源无法正确释放,造成内存泄漏。

2.C++智能指针

图片来源程序员陈子青

具体的底层实现

计数器类

cpp 复制代码
template <typename T>
class RefCount {
public:
    T* ptr;  // 指向实际管理的对象
    size_t count;  // 引用计数
    RefCount(T* p) : ptr(p), count(1) {}
    ~RefCount() {
        delete ptr;  // 当引用计数为0时,释放管理的对象
    }
};

share_ptr类

cpp 复制代码
template <typename T>
class MySharedPtr {
private:
    RefCount<T>* ref;  // 指向引用计数管理对象

public:
    // 构造函数,初始化指向对象并设置引用计数为1
    MySharedPtr(T* p = nullptr) {
        if (p) {
            ref = new RefCount<T>(p);
        } else {
            ref = nullptr;
        }
    }

    // 拷贝构造函数
    MySharedPtr(const MySharedPtr& other) {
        ref = other.ref;
        if (ref) {
            ++(ref->count);  // 增加引用计数
        }
    }

    // 移动构造函数
    MySharedPtr(MySharedPtr&& other) noexcept {
        ref = other.ref;
        other.ref = nullptr;  // 转移所有权后,原对象的引用计数指针置空
    }

    // 赋值运算符重载
    MySharedPtr& operator=(const MySharedPtr& other) {
        if (this != &other) {
            release();  // 先释放当前对象的资源(若有)
            ref = other.ref;
            if (ref) {
                ++(ref->count);  // 增加引用计数
            }
        }
        return *this;
    }

    // 移动赋值运算符重载
    MySharedPtr& operator=(MySharedPtr&& other) noexcept {
        if (this != &other) {
            release();
            ref = other.ref;
            other.ref = nullptr;
        }
        return *this;
    }

    // 获取管理的对象指针
    T* get() const {
        return ref? ref->ptr : nullptr;
    }

    // 解引用操作
    T& operator*() const {
        return *(ref->ptr);
    }

    // 箭头操作符,用于访问对象成员
    T* operator->() const {
        return ref->ptr;
    }

    // 获取引用计数
    size_t use_count() const {
        return ref? ref->count : 0;
    }

    // 析构函数
    ~MySharedPtr() {
        release();
    }

private:
    // 释放资源的辅助函数
    void release() {
        if (ref) {
            --(ref->count);
            if (ref->count == 0) {
                delete ref;  // 引用计数为0时,删除引用计数管理对象
            }
        }
    }
};

3.左值和右值

  • 左值(lvalue):左值指的是可以出现在赋值运算符左边的表达式。它代表一个具名的、有确定内存地址的对象,其生命周期相对较长,在表达式结束后依然存在。

  • 右值(rvalue):右值指的是只能出现在赋值运算符右边的表达式。它代表临时对象、字面量或即将销毁的对象,其生命周期较短,在表达式结束后就会被销毁。

    右值引用会给右值一个临时的地址,并且延长它的生命周期。

左值作为参数的函数只可以接受左值,强调不要复制。void print(int& a);

右值作为参数的函数传入只能接受右值,可以修改右值。void print(int&& a);

左值作为参数传入很常见,避免复制影响空间内存和性能。

那为什么要用右值引用呢?

  • 右值引用可以避免深拷贝
    假设A是一个类实例,类内有拷贝构造函数。
    如果使用ClassName B = A;那么会复制一份A的内容到B上,堆内就有两份相同的数据了。
    然而如果使用ClassName&& B = move(A);就只有一份数据,将A的所有权转移给B,避免了深拷贝。前提是类中必须有移动构造函数。为什么不用左值引用?
    只是为了转移资源,避免拷贝,把原来的A指向空,B把A的内容偷过来了,后续的修改也不会影响A的内容,这使得A能被重新赋值。
    右值引用在实现移动语义时,通常能在不影响原变量(更准确地说是源对象)的前提下(除了初次的使它置空),在不拷贝的情况下进行资源转移。
cpp 复制代码
#include <iostream>
class MyArray {
private:
    int* data;
    size_t size;
public:
    MyArray(size_t s) : size(s) {
        data = new int[s];
        for (size_t i = 0; i < s; ++i) {
            data[i] = i;
        }
    }
    // 拷贝构造函数
    MyArray(const MyArray& other) : size(other.size) {
        data = new int[size];
        for (size_t i = 0; i < size; ++i) {
            data[i] = other.data[i];
        }
    }
    // 移动构造函数
    MyArray(MyArray&& other) noexcept : data(other.data), size(other.size) {
        other.data = nullptr;
        other.size = 0;
    }
    ~MyArray() {
        delete[] data;
    }
};

完美转发

通过forward(arg)这个API能让函数传入的是左值或者右值的属性保留传递下去。

去掉forward直接传递arg的话会导致,第一次传入forwardExample这个函数时,在这函数生命周期中是右值,然而第二次传递进去的是一个arg的参数会被识别为左值。

图片来源程序员陈子青

4.深拷贝与浅拷贝

C++默认的拷贝构造函数就是浅拷贝,只是将新的指针指向同一块内存地址,这会导致某一个指针被释放的时候,另外一个成为野指针。

深拷贝需要重写,会在堆上分配一份新的内容并且将指针指向这个内容的地址。

而C++11后的智能指针shared_ptr可以完美解决这个问题,通过引用计数器的方式判断是否需要释放指向的空间。

5.malloc VS new 你们知道吗

malloc菜,不会自动分配内存,要自己sizeof分配,不知道类型,要自己转换。

new强,会自己分配内存,自己类型转换。

malloc不安全,new更安全。

free不会调用析构函数,delete会调用析构函数。

相关推荐
恋喵大鲤鱼2 小时前
Golang 后台技术面试套题 1
面试·golang
why技术2 小时前
也是震惊到我了!家里有密码锁的注意了,这真不是 BUG,是 feature。
后端·面试
★YUI★6 小时前
学习游戏制作记录(玩家掉落系统,删除物品功能和独特物品)8.17
java·学习·游戏·unity·c#
SmalBox6 小时前
【渲染流水线】[光栅阶段]-[光栅插值]以UnityURP为例
unity·渲染
谷宇.6 小时前
【Unity3D实例-功能-拔枪】角色拔枪(二)分割上身和下身
游戏·unity·c#·游戏程序·unity3d·游戏开发·游戏编程
DashVector6 小时前
如何通过Java SDK分组检索Doc
java·数据库·面试
Code_Artist6 小时前
[Go]结构体实现接口类型静态校验——引用类型和指针之间的关系
后端·面试·go
程序员清风7 小时前
跳表的原理和时间复杂度,为什么还需要字典结构配合?
java·后端·面试
张元清8 小时前
电商 Feeds 流缓存策略:Temu vs 拼多多的技术选择
前端·javascript·面试
Jenny8 小时前
第九篇:卷积神经网络(CNN)与图像处理
后端·面试