1.3 C/C++ 指针、引用(重头戏)

一、数组指针和指针数组的区别

数组指针:int (*ptr)[10];指针数组:int * ptr[10]

二、函数指针和指针函数的区别

函数指针:int (*ptr)(int,int);指针函数:int * ptr(int,int)

三、数组名和指针的区别

数组名是常量,不可以改值;指针是变量,可以改值;

数组名使用的时候会隐式转换为第一个元素的地址,只有 &数组名 才表示整个数组的起始地址。

四、为什么要字节对齐

优化存储器访问速度:硬件架构要求数据按特定边界进行存储,可以提高内存访问效率,否则就可能需要额外的访问次数。

五、 结构体内存对其规则

1、每个成员的起始地址必须是该成员类型大小的整数倍。(成员变量对齐)

2、结构体的总大小必须是结构体中最大对齐边界的整数倍。(结构体整体对齐)

struct BBB {
    long num;
    char *name;
    short int data;
    char ha;
    short ba[5];
};
// 4+4+2+1+1个空白字节+2*5=22个字节 第一个规则
// 因为最大对其边界是4,所以是24字节。

六、注意指针加法和数值加法区别

指针加法:p+0x20 = p + sizeof(p的类型)×0x20;

七、指针常量、常量指针、指向常量的常量指针区别?

指针常量:int * const ptr;

常量指针:const int * ptr;

指向常量的常量指针:const int * const ptr;

八、有常量引用,但没引用常量

因为引用本身就是常量,一旦绑定到一个对象后,不能再绑定到其他对象。

c++ 也不允许 int& const 这种写法。

九、指针和引用的区别?

1、指针是实体,是一个变量;引用是别名,不占额外存储空间,由编译器实现替代;

2、 指针和引用的自增运算符意义不同,指针是加 sizeof(类型),引用是对引用的变量值自增。

3、引用使用无需解引用,指针需要;

4、引用本质上是个指针常量,只能在定义时被初始化。

5、引用不能为空,指针可以;

6、sizeof 引用 返回的是引用的变量的类型大小;sizeof 指针 返回的是指针类型的大小(4)。

特性

指针

引用

本质

存储内存地址的变量

变量别名(编译器的语法糖)

初始化

可以延迟初始化

必须在定义时初始化

可空性

可以为 nullptr

必须绑定到有效对象(不可空)

重绑定

可以修改指向的地址

一旦绑定后不可更改

内存占用

占用独立存储空间(通常 4/8 字节)

通常不占用额外存储空间(由编译器实现)

多级间接

支持多级指针(如 int**)

只有一级引用

运算符

使用 * 和 -> 操作

直接像普通变量一样使用

十、指针不能指向引用,但可以有指针的引用

因为引用本身就是一个逻辑概念,不是物理实体,没有地址,自然无法取地址给指针。

因为指针本身就是一个实体,是变量。所以可以引用。

十一、 野指针是什么

在指针定义之后,没有将其指向其他变量或内存,也没有将其赋值为 NULL

当指针指向的内存空间被释放后,没有将其赋值为 NULL(悬空指针)

十二、 如何避免野指针

1、指针在使用前对其赋值 NULL。

2、指针用完后释放内存,将指针赋值为 NULL。

3、当指针指向 malloc 申请的内存时,需要检查内存是否申请成功。

十三、NULL 和 nullptr 区别

NULL 作为宏定义,它可能被错误地解释为整型 0,可能会导致二义性问题。

nullptr 是一个专门的空指针类型,类型为 std::nullptr_t,因此避免了 NULL 所引起的二义性。

NULL 可以在整型上下文中使用(例如传递给整型函数),但 nullptr 只能用于指针上下文

void func(int) {
    cout << "Function with int argument"<< endl;
}
void func(char*){
    cout << "Function with char* argument"<< endl;
}
int main(){
    int* ptr1 = NULL;  // NULL用法,指针初始化为空
    int* ptr2 = nullptr;  //nullptr用法,指针初始化为空
    
    cout << "ptr1: "<< ptr1 << endl; // //打印 NULL指针的地址
    cout <<"ptr2:"<< ptr2<<endl; //打印 nullptr 指针的地地址

    //使用NULL和nullptr进行函数调用
    func(NULL);  //这里NULL会导致二义性而报错
    
    func(nullptr);  //这里会调用func(char*),因为nullptr是一个空指针
    return 0;
}

十四、使用一级指针无法解决的问题

传入一级指针只可以修改指针指向内存的内容,但无法修改指针本身的值,因为指针本身传入的时候也是临时变量,只是里面存放的地址和实参一样。

#include <stdio.h>
#include <stdlib.h>
//正确的函数声明,接收 int*类型的指针
void wrongFunction(int *ptr) {
    ptr = (int*)malloc(sizeof(int)); //修改指针,但不能影响外部指抖
    *ptr = 42;
}
int main() {
    int* ptr = NULL;//初始化为NULL

    //调用函数时,确保传递的是一个 int*类型的指针
    wrongFunction(ptr);

    //检查ptr是否改变
    if (ptr == NULL) {
        printf("Pointer is still NULL, memory was not allocatedin wrongFunction.\n");
    } else {
        printf("Pointer points to: %d\n",*ptr);//这行不会被执入行
    }
    return 0;
}

使用二级指针

#include <stdio.h>
#include <stdlib.h>

//正确的函数声明,接收 int **类型的指针
void correctFunction(int **ptr) {
    *ptr=(int*)malloc(sizeof(int)); //修改外部指针
    **ptr=42; //修改堆内存中的值
}

int main() {
    int*ptr=NULL;//初始化为NULL

    //调用正确的函数,传入二级指针
    correctFunction(&ptr);
    //检查ptr是否改变
    if (ptr != NULL) {
        printf("Pointer points to: %d\n",*ptr); //输出 42
        free(ptr);//释放堆内存
    }
    return 0;
}

十五、C++ 中的智能指针是什么

智能指针为了解决野指针、重复释放、内存泄露问题而被发明。

智能指针是一个类,常用的有 unique_ptr、shared_ptr、weak_ptr。(注意 auto_ptr 因为安全问题已经在 C++17 中移除了

unique_ptr 每个 unique_ptr 独占一块内存,不允许被赋值,但可以通过 std::move() 转交所有权,此时原来的 unique_ptr 变为了 nullptr;当 unique_ptr 被销毁时,其指向的对象实例会被自动调用析构函数进行释放,没有引用计数。

在多线程环境中将一个 unique_ptr的资源所有权从一个线程转移到另一个线程时,必须通过线程同步机制(如互斥锁、原子操作等)保证这一操作的安全性。

shared_ptr 是一种共享所有权的智能指针。通过引用计数的方式可以使多个 shared_ptr 指针指向同一块内存区域,只有计数值为 1 的时候,释放指针才会真正释放内存,引用计数是原子操作的(线程安全)。

shared_ptr 大小为 2 个指针大小:一个是原始指针;一个指向控制快。

template <class T>
class shared_ptr {
public:
    shared_ptr(T* ptr = nullptr)
        : ptr_(ptr), block_(nullptr) {
        if (ptr_) {
            block_ = new ControlBlock;
            block_->refcnt.store(1);
        }
    }
    ~shared_ptr() {
        release();
    }
    shared_ptr(const shared_ptr& sp)
        : ptr_(sp.ptr_), block_(sp.block_) {
        if (block_) block_->refcnt.fetch_add(1, std::memory_order_relaxed);
    }
    shared_ptr& operator=(const shared_ptr& sp) {
        if (this != &sp) {
            release();
            ptr_ = sp.ptr_;
            block_ = sp.block_;
            if (block_) block_->refcnt.fetch_add(1, std::memory_order_relaxed);
        }
    }
    shared_ptr(shared_ptr&& other)
        : ptr_(other.ptr_), block_(other.block_) {
        other.ptr_ = nullptr;
        other.block_ = nullptr;
    }
    shared_ptr& operator=(shared_ptr&& other) {
        if (this != &other) {
            release();
            ptr_ = other.ptr_;
            block_ = other.block_;
            other.ptr_ = nullptr;
            other.block_ = nullptr;
        }
        return *this;
    }

    T& operator*() {
        return *ptr_;
    }
    T* operator->() {
        return ptr_;
    }

    T* get() const {
        return ptr_;
    }

    long use_count() const {
        return block_->refcnt.load();
    }

    void reset(T* ptr = nullptr) {
        if (ptr_ != ptr) {
            release();
            if (ptr) {
                block_ = new ControlBlock();
                block_->refcnt.store(1);
                ptr_ = ptr;
            } else {
                block_ = nullptr;
                ptr_ = nullptr;
            }
        }
    }

    void swap(shared_ptr& other) {
        std::swap(ptr_, other.ptr_);
        std::swap(block_, other.block_);
    }

private:
    void release() {
        if (block_) {
            if (block_->refcnt.fetch_sub(1, std::memory_order_acq_rel) == 1) {
                delete ptr_;
                delete block_;
            }
            ptr_ = nullptr;
            block_ = nullptr;
        }
    }

    struct ControlBlock {
        std::atomic<long> refcnt{0};
    };

    T* ptr_ {nullptr};
    ControlBlock* block_ {nullptr};
};

weak_ptr 是一种不共享所有权的指针,用于解决循环引用的问题,不会增加引用计数,也不会阻止所指内容被销毁

十六、shared_ptr 是线程安全的吗?

不是。

引用计数的更改是安全的,因为是原子操作,要么执行完毕,要么不执行。

但多个线程的拷贝、释放不是安全的,因为内部 ptr 指向的更改、引用计数的修改,两者是分步执行的,不是原子的。需要加锁。

十七、 智能指针是怎么释放的?

1、unique_ptr

  • 当 unique_ptr 离开作用域(如函数结束、块作用域结束)时,它管理的对象会被自动释放。
  • 如果显式调用 reset() 或赋值为 nullptr,也会立即释放对象。
  • 如果 unique_ptr 被 std::move 转移所有权,原指针变为 nullptr,新指针接管所有权。

2、shared_ptr

  • 当最后一个持有对象的 shared_ptr 被销毁或 reset() 时,对象才会被释放。
  • 引用计数(use_count())降为 0 时触发释放。

3、weak_ptr:仅观察,不释放。

十八、如何把 shared_ptr 转给 unique_ptr?

只有当共享指针的引用计数为 1(即没有其他共享指针指向该对象)时,才能安全地转换。

#include <memory>

std::shared_ptr<int> shared = std::make_shared<int>(42);

// 检查引用计数是否为1
if (shared.use_count() == 1) {
    std::unique_ptr<int> unique = std::unique_ptr<int>(shared.get()); // 裸指针
    shared.reset();  // 必须手动释放shared_ptr的所有权
    // 现在unique独占所有权
} else {
    // 不能转换,因为有其他shared_ptr共享所有权
}

十九、循环引用

循环引用指的是两个或多个对象之间相互持有对方的引用,形成一个闭环。当使用智能指针(如 std::shared_ptr)时,这种循环引用会导致资源无法正常释放,进而造成内存泄漏。

#include <iostream>
#include <memory>

class B;  // 前向声明

class A {
public:
    std::shared_ptr<B> b_ptr;
    ~A() { std::cout << "A destroyed" << std::endl; }
};

class B {
public:
    std::shared_ptr<A> a_ptr;
    ~B() { std::cout << "B destroyed" << std::endl; }
};
int main() {
    std::shared_ptr<A> a = std::make_shared<A>();
    // 此时创建了一个 A 类型的对象,并且由 std::shared_ptr a 管理。
    // A 对象的引用计数为 1,因为只有 a 指向它。

    std::shared_ptr<B> b = std::make_shared<B>();
    // 此时创建了一个 B 类型的对象,并且由 std::shared_ptr b 管理。
    // B 对象的引用计数为 1,因为只有 b 指向它。

    a->b_ptr = b;
    // a 对象中的成员变量 b_ptr 指向了 b 所管理的 B 对象。
    // 此时 B 对象的引用计数变为 2,因为 b 和 a->b_ptr 都指向它。

    b->a_ptr = a;
    // b 对象中的成员变量 a_ptr 指向了 a 所管理的 A 对象。
    // 此时 A 对象的引用计数变为 2,因为 a 和 b->a_ptr 都指向它。

    return 0;
    // 当 main 函数结束时,局部变量 a 和 b 超出作用域。
    // a 超出作用域,A 对象的引用计数减 1,但由于 b->a_ptr 仍然指向 A 对象,所以 A 对象的引用计数变为 1。
    // b 超出作用域,B 对象的引用计数减 1,但由于 a->b_ptr 仍然指向 B 对象,所以 B 对象的引用计数变为 1。
    // 由于 A 对象和 B 对象的引用计数都不为 0,它们不会被销毁,从而造成内存泄漏。
    // 但程序退出后,系统依然能够回收该程序所占用的所有内存资源。因为操作系统在进程终止时,会清理该进程的所有资源,包括虚拟内存空间、文件描述符等。
}

二十、如何避免 shared_ptr 循环引用(这个在我实习面试的时候真考了

为了解决循环引用问题,可以使用std::weak_ptrstd::weak_ptr是一种不控制对象生命周期的智能指针,它指向由std::shared_ptr管理的对象,但不会增加对象的引用计数。

#include <iostream>
#include <memory>

class B;  // 前向声明

class A {
public:
    std::shared_ptr<B> b_ptr;
    ~A() { std::cout << "A destroyed" << std::endl; }
};

class B {
public:
    std::weak_ptr<A> a_ptr;  // 使用 std::weak_ptr
    ~B() { std::cout << "B destroyed" << std::endl; }
};

int main() {
    std::shared_ptr<A> a = std::make_shared<A>();
    // 此时创建了一个 A 类型的对象,并且由 std::shared_ptr a 管理。
    // A 对象的引用计数为 1,因为只有 a 指向它。

    std::shared_ptr<B> b = std::make_shared<B>();
    // 此时创建了一个 B 类型的对象,并且由 std::shared_ptr b 管理。
    // B 对象的引用计数为 1,因为只有 b 指向它。

    a->b_ptr = b;
    // a 对象中的成员变量 b_ptr 指向了 b 所管理的 B 对象。
    // 此时 B 对象的引用计数变为 2,因为 b 和 a->b_ptr 都指向它。

    b->a_ptr = a;
    // b 对象中的成员变量 a_ptr 是 std::weak_ptr 类型,它指向 a 所管理的 A 对象。
    // 但由于 std::weak_ptr 不会增加对象的引用计数,所以 A 对象的引用计数仍然为 1。

    return 0;
    // 当 main 函数结束时,局部变量 a 和 b 超出作用域。
    // a 超出作用域,A 对象的引用计数减 1,变为 0。此时 A 对象被销毁,A 的析构函数被调用,输出 "A destroyed"。
    // 随着 A 对象被销毁,a->b_ptr 不再指向 B 对象,B 对象的引用计数减 1,变为 1。
    // b 超出作用域,B 对象的引用计数再减 1,变为 0。此时 B 对象被销毁,B 的析构函数被调用,输出 "B destroyed"。
}

C++/嵌入式开发 秋招面经 文章被收录于专栏

一名985硕,在25年秋招中斩获多个C++/嵌入式开发Offer。本专栏将分享我的面经,涵盖C/C++、操作系统、计算机网络、ARM体系与架构、Linux应用/驱动开发、Qt、通信协议及开发工具链等核心内容。

全部评论
收藏了
1 回复 分享
发布于 04-02 20:50 河北
数组名是常量😀
点赞 回复 分享
发布于 04-03 00:21 河北
求面经
点赞 回复 分享
发布于 04-03 00:15 河北
欢迎订阅专栏《C++/嵌入式开发 秋招面经》 :https://www.nowcoder.com/creation/manager/columnDetail/MKaoll
点赞 回复 分享
发布于 03-30 17:04 河北

相关推荐

04-26 14:36
已编辑
郑州信息科技职业学院 Java
由于高考成绩不是很理想,听取了张雪峰老师的建议,优先选了专业并且当时的想法就是选一个能赚钱的专业,于是最终选择了报了一个能收留我的有计算机专业的学校。当时听张雪峰老师说河南的学习氛围很好,所以就想去体验一下,事实雀食如张雪峰老师所说,大家都一股脑的铺在学习这条路上。可能是因为那边氛围导致的吧,我一开始想的也是卷学习卷绩点,所以大一的时候就一直在学习硬试教育的一些东西,学期结束了,排名出来的时候中上水平吧,据我了解保研的只有前5名可能会有机会,当时的心里就想着,我这成绩再卷也卷不到哪去了,并且保研也无望了,总结的说,一些事情只有真正做了才知道是不是自己所追求的。说了很多废话吧,剩下的关于学校的就长话短说了吧。大二很多专业课基本上要从早八上到晚上,但基本上我都是不去,不如自学现在新媒体技术这么发达,并且还可以学一下自己需要的技术栈,由于学校的课程原因对其他的技术栈不是很了解,所以,一心就投入在Java这个方向了,但是,Python也会学一下,这是因为加入实验室,实验室老师是做人工智能方向的缘故。现在回想,我大二当时还是学的太慢了,还有就是信息差太大了,出来工作之后才发现有些佬们已经大二就出来实习,并且八股就背的滚瓜烂熟了。只能说这里的学习氛围很好吧,走廊里都是背书刷题的声音,跟身边的同学和实验室的同学谈是否直接就业的事,他们要么都是说考研,要么对直接就业很含糊,可能是因为觉得自己学的还不够吧,我想说,学的不够就干中学呗,反正,我先迈出去这步再说。到了大三上还是没有找工作的打算,因为身边的人也都还没有这个意识吧,现在跟了身边的同事聊天才知道,我的信息差太大了。到了大三下刚开始,我才开始正式的踏上求职路,当时的信息差还是很大的,根本就不敢碰瓷大厂,想着有一个公司能要再说吧,并且地域也限制的很死,只想着在本地找一下,因为怕学校找事(我想这是学校一贯操作了),在本地吧,他们大多数都是接受的线下面,一开始面了一个,可能自己比较摆也很悲观,就显得我很差吧,hr面完就没后续了,最终终于有一个面,并且也展示出自己的自信和对专业的理解了,最后,我也没想着这么多背调公司呀,当个备选什么的就直接去了。也算是我的第一家正式的公司吧(之前都是线上的码农兼职),干多了就发现,这个公司压根学不到东西,并且薪资低的,因为我是第一个进来的计算机实习生,有一个同事干了两三年的吧,带着我做的时候是真能学到东西,但是,最后那个同事离职了,我就只能和学艺术的老板直接汇报项目进度,一个学艺术的来指导我这个科班出身的就很离谱的好吧。最后,我也离职了,也跟前同事聊了很久,她说我是她见过大三就能学到这程度,已经超过很多人了,并且她当时在的时候还说我是内定能转正的。并且还说我真的可以去考研。我也仔细思考了一下,我决定让自己沉淀一下再出发吧,先备考了软件设计师,然后期末考,大三暑期的时候就充实自己的简历,并且也认识了一个某东的老哥,也用了内推码,教我了怎么写好简历量化成果之类的,总之,很感谢一路走来帮助我的人吧,并且我在边充实自己的同时也在边投递简历,但当时卡的也很死,要选base地在河南附近的,不像现在全国可飞。面了很多base地在学校附近的,然后,还有一个北京的py和杭州的java,最终就这两个地方给了offer,但是都是实习转正的,不是秋招offer,因为觉得Java的太卷了,然后,面试的时候也会感觉压力很大,所以就把杭州的那个拒了,去了北京的,北京是免费住的房子(三个月这是伏笔),当时觉得环境很好,但是合租室友的作息跟自己的作息不一样就很不习惯,于是,我就想着要是三个月后我一定要找一个单间的哪怕破一点。北京这个公司吧就很像国企的感觉,早九晚五,当月发当月工资,并且干的活接触的数据量都不是很大,就是干了很多杂活,并且mentor和部门的领导都不是技术出身,所以,我能学到的东西少之又少,但是吧,学习是自己的事,而且这部门不是很忙对于实习生来说,我完全可以学自己的东西(前提是不被发现)。到最后这个部门的氛围就很微妙,我遇到不会的问他们我应该怎么做的时候,他们说让我自己想,我当时就想说,神人一个,啥都不说让我自己干,干出来又不满意,你说你让我干py的东西你不会我就不说啥了,让我干无关代码的东西,让我调研项目应该做些什么内容,现在回想都是泪呀,我就这样被欺压的过完了三个月,最后免费住的地方也到期了,伏笔来了,最后,找我谈话说你技术可以了能看出来,因为你也自己独立完成了消息通知那一块内容嘛,但是,由于我们部门干的活比较杂并且我也缺少一些电力相关的一些知识,所以,觉得不合适。(OS:其实我对每一份工作都是真心换真心的,并且这些电力知识我也知道我有一点欠缺所以我也有自己再学习,你们啥也不教我,最后把屎盆子把我头上扣)最后,回到了学校,心态也发生了变化,想着做啥都不如找一个稳定的工作重要,想着回家沉淀吧,少年终有出头日。但是,计划赶不上变化,之前那个同事,内推了我去她现在的公司,并且是做AI应用的也是我想接触的,并且还是与我上家的业务场景类似的,真的感谢那个同事,俗话说:千里马常有而伯乐不常有。并且那里的部门领导也很好,并且说我虽然不是电力相关出身的,但是能做的这样已经很不错了,所以DDDD,由于各种不可抗力因素吧,还是想找一个离家近,然后不是很像小作坊的感觉(这个公司虽然比较小,但是比之前那个大的公司的氛围和待遇一点都不差的好吧甚至更好)。最终,在学校也呆了一个月吧,也陆陆续续面了一个月有一个C厂的面答的都挺好直接就谈薪了,但是风评不好还是保命要紧,还有各种的中小厂面吧,但感觉都不是自己想要的,只是想刷刷面试经验吧(这是某东哥告诉我的,与其一直改简历不如去多面)。最后,在校期间面了一个比较合适的某鸦智能,一直推进到了HR面,但是最后被横向了,开始复盘,被横向了属实是没招了,经历了这么多大风大浪什么场面没见过。过年期间,求职路线关闭,把自己缺少的技术栈和简历中的项目业务理清楚说明白。年过完就要开始加入找工作大军中了,把节前没面完的先面了,节后一开始就是某鸟的HRG面,聊的就很憋屈的感觉,问我技术方面的,说我说的很像AI的(我心想跟你说具体的细节你又说我不想听技术的,说的比较宽泛浅显说我AI)。最后,反正体验感不是很好的结束了吧。说一个星期等通知,等了两个星期才说是通过的(我认为是排名靠前的那些人没去,顺位到我了)。那你既然这样说了,那我就接受吧。还没入职就问我要身份证信息要这要那的,最后都给过去了,说HC调整,要重新review,又又又一次被恶心到了。后面就是陆续的沉淀面试等,我当时的重心已经完全的想着私企没人要,就去试试考公和考央国企了,毕竟我的履历不看学历的话放到电网当中还是可以的。私企的话有一个外企洋里洋气的说话,问我怎么口语这么好?我说这叫智取,宝贝。虽然这个tek外企过了,但是还有一个openday要去线下,来回的衣食住行不是很方便也不是很想去所以就拒绝了没去。后来就收到了,国网网申通过的通知,说实话,我之前问了很多我们学校历年有没有考央国企之类的案例,很显然都不知道,也可以说少之又少吧,于是我就奔赴京城进京赶考,唉,时间不太合适就想着算了吧,再等等,好事多磨,宁缺毋滥吧。金三银四终于等来了面试的机会,这个岗位我只能说我不是很熟悉,但是语言这东西吧都是相通的,重要的是我要把其中的内核搞懂,梳理清楚业务逻辑。最终,来到了这家公司,目前来说是我遇到过最好的了,能有hc且不是要通过实习评估的那种,并且合同期限是三年的,并且是12%的公积金。我认为这就是我所遇到的最好的了。希望能真心换真心吧,不再把我当创口贴/路边一条了,并且也遇到了很多优秀的同事。总的来说,就是要是能重来我要选李白。我肯定会打破这些信息差,后悔知道的太晚,并且跟优秀的人聊天说话真的可以学到很多东西,之前上文提到的贵人就不说了,说说最近的,他是跟我一届,学校后缀甚至不如我的后缀,但是真正了解的才会知道真是佬👍,他跟我找工作的时间线差不多,但是他在中大厂甚至大厂都呆过,因为跟他聊了才知道我当时的信息差有多大,并且毅力也是我甚至…都没有的。并且也听说了他们学校找工作的氛围很好,不像我阿巴阿巴阿巴,只有考研等相关的一些。并且说的一些观点都是很认同的。总之,希望能在这好好的吧,我真的不想经历大起大落了。经历了,打招呼挂,简历挂,一面挂,HR面挂,offer挂的,现在的心态已经放宽了很多了,但是难过还是有的,希望这家公司诚不欺我吧。也祝大家遇到自己的梦中情厂
选择和努力,哪个更重要?
点赞 评论 收藏
分享
04-24 18:13
南京大学 Java
不吃酸菜血肠:看力竭了
点赞 评论 收藏
分享
评论
5
3
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务