【C++】---STL之list的模拟实现

news/2024/5/19 13:21:04

【C++】---STL之list的模拟实现

  • 一、list模拟实现思路
  • 二、结点类的实现
  • 三、list迭代器的实现
    • 1、ListIterator类
    • 2、构造函数
    • 3、operator*运算符重载
    • 5、operator->运算符重载
    • 6、operator!=运算符重载
    • 7、operator==运算符重载
    • 8、前置++
    • 9、后置++
    • 10、前置--
    • 11、后置--
  • 四、list类的实现
    • 1、list类
    • 2、构造
    • 3、析构
    • 4、拷贝构造
    • 5、赋值运算符重载
      • (1)传统的赋值运算符重载
      • (2)现代的赋值运算符重载
    • 6、迭代器
    • 7、insert()
    • 8、erase()
    • 9、clear()
    • 10、push_front()
    • 11、push_back()
    • 12、pop_front()
    • 13、pop_back()
    • 14、empty()
    • 15、size()
  • 五、完整代码

一、list模拟实现思路

list的模拟实现比 string vector的模拟实现略微复杂一点:

(1)由于链表的每一个结点本身就是一个结构体,里面包括数据和指针,所以在接下来的模拟中,我们会将链表的每一个结点封装为一个类,也就是结点类。

(2)链表中数据的物理储存空间是不连续的,但是string和vector他们的数据储存物理空间是连续的。因此在访问链表的数据的时候,不能用原生的迭代器来进行访问,我们需要自己重载一个迭代器,自己封装一个迭代器的类。

在这里插入图片描述
list的模拟的大体思路:
在这里插入图片描述

二、结点类的实现

单个结点类的成员变量有三个:

(1)结点值:_val

(2)指向前一个结点的指针:_prev

(3)指向后一个结点的指针:_next

结点无需拷贝构造、赋值运算符重载,由于没有额外申请空间,因此也不需要析构

	// 1.单个的结点类:template<class T>struct Listnode{T _val;Listnode<T>* _next;Listnode<T>* _prev;// 构造:Listnode(const T& x = T()):_val(x), _next(nullptr), _prev(nullptr){}};

三、list迭代器的实现

1、ListIterator类

(1)我们为什么要对链表的迭代器进行一个单独的封装?

因为之前普通的迭代器++都是连续,可以直接进行访问数据。

但是链表不一样,物理空间连续所以说我要把这个迭代器进行一个类的封装,然后在里面对他运算符重载(例如:++)我们就可以掌控这个迭代器的行为!

当原生的迭代器或者运算符不合我们所需要的预期的话,就可以把它进行一个封装,我们自己来重载,达到我们所需要的预期

(2)迭代器有两种,一种是普通迭代器,一种是const的迭代器

为了不使代码冗余,我们就会将两个迭代器写在一起,用模板!

对于T&,类模板实例化出两个类,一个是T&类,一个是const T&类,同理,T*也一样。使用 :

template<class T,class Ref,class Ptr>// Ref==T&      Ptr==T*

类模板就会实例化出来两个类,一个是普通的、不带const的T,T&, T*,另一个是带const的T,const T&, const T*,其中Ref是引用,Ptr是指针,该类模板实例化了以下这两个类模板:

template class<T,T&,T*> iterator;
template class<const T, const T& ,const T*> const_iterator;

这样我们就解决了两个类的问题。

2、构造函数

template<class T,class Ref,class Ptr>struct ListIterator{typedef Listnode<T> Node;// 1.(这个是单个结点“类型”的重定义) 不管你是什么类型的结点 我都给你整成Node,因为Listnode<T>是一个结点模版!typedef ListIterator<T, Ref, Ptr> Self; // 2.(这个是本迭代器指针“类型”的重定义)// 成员变量:Node* _node;// 构造:ListIterator(const Node* node):_node(node){}};

3、operator*运算符重载

// 重载*(*it)// Ref==T&Ref operator*()// 为什么要传引用返回呢?因为有的时候我们可能需要对it进行修改+1,-1等等。{return _node->_data;}

5、operator->运算符重载

// 重载->//Ptr==T*Ptr operator->(){

6、operator!=运算符重载

对于==和!=的重载的时候,我们一定要想清楚到底是对它里面节点的值来判断相不相等,还是说来判断指向这个结点的迭代器指针相不相等。很明显我们这里重载( = =)和(!=)通过判断结点的迭代器相等不相等来进行重载的。

// !=bool operator!=(const Self& it){return _node != it._node;}

7、operator==运算符重载

比较两个迭代器相等不相等的时候一定不能比较所指向节点中的值,万一所有的节点里面值相等都是一样,那你意思就是说:这里面的所有迭代器都是相等吗?不就扯淡吗?!所以说比较迭代器相不相等:就是比较两者是不是指向同一个结点(即:比较指针是否相等!)因为迭代器本质上是指针!

bool operator==(const Self& it){return _node == it._node;}

8、前置++

//前置++,(++it)Self& operator++()//因为++对内容进行了修改,所以要传引用返回!{_node = _node->_next;return *this;}

9、后置++

//后置++,(it++)Self operator++(int){Self tmp = *this;//因为后置++,要返回的是++之前的值,所以要先保存未++的值在tmp里面!_node = _node->_next;return tmp;}

10、前置–

//前置--,(--it)Self& operator--()//因为--对内容进行了修改,所以要传引用返回!{_node = _node->_prev;return *this;}

11、后置–

//后置--,(it)Self operator--(int){Self tmp = *this;//因为后置--,要返回的是--之前的值,所以要先保存未--的值在tmp里面!_node = _node->_prev;return tmp;}

四、list类的实现

1、list类

list的成员只需要一个头节点,然后通过迭代器来访问后面的其他元素即可。

2、构造

//1、构造:list(){_head = new Node;//会调ListNode的构造函数_head->_next = _head;//整个链表只有头节点,先构造一个没有实际节点的链表_head->_prev = _head;//整个链表只有头节点,先构造一个没有实际节点的链表}

3、析构

// 2、析构~list(){clear();delete[] _head;_head = nullptr;}

4、拷贝构造

		//特意写一个,初始化一个哨兵位void empty_init(){_head = new Node;_head->_next = _head;_head->_prev = _head;_size = 0;}// 3、拷贝构造// lt2(lt1)list(const list<T> lt){empty_init();//先初始化一个头结点for (auto& e : lt)// 接下来在哨兵位后面 尾插 就可以实现拷贝构造!{push_back(e);}}// 需要析构,一般就需要自己写深拷贝// 不需要析构,一般就不需要自己写深拷贝,默认浅拷贝就可以};

5、赋值运算符重载

(1)传统的赋值运算符重载

        //赋值运算符重载  lt1 = lt  传统写法list<T> operator=(const list<T>& lt){//链表已存在,只需将节点尾插进去即可if(this != lt){for (auto& e : lt){push_back(e);}}}

(2)现代的赋值运算符重载

//4、赋值运算符重载(深拷贝)// lt1=lt2list<T>& operator=(list<T> lt){swap(lt);return  *this;}void swap(list<T>& lt){std::(_head, lt._head);std::(_size, lt._size);}

6、迭代器

(1)普通迭代器:

iterator begin(){//iterator it = _head->_next;// 有名对象 //调用迭代器的构造函数创建一个迭代器it//return it;return iterator(_head->_next);// 匿名对象// return _head->_next; // 不能这样写,因为返回类型是迭代其指针,而你这样返回的是一个结点。}iterator end(){return iterator(_head);}

只要你有节点的指针,就可以构造迭代器:
下面这里就是构造了一个迭代器,因为它的返回类型是迭代器,你只要有节点的指针我就可以构造一个迭代器,只不过有两种情况是匿名对象,另外一种是有名对象:
在这里插入图片描述
(2)const迭代器:

		const_iterator begin() const{return const_iterator(_head->_next);//头节点不存数据}const_iterator end() const{return const_iterator(_head);//尾节点的下一个节点位置即头节点}

7、insert()

// 3.insertvoid insert(iterator pos, const T& val)//在pos位置之前插入val{//先用一个指针保存pos的位置!Node* cur = pos._node;//创建一个新的节点newnode来接受val的值Node* newnode = new Node(val);//再保存pos位置前一个方便newnode插入!Node* prev = cur->_prev;//prev newnode cur三者之间的交换newnode->_prev = prev;prev->_next = newnode;newnode->_next = cur;cur->_prev = newnode;}

8、erase()

iterator erase(iterator pos){// 1、先保存pos位置的前后!Node* cur = pos._node;Node* prev = cur->_prev;Node* next = cur->_next;// 2、prev 和 next两者之间进行链接!prev->_next = next;next->_prev = prev;// 3、直接删除curdelete cur;// 4、因为是模拟原本库里面的erase函数,返回的就是要删除pos位置的下一个位置的迭代器。return iterator(next);}

9、clear()

	void clear(){iterator it = begin();while (it != end()){it = erase(it);//因为erase会返回要删除结点的下一个位置,所以要用iterator类型的it接受!}}

10、push_front()

// 头插void push_front(const T& x){insert(begin(), x);}

11、push_back()

// 尾插void push_back(const T& x){insert(end(), x);}

12、pop_front()

// 头删void pop_front(){erase(begin());}

13、pop_back()

// 尾删void pop_back(){erase(--end());}

14、empty()

bool empty(){return (_head->_next == _head);}

15、size()

size_t size()const{size_t count = 0;Node* cur = _head;while (cur->_next != _head){cur = cur->_next;count++;}return count;}

五、完整代码

#pragma once
#include <assert.h>
#include<iostream>
using namespace std;namespace yjl
{template<class T>struct Listnode{Listnode<T>* _prev;Listnode<T>* _next;T  _data;//单个节点之间的内部构造Listnode(const T& x = T()):_prev(nullptr), _next(nullptr), _data(x){}};/// ///list迭代器的封装://template<class T>//struct ListIterator//{//	typedef Listnode<T> Node;// 1.(这个是单个结点“类型”的重定义) 不管你是什么类型的结点 我都给你整成Node,因为Listnode<T>是一个结点模版!//	typedef ListIterator<T> Self; // 2.(这个是本迭代器指针“类型”的重定义)//	Node* _node;//	//构造//	ListIterator(Node* node)//		:_node(node)//	{}//	// 重载*(*it)//	const T& operator*()// 为什么要传引用返回呢?因为有的时候我们可能需要对it进行修改+1,-1等等。//	{//		return _node->_data;//	}//	// 重载->//	const T* operator->()//	{//		return &_node->_data;//得到的是地址:T*//	}//	//前置++,(++it)//	Self& operator++()//因为++对内容进行了修改,所以要传引用返回!//	{//		_node = _node->_next;//		return *this;//	}//	//后置++,(it++)//	Self operator++(int)//	{//		Self tmp = *this;//因为后置++,要返回的是++之前的值,所以要先保存未++的值在tmp里面!//		_node = _node->_next;//		return tmp;//	}//	//前置--,(--it)//	Self& operator--()//因为--对内容进行了修改,所以要传引用返回!//	{//		_node = _node->_prev;//		return *this;//	}//	//后置--,(it)//	Self operator--(int)//	{//		Self tmp = *this;//因为后置--,要返回的是--之前的值,所以要先保存未--的值在tmp里面!//		_node = _node->_prev;//		return tmp;//	}//	bool operator!=(const Self& it)//	{//		return _node != it._node;//	}//	bool operator==(const Self& it)//	{//		return _node == it._node;//	}//};// typedef ListIterator<T,T&,T*> iterator;// typedef ListIterator<T,const T&,const T*>  const_iterator;//list迭代器的封装:template<class T,class Ref,class Ptr>// Ref==T&      Ptr==T*struct ListIterator{typedef Listnode<T> Node;// 1.(这个是单个结点“类型”的重定义) 不管你是什么类型的结点 我都给你整成Node,因为Listnode<T>是一个结点模版!typedef ListIterator<T,Ref,Ptr> Self; // 2.(这个是本迭代器指针“类型”的重定义)Node* _node;//构造ListIterator(Node* node):_node(node){}// 重载*(*it)// Ref==T&Ref operator*()// 为什么要传引用返回呢?因为有的时候我们可能需要对it进行修改+1,-1等等。{return _node->_data;}// 重载->//Ptr==T*Ptr operator->(){return &_node->_data;//得到的是地址:T*}//前置++,(++it)Self& operator++()//因为++对内容进行了修改,所以要传引用返回!{_node = _node->_next;return *this;}//后置++,(it++)Self operator++(int){Self tmp = *this;//因为后置++,要返回的是++之前的值,所以要先保存未++的值在tmp里面!_node = _node->_next;return tmp;}//前置--,(--it)Self& operator--()//因为--对内容进行了修改,所以要传引用返回!{_node = _node->_prev;return *this;}//后置--,(it)Self operator--(int){Self tmp = *this;//因为后置--,要返回的是--之前的值,所以要先保存未--的值在tmp里面!_node = _node->_prev;return tmp;}bool operator!=(const Self& it){return _node != it._node;}bool operator==(const Self& it){return _node == it._node;}};/// ///template<class T>class list{typedef Listnode<T> Node;public:typedef ListIterator<T,T&,T*> iterator;typedef ListIterator<T, const T&,const T*> const_iterator;iterator begin(){//iterator it = _head->_next;// 有名对象 //调用迭代器的构造函数创建一个迭代器it//return it;return iterator(_head->_next);// 匿名对象// return _head->_next; // 不能这样写,因为返回类型是迭代其指针,而你这样返回的是一个结点。}iterator end(){return iterator(_head);}// 1.多个节点之间的构造:初始化一个哨兵位//特意写一个,初始化一个哨兵位void empty_init(){_head = new Node;_head->_next = _head;_head->_prev = _head;_size = 0;}// 构造list(){empty_init();}// 拷贝构造函数// lt2(lt1)list(const list<T>& lt){empty_init();// 先构造一个哨兵位头结点for (auto& e : lt)// 接下来在哨兵位后面 尾插 就可以实现拷贝构造!{push_back(e);}}// 需要析构,一般就需要自己写深拷贝// 不需要析构,一般就不需要自己写深拷贝,默认浅拷贝就可以//赋值运算符重载(深拷贝)// lt1=lt2list<T>& operator=(list<T> lt){swap(lt);return *this;}void swap(list<T>& lt){std::swap(_head, lt._head);std::swap(_size, lt._size);}// 析构~list(){clear();delete _head;_head = nullptr;}// 2.push_back() //void push_back(const T& x)//{//	Node* tmp = new Node(x);//	Node* tail = _head->_prev;// 因为要尾插,所以保存好尾节点!//	tail->_next = tmp;//	tmp->_prev = tail;//	tmp->_next = _head;//	_head->_prev = tmp;//}// 头插void push_front(const T& x){insert(begin(), x);}// 尾插void push_back(const T& x){insert(end(), x);}// 头删void pop_front(){erase(begin());}// 尾删void pop_back(){erase(--end());}// 3.insertvoid insert(iterator pos, const T& val)//在pos位置之前插入val{//先用一个指针保存pos的位置!Node* cur = pos._node;//创建一个新的节点newnode来接受val的值Node* newnode = new Node(val);//再保存pos位置前一个方便newnode插入!Node* prev = cur->_prev;//prev newnode cur三者之间的交换newnode->_prev = prev;prev->_next = newnode;newnode->_next = cur;cur->_prev = newnode;}iterator erase(iterator pos){Node* cur = pos._node;Node* prev = cur->_prev;Node* next = cur->_next;prev->_next = next;next->_prev = prev;delete cur;// 我们delete cur之后,原来的pos迭代器指针也就消失了,但是我们为什么必须要返回一个:迭代器指针?return iterator(next);// 因为删除的数据是有不确定性的,万一要删除偶数或者后面有其他的用途,我们没有原来pos的位置,我们如何再找到其他的数据呢?}void clear(){iterator it = begin();while (it != end()){it = erase(it);//因为erase会返回要删除结点的下一个位置,所以要用iterator类型的it接受!}}//size_t size()const//{//	size_t count = 0;//	while (_head->_next != _head)//	{//		_head = _head->_next;// 因为_head是不能被修改的!!!,所以要创建一个临时指针来指向_head//		count++;//	}//	return count;//}size_t size()const{size_t count = 0;Node* cur = _head;while (cur->_next != _head){cur = cur->_next;count++;}return count;}bool empty(){return (_head->_next == _head);}private:Node* _head;size_t _size;};

好了,今天的分享就到这里了
如果对你有帮助,记得点赞👍+关注哦!
我的主页还有其他文章,欢迎学习指点。关注我,让我们一起学习,一起成长吧!
在这里插入图片描述


http://www.mrgr.cn/p/44571475

相关文章

华为云Stack8.3面向香港正式发布,六大亮点激发云上跃迁

近日,在华为云香港峰会2024上,华为混合云副总裁胡玉海面向香港市场发布华为云Stack8.3,提供110+本地云服务和六大亮点,帮助中国香港政企持续提升用云深度,激发业务创新。本文分享自华为云社区《华为云Stack8.3面向香港正式发布,六大亮点激发云上跃迁》,作者:华为云头条…

【AIGC调研系列】Bunny-Llama-3-8B-V与其他多模态大模型相比的优劣

Bunny-Llama-3-8B-V作为基于Llama-3的多模态大模型&#xff0c;其优势主要体现在以下几个方面&#xff1a; 性能超越其他模型&#xff1a;根据我搜索到的资料&#xff0c;Bunny-Llama-3-8B-V在多个主流Benchmark上表现良好&#xff0c;超越了LLaVA-7B、LLaVA-13B、Mini-Gemini…

抽象的代理模式1.0版本

前言&#xff1a; 在阅读Spring Security官方文档时&#xff0c;里面设计到了一种设计模式——代理模式Proxy 众里寻她千百度&#xff0c;蓦然回首&#xff0c;那人却在灯火阑珊处 开始 在之前的文章里陈述了一个观点——编程语言和语言没有区别 现看看我们日常生活中的代理…

怎么设置 idea terminal 窗口的编码格式

1 修改Terminal 窗口为 Git bash 窗口 打开 settings 设置界面&#xff0c;选择 Tools 中的 Terminal (File -> settings -> Tools -> Terminal) 修改 Shell path 为你的 Git bash 安装路径&#xff0c;我的在 C:\my_software\java\Git\bin\bash.exe 2 解决中文显示…

python r代表什么意思

r/R&#xff0c;即raw的缩写&#xff0c;意思是未经加工的&#xff1b;自然状态的&#xff1b;未经处理的&#xff1b;未经分析的&#xff1b;原始的。 在Python中r/R表示非转义的原始字符串。与普通字符相比&#xff0c;其他相对特殊的字符&#xff0c;其中可能包含转义字符&…

添加阿里云yum源

添加阿里云yum源 要添加阿里云的 yum 源&#xff0c;可以执行以下步骤&#xff1a; 首先&#xff0c;备份你的现有 yum 源配置文件&#xff0c;以防止意外更改&#xff1a; sudo cp /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backup然后&#xf…

探讨mfc100u.dll丢失的解决方法,修复mfc100u.dll有效方法解析

mfc100u.dll丢失是一个比较常见的情况&#xff0c;由于你电脑的各种操作&#xff0c;是有可能引起dll文件的缺失的&#xff0c;而mfc100u.dll就是其中的一个重要的dll文件&#xff0c;它的确实严重的话是会导致程序打不开&#xff0c;系统错误的。今天我们就来给大家科普一下mf…

【白盒测试】单元测试的理论基础及用例设计技术(6种)详解

目录 &#x1f31e;前言 &#x1f3de;️1. 单元测试的理论基础 &#x1f30a;1.1 单元测试是什么 &#x1f30a;1.2 单元测试的好处 &#x1f30a;1.3 单元测试的要求 &#x1f30a;1.4 测试框架-Junit4的介绍 &#x1f30a;1.5 单元测试为什么要mock &#x1f3de;️…

【产品经理修炼之道】- 从需求到功能的转化过程

产品经理的最大作用是将需求转化为产品或者功能&#xff0c;从需求到功能&#xff0c;会经历哪些过程&#xff1f;本文总结了从需求到功能的转化过程&#xff0c;希望对你进一步了解有所帮助。 “大部分的产品经理特别是数字化产品经理其核心价值就是如何去解决如何把需求转化为…

韩国机器人公司Rainbow Robotics推出RB-Y1轮式双臂机器人

文 | BFT机器人 近日&#xff0c;韩国机器人领域的佼佼者Rainbow Robotics揭开了RB-Y1移动机器人的神秘面纱&#xff0c;这款机器人以其创新的设计和卓越的功能引起了业界的广泛关注。与此同时&#xff0c;Rainbow Robotics还携手舍弗勒集团&#xff08;提供汽车、工业技术服务…

[转帖]历代x86架构IPC提升及12代酷睿与至强洋垃圾的简单对比

https://www.bilibili.com/read/cv16170718/ 好奇查找了一下历代x86架构CPU的IPC性能提升,发现知乎用户@MebiuW已经总结了10代酷睿和ZEN3之前历代架构的情况,于是摘录下来。IPC指的是每个周期CPU核心处理的指令数,频率代表CPU核心每1秒钟计算多少个周期,二者的乘积就是CPU…

浅谈菊风实时音视频 (RTC)与实时操作系统 (RTOS) 在智能硬件领域应用

近年来&#xff0c;菊风通过实时音视频赋能智能手表、智能门禁、智能门锁/门铃、智能眼镜等数十种智能硬件&#xff0c;与一众合作伙伴共同探索在IoT智能硬件领域的不同场景应用&#xff0c;积累了丰富的实践经验。在智能硬件中&#xff0c;RTOS因其轻量化的系统内核&#xff0…

Docker 的数据管理 端口映射 容器互联 镜像的创建

目录 概念 概念 管理 Docker 容器中数据主要有两种方式&#xff1a;数据卷&#xff08;Data Volumes&#xff09;和数据卷容器&#xff08;DataVolumes Containers&#xff09;。总结&#xff1a;因为容器数据是临时保存的为了安全&#xff0c;就要让数据保持持久化。 1&#…

.NET 个人博客-添加RSS订阅功能

个人博客-添加RSS订阅功能 前言 个人博客系列已经完成了 留言板文章归档推荐文章优化推荐文章排序 博客地址 然后博客开源的原作者也是百忙之中添加了一个名为RSS订阅的功能&#xff0c;那么我就来简述一下这个功能是干嘛的&#xff0c;然后照葫芦画瓢实现一下。 RSS简述…

专利视角下的量子竞赛:《2024全球专利格局白皮书》

2024年1月&#xff0c;欧洲量子产业联盟&#xff08;QuIC&#xff09;发布了题为《全球量子技术专利格局描述》的综合白皮书。 该文件以透明的视角展示了当今的知识产权格局&#xff0c;包括知识产权持有人的地理分布。该文件由 QuIC 知识产权&#xff08;IP&#xff09;与贸易…

VMware配置centos虚拟机实现内网互通

VMware配置centos虚拟机实现内网互通 一、安装无桌面模式 环境说明&#xff1a; VMWare版本&#xff1a;VMware Workstation 17 Pro Centos版本&#xff1a;CentOS-7.9-x86_64-DVD-2009.iso 一键下载本文资源包 1. 安装虚拟机 下面是创建具体步骤,其中需要注意的是&#xff1…

42. UE5 RPG 实现火球术伤害

上一篇&#xff0c;我们解决了火球术于物体碰撞的问题&#xff0c;现在火球术能够正确的和攻击目标产生碰撞。接下来&#xff0c;我们要实现火球术的伤害功能&#xff0c;在火球术击中目标后&#xff0c;给目标造成伤害。 实现伤害功能的思路是给技能一个GameplayEffect&#x…

OBMysql4.3.0.1的升级与备份恢复

OBMysql4.3.0.1的升级与备份恢复吐槽 OBMysql开源版本的升级路线图太操蛋了. 我最开始安装的是 4.2.2.0 的版本 耗费了一个多小时想升级 4.3.0.1 发现总是报错 自己以为是自己的人品有问题. 结果尝试升级 4.2.2.1 就可以升级 现在明显是 升级路线有问题. [ERROR] fail to get …

Redis的性能与CPU主频的关系

Redis的性能与CPU主频的关系背景 最近想验证一下不同主频下 CPU性能相关内容. 本来想开启一个高主频的 虚拟机 但是经过自己一翻 百度 bing 发现大部分资料告知我 虚拟机没法调整主频. WTF 只能用物理机来进行相关的处理. 经过十分钟的努力 得出结论 Redis的性能在相同架构, …

赋能智慧校园!A3D数字孪生可视化,轻量又高效!

放假之后&#xff0c;学生们会逐步返学&#xff0c;大量人员出入校园&#xff0c;安全更是不容忽视&#xff0c;如何在短时间内对大批人员及设施进行智能监管&#xff1f;数字化转型是关键手段&#xff0c;我们可以融合线上线下数据&#xff0c;搭建3D立体的智慧校园&#xff0…