C++:二叉搜索树

C++:二叉搜索树
 Hello大家好! 很高兴与大家见面! 给生活添点快乐,开始今天的编程之路。



我的博客:<但愿.

我的专栏:C语言题目精讲算法与数据结构C++

欢迎点赞,关注

目录

  一  ⼆叉搜索树的性能分析和概念

      1.1⼆叉搜索树的概念

      1.2⼆叉搜索树的性能分析

 二   ⼆叉搜索树增删查的实现,这里不支持修改(修改一个就可能不是二叉搜索树的结构了)

      2.1⼆叉搜索树基本结构的实现

                2.1.1节点的定义(由于后面实现⼆叉搜索树要访问节点成员所以这里使用struct定义(默认是公有))

                2.1.2默认成员函数

                        2.1.2.1⼆叉搜索树的基本结构

                        2.1.2.2⼆叉搜索树的默认构造

                        2.1.2.3⼆叉搜索树的拷贝构造

                        2.1.2.4⼆叉搜索树的赋值重载

                        2.1.2.5⼆叉搜索树的析构

        2.2⼆叉搜索树的插⼊

                 2.2.1⼆叉搜索树的插⼊整体思路

                 2.2.2⼆叉搜索树的插⼊代码实现

        2.3⼆叉搜索树的查找

                 2.3.1⼆叉搜索树的查找整体思路

                 2.3.2⼆叉搜索树的查找代码实现

       2.4⼆叉搜索树的删除(最难)

                 2.4.1⼆叉搜索树删除的整体思路

                 2.4.2⼆叉搜索树删除的代码实现

      2.5⼆叉搜索树的排序(使用中序遍历)

  三  二叉搜索树增删查排序的整体代码实现

  四 key和(key,value)结构

      4.1key和(key,value)结构搜索二叉树的定义

      4.2(key,value)结构搜索二叉树的实现

  五  ⼆叉搜索树key和key/value使⽤场景

      5.1⼆叉搜索树key使⽤场景

      5.2⼆叉搜索树key/value使⽤场景

                

一  ⼆叉搜索树的性能分析和概念

1.1⼆叉搜索树的概念

⼆叉搜索树⼜称⼆叉排序树,它或者是⼀棵空树,是一种具有以下性质的⼆叉树:

•若它的左⼦树不为空,则左⼦树上所有结点的值都⼩于等于根结点的值•若它的右⼦树不为空,则右⼦树上所有结点的值都⼤于等于根结点的值• 它的左右⼦树都是⼆叉搜索树•⼆叉搜索树中可以⽀持插⼊相等的值,也可以不⽀持插⼊相等的值,具体看使⽤场景定义,例如:map/set/multimap/multiset系列容器底层就是⼆叉搜索树,其中map/set不⽀持插⼊相等值,multimap/multiset⽀持插⼊相等值

1.2⼆叉搜索树的性能分析

由于它本身自己和它的子树都是⼆叉搜索树,这里观察上面的特点,不难发现:⼆叉搜索树的中序遍历一定是升序序列(所以⼆叉搜索树⼜称⼆叉排序树)。

⼆叉搜索树顾名思义,其只有由于搜索(或者搜索效率高),下面我们来分析其最优情况和最插情况

最优情况下,⼆叉搜索树为完全⼆叉树(或者接近完全⼆叉树),其⾼度为: log2 N最差情况下,⼆叉搜索树退化为单⽀树(或者类似单⽀),其⾼度为: N所以综合⽽⾔⼆叉搜索树增删查改时间复杂度为: O(N)那么这样的效率显然是⽆法满⾜我们需求的,必须将二叉搜索树优化即⼆叉搜索树的变形:平衡⼆叉搜索树AVL树和红⿊树,才能适⽤于我们在内存中存储和搜索数据。那有的人说使用另⼆分查找也可以实现 O(log2 N) 级别的查找效率为啥还有将二叉搜索树变形外。需要说明的是,⼆分查找也可以实现 O(log2 N) 级别的查找效率,但是⼆分查找有两⼤缺陷:1. 需要存储在⽀持下标随机访问的结构中,并且有序。2. 插⼊和删除数据效率很低,因为存储在下标随机访问的结构中,插⼊和删除数据⼀般需要挪动数据。这⾥也就体现出了平衡⼆叉搜索树的价值。

二   ⼆叉搜索树增删查的实现,这里不支持修改(修改一个就可能不是二叉搜索树的结构了)

2.1⼆叉搜索树基本结构的实现

2.1.1节点的定义(由于后面实现⼆叉搜索树要访问节点成员所以这里使用struct定义(默认是公有))
template<class K> struct BSTreeNode { BSTreeNode* _left; BSTreeNode* _right; K _key; BSTreeNode(const K& key) :_left(nullptr) , _right(nullptr) , _key(key) { } };
2.1.2默认成员函数

由于⼆叉搜索树涉及到节点资源的开辟、拷贝、删除,所以这里我们需要自己写拷贝构造,赋值重载,析构函数,默认构造一般不满足我们的需求所以看情况是否要写,而这里由于我们需要写拷贝构造(一种特殊的默认构造)所以这里一定要手写默认规则(可以缺省值)

2.1.2.1⼆叉搜索树的基本结构
template<class K> class BSTree { typedef BSTreeNode<K> Node; private: Node* _root=nullptr; };
2.1.2.2⼆叉搜索树的默认构造
//默认函数两种方法 //方法1-前面定义有缺省值 /*BSTree() { }*/ //方法二 BSTree() = default;
2.1.2.3⼆叉搜索树的拷贝构造

前面的链式结构的树的拷贝需要递归完成(即使用了递归),所以访问私有成员(在类外面调用该函数会出错),而这个函数的实现过程仅供内部使用所以这里应该定义一个私有函数,在定义一个公有函数来调用对应的私有函数。

public: //拷贝构造 BSTree(const BSTree<K>& t) { void _Copy(t._root); } private: Node* Copy(Node* root) { if (root == nullptr) { return nullptr; } Node* copy = root; Node* copy->_left = Copy(root->_left); Node* copy->right = Copy(root->right); return copy; }
2.1.2.4⼆叉搜索树的赋值重载
 pubilc: //赋值重载 BStree<K>& operator(BSTree<K> t) { swap(_root, t._root); return *this; }
2.1.2.5⼆叉搜索树的析构

前面的链式结构的树的拷贝需要递归完成(即使用了递归),所以访问私有成员(在类外面调用该函数会出错),而这个函数的实现过程仅供内部使用所以这里应该定义一个私有函数,在定义一个公有函数来调用对应的私有函数。

public: //析构函数 ~BSTree() { Destroy(_root); } private: void Destroy(Node* root) { if (root == nullptr) { return; } Destroy(root->_left); Destroy(root->_right); delete root; }

2.2⼆叉搜索树的插⼊

2.2.1⼆叉搜索树的插⼊整体思路
1. 树为空,则直接新增结点,赋值给root指针2. 树不空,按⼆叉搜索树性质,插⼊值⽐当前结点⼤往右⾛,插⼊值⽐当前结点⼩往左⾛,找到空位置,插⼊新结点。注意:如果⽀持插⼊相等的值,插⼊值跟当前结点相等的值可以往右⾛,也可以往左⾛,找到空位置,插⼊新结点。(要注意的是要保持逻辑⼀致性,插⼊相等的值不要⼀会往右⾛,⼀会往左⾛)【这里我们不讲,一般不用】
2.2.2⼆叉搜索树的插⼊代码实现
//1插入数据-不支持插入已有数据 bool Insert(const K& key) { //树为空,则直接新增结点,赋值给root指针 if (_root == nullptr) { _root = new Node(key); return true; } //由于二插搜索树的特征,所以要先找到插入的位置; //由于要改变节点直接之间的指向关系,所以要定义两个变量 // 一个变量用于遍历,一个用于储存要插入位置的父节点,方便改变节点之间的指向关系 Node* cur = _root;//用于遍历 Node* parent = nullptr;//用于储存要插入位置的父节点 while(cur) { if (cur->_key < key)//插⼊值⽐当前结点⼤往右⾛ { parent = cur; cur = cur->_right; } else if (cur->_key > key)//插⼊值⽐当前结点⼩往左⾛ { parent = cur; cur = cur->_left; } else//找到插入位置 { return false; } }

2.3⼆叉搜索树的查找

2.3.1⼆叉搜索树的查找整体思路
1. 从根开始⽐较,查找x,x⽐根的值⼤则往右边⾛查找,x⽐根值⼩则往左边⾛查找。2. 最多查找⾼度次,⾛到到空,还没找到,这个值不存在。3. 如果不⽀持插⼊相等的值,找到x即可返回注意:如果⽀持插⼊相等的值,意味着有多个x存在,⼀般要求查找中序的第⼀个x。
2.3.2⼆叉搜索树的查找代码实现
//2查找数据 bool Find(const K& key) { Node* cur = _root; while (cur) { if (cur->_key < key)//从根开始⽐较,查找x,x⽐根的值⼤则往右边⾛查找 { cur = cur->_right; } else if(cur->_key > key)//x⽐根值⼩则往左边⾛查找 { cur = cur->_left; } else { return true;//找到返回 } } return false;//⾛到到空,还没找到,这个值不存在。 }

2.4⼆叉搜索树的删除(最难)

2.4.1⼆叉搜索树删除的整体思路
⾸先查找元素是否在⼆叉搜索树中,如果不存在,则返回false。如果查找元素存在则分以下四种情况:(假设要删除的结点为x)情况一:问题:要删除结点x左右孩⼦均为空解决办法:把x结点的⽗亲对应孩⼦指针指向空,直接删除x结点情况二:问题:要删除的结点x左孩⼦位空,右孩⼦结点不为空解决办法:把x结点的⽗亲对应孩⼦指针指向x的右孩⼦,在直接删除x结点情况三:问题:要删除的结点x右孩⼦位空,左孩⼦结点不为空解决办法:把x结点的⽗亲对应孩⼦指针指向x的左孩⼦,在直接删除x结点情况四(最复杂)问题:要删除的结点x左右孩⼦结点均不为空。解决办法: ⽆法直接删除x结点,因为x的两个孩⼦⽆处安放,只能⽤替换法删除。找x左⼦树的值最⼤结点R(最右结点)或者x右⼦树的值最⼩结点R(最左结点)替代x,因为这两个结点中任意⼀个,放到x的位置,都满⾜⼆叉搜索树的规则。替代x的意思就是x和R的两个结点的值交换,转⽽变成删除R结点,R结点符合情况2或情况3,可以直接删除。总结:情况1可以当成2或者3处理,效果是⼀样的。这些情况就好比一个人生孩子,如果只是生了一个孩子甚至没生孩子可以找父母帮忙,如果生了两个孩子父母忙不过来怎么办呢?这时候可以找一个保姆的思路。 
 
2.4.2⼆叉搜索树删除的代码实现
//删除数据 bool Erase(const K& key) { //先找到要删除的节点的位置 Node* parent = nullptr; Node* cur = _root; while (cur) { if (cur->_key < key) { parent = cur; cur = cur->_right; } else if (cur->_key > key) { parent = cur; cur = cur->_left; } else { //找到了,准备删除数据 //分为三种情况 //1左孩子为空,左右孩子为空可以一起处理 if (cur->_left == nullptr) { if (cur == _root)//要删除的节点就是根节点,直接改变根节点的指向即可 { _root = cur->_right; } else { //要删除的节点不是根节点,要判断要删除的节点 //是其父节点的左孩子还是右孩子,方便改变节点直接的指向关系 if (cur == parent->_left) { parent->_left= cur->_right; } else { parent->_right = cur->_right; } } delete cur;//删除要删除的结点 } else if (cur->_right == nullptr)//右孩子为空 { if (cur == _root)////要删除的节点就是根节点,直接改变根节点的指向即可 { _root = cur->_left; } else { //要删除的节点不是根节点,要判断要删除的节点 //是其父节点的左孩子还是右孩子,方便改变节点直接的指向关系 if (cur == parent->_left) { parent->_left = cur->_left; } else { parent->_right = cur->_left; } } delete cur;//删除要删除的结点 } else {//有两孩子-两种处理方法这里使用找有子数中最小值节点 Node* pMinRight = cur; Node* minRight = cur->_right; //找到右子树中值最小的节点 while (minRight->_left) { pMinRight = minRight; minRight = minRight->_left; } swap(minRight->_key, pMinRight->_key); if (pMinRight->_left == minRight) { pMinRight->_left = minRight->_right; } else { pMinRight->_right = minRight->_right; } delete minRight; } return true; } } return true; }

2.5⼆叉搜索树的排序(使用中序遍历)

由于使用到了中序遍历即使用了递归,所以访问私有成员(在类外面调用该函数会出错),所以这里应该定义一个私有函数,在定义一个公有函数来调用对应的私有函数。

代码实现:

 //排序-利用其特点这里可以使用中序遍历 //由于使用到了中序遍历即使用了递归,所以访问私有成员(在类外面调用该函数回出错) void InOrder()//定义一个公有函数来调用对应的私有函数 { _InOrder(_root); cout << endl; } private: void _InOrder(Node* root)//定义一个私有函数 { if (root == nullptr) { return; } _InOrder(root->_left); cout << root->_key << " "; _InOrder(root->_right); }

三  二叉搜索树增删查排序的整体代码实现

namespace key { template<class K> struct BSTreeNode { BSTreeNode* _left; BSTreeNode* _right; K _key; BSTreeNode(const K& key) :_left(nullptr) , _right(nullptr) , _key(key) { } }; template<class K> class BSTree { typedef BSTreeNode<K> Node; public: //默认函数 /*BSTree() { }*/ BSTree() = default; //2拷贝构造 BSTree(const BSTree<K>& t) { void _Copy(t._root); } //赋值重载 BStree<K>& operator(BSTree<K> t) { swap(_root, t._root); return *this; } //析构函数 ~BSTree() { Destroy(_root); } // //1插入数据-不支持插入已有数据 bool Insert(const K& key) { //树为空,则直接新增结点,赋值给root指针 if (_root == nullptr) { _root = new Node(key); return true; } //由于二插搜索树的特征,所以要先找到插入的位置; //由于要改变节点直接之间的指向关系,所以要定义两个变量 // 一个变量用于遍历,一个用于储存要插入位置的父节点,方便改变节点之间的指向关系 Node* cur = _root;//用于遍历 Node* parent = nullptr;//用于储存要插入位置的父节点 while(cur) { if (cur->_key < key)//插⼊值⽐当前结点⼤往右⾛ { parent = cur; cur = cur->_right; } else if (cur->_key > key)//插⼊值⽐当前结点⼩往左⾛ { parent = cur; cur = cur->_left; } else//找到插入位置 { return false; } } cur = new Node(key); //要判断插入的是当前节点的左子节点还是右子节点 if (parent->_key < key) { parent->_right = cur; } else { parent->_left = cur; } return true; } //2查找数据 bool Find(const K& key) { Node* cur = _root; while (cur) { if (cur->_key < key)//从根开始⽐较,查找x,x⽐根的值⼤则往右边⾛查找 { cur = cur->_right; } else if(cur->_key > key)//x⽐根值⼩则往左边⾛查找 { cur = cur->_left; } else { return true;//找到返回 } } return false;//⾛到到空,还没找到,这个值不存在。 } //删除数据 bool Erase(const K& key) { //先找到要删除的节点的位置 Node* parent = nullptr; Node* cur = _root; while (cur) { if (cur->_key < key) { parent = cur; cur = cur->_right; } else if (cur->_key > key) { parent = cur; cur = cur->_left; } else { //找到了,准备删除数据 //分为三种情况 //1左孩子为空 if (cur->_left == nullptr) { if (cur == _root) { _root = cur->_right; } else { if (cur == parent->_left) { parent->_left= cur->_right; } else { parent->_right = cur->_right; } } delete cur; } else if (cur->_right == nullptr)//右孩子为空 { if (cur == _root) { _root = cur->_left; } else { if (cur == parent->_left) { parent->_left = cur->_left; } else { parent->_right = cur->_left; } } delete cur; } else {//有两孩子-两种处理方法这里使用找有子数中最小值节点 Node* pMinRight = cur; Node* minRight = cur->_right; //找到右子树中值最小的节点 while (minRight->_left) { pMinRight = minRight; minRight = minRight->_left; } swap(minRight->_key, pMinRight->_key); if (pMinRight->_left == minRight) { pMinRight->_left = minRight->_right; } else { pMinRight->_right = minRight->_right; } delete minRight; } return true; } } return true; } //排序-利用其特点这里可以使用中序遍历 //由于使用到了中序遍历即使用了递归,所以访问私有成员(在类外面调用该函数回出错) void InOrder()//定义一个公有函数来调用对应的私有函数 { _InOrder(_root); cout << endl; } private: void _InOrder(Node* root)//定义一个私有函数 { if (root == nullptr) { return; } _InOrder(root->_left); cout << root->_key << " "; _InOrder(root->_right); } void Destroy(Node* root) { if (root == nullptr) { return; } Destroy(root->_left); Destroy(root->_right); delete root; } Node* Copy(Node* root) { if (root == nullptr) { return nullptr; } Node* copy = root; Node* copy->_left = Copy(root->_left); Node* copy->right = Copy(root->right); return copy; } private: Node* _root=nullptr; }; }

四 key和(key,value)结构

4.1key和(key,value)结构搜索二叉树的定义

上面我们实现的二叉搜索树,节点中只储存了一个值(key),key,称为关键码,关键码即为需要搜索到的值,使用这种搜索二叉树只需要判断key是否存在。key的搜索场景实现的⼆叉树搜索树⽀持增删查,但是不⽀持修改,修改key破坏搜索树结构了(性质)。

这里还有一种搜索二叉树:(key,value)结构的二叉搜索树:

每⼀个关键码还是key,但是每⼀个关键码都有与之对应的值value,value可以任意类型对象。所以树的结构中(结点)除了需要存储key还要存储对应的value,注意增/删/查还是以key为关键字在⼆叉搜索树的规则下操作,这里不支持修改key,支持修改value。

4.2(key,value)结构搜索二叉树的实现

namespace key_value { template<class K, class V> struct BSTreeNode { BSTreeNode<K, V>* _left; BSTreeNode<K, V>* _right; K _key; // 中文 V _value; // 英文 BSTreeNode(const K& key, const V& value) :_left(nullptr) , _right(nullptr) , _key(key) , _value(value) {} }; template<class K, class V> class BSTree { typedef BSTreeNode<K, V> Node; public: // 强制生成 BSTree() = default; // BSTree(const BSTree& t) BSTree(const BSTree<K, V>& t) { _root = Copy(t._root); } // t1 = t3 // BSTree& operator=(BSTree t) BSTree<K, V>& operator=(BSTree<K, V> t) { swap(_root, t._root); return *this; } ~BSTree() { Destory(_root); _root = nullptr; } bool Insert(const K& key, const V& value) { if (_root == nullptr) { _root = new Node(key, value); return true; } Node* parent = nullptr; Node* cur = _root; while (cur) { if (cur->_key < key) { parent = cur; cur = cur->_right; } else if (cur->_key > key) { parent = cur; cur = cur->_left; } else { return false; } } cur = new Node(key, value); if (parent->_key < key) { parent->_right = cur; } else { parent->_left = cur; } return true; } Node* Find(const K& key) { Node* cur = _root; while (cur) { if (cur->_key < key) { cur = cur->_right; } else if (cur->_key > key) { cur = cur->_left; } else { return cur; } } return nullptr; } bool Erase(const K& key) { Node* parent = nullptr; Node* cur = _root; while (cur) { if (cur->_key < key) { parent = cur; cur = cur->_right; } else if (cur->_key > key) { parent = cur; cur = cur->_left; } else { // 准备删除 if (cur->_left == nullptr) { //if (parent == nullptr) if (cur == _root) { _root = cur->_right; } else { if (cur == parent->_left) { parent->_left = cur->_right; } else { parent->_right = cur->_right; } } delete cur; } else if (cur->_right == nullptr) { if (cur == _root) { _root = cur->_left; } else { if (cur == parent->_left) { parent->_left = cur->_left; } else { parent->_right = cur->_left; } } delete cur; } else // 两个孩子 { Node* pMinRight = cur; Node* minRight = cur->_right; while (minRight->_left) { pMinRight = minRight; minRight = minRight->_left; } swap(cur->_key, minRight->_key); if (pMinRight->_left == minRight) { pMinRight->_left = minRight->_right; } else { pMinRight->_right = minRight->_right; } delete minRight; } return true; } } return false; } void InOrder() { _InOrder(_root); cout << endl; } private: void _InOrder(Node* root) { if (root == nullptr) { return; } _InOrder(root->_left); cout << root->_key << ":" << root->_value << endl; _InOrder(root->_right); } void Destory(Node* root) { if (root == nullptr) { return; } Destory(root->_left); Destory(root->_right); delete root; } Node* Copy(Node* root) { if (root == nullptr) return nullptr; Node* copy = new Node(root->_key, root->_value); copy->_left = Copy(root->_left); copy->_right = Copy(root->_right); return copy; } private: Node* _root = nullptr; }; }

五  ⼆叉搜索树key和key/value使⽤场景

5.1⼆叉搜索树key使⽤场景

场景1:⼩区⽆⼈值守⻋库,⼩区⻋库买了⻋位的业主⻋才能进⼩区,那么物业会把买了⻋位的业主的⻋牌号录⼊后台系统,⻋辆进⼊时扫描⻋牌在不在系统中,在则抬杆,不在则提⽰⾮本⼩区⻋辆,⽆法进⼊。场景2:检查⼀篇英⽂ 章单词拼写是否正确,将词库中所有单词放⼊⼆叉搜索树,读取⽂章中的单词,查找是否在⼆叉搜索树中,不在则波浪线标红提⽰

5.2⼆叉搜索树key/value使⽤场景

场景1:简单中英互译字典,树的结构中(结点)存储key(英⽂)和vlaue(中⽂),搜索时输⼊英⽂,则同时查找到了英⽂对应的中⽂。场景2:商场⽆⼈值守⻋库,⼊⼝进场时扫描⻋牌,记录⻋牌和⼊场时间,出⼝离场时,扫描⻋牌,查找⼊场时间,⽤当前时间-⼊场时间计算出停⻋时⻓,计算出停⻋费⽤,缴费后抬杆,⻋辆离场。场景3:统计⼀篇⽂章中单词出现的次数,读取⼀个单词,查找单词是否存在,不存在这个说明第⼀次出现,(单词,1),单词存在,则++单词对应的次数。本篇文章就到此结束,欢迎大家订阅我的专栏,欢迎大家指正,希望有所能帮到读者更好理解C++相关知识 ,觉得有帮助的还请三联支持一下~后续会不断更新C/C++相关知识,我们下期再见。

Read more

实测Gemini Pro:谷歌王牌AI,到底能帮我们解决多少实际问题?

实测Gemini Pro:谷歌王牌AI,到底能帮我们解决多少实际问题?

🔥草莓熊Lotso:个人主页 ❄️个人专栏: 《C++知识分享》《Linux 入门到实践:零基础也能懂》 ✨生活是默默的坚持,毅力是永久的享受! 🎬 博主简介: 文章目录 * 前言: * 一、核心亮点实测:不止是“多模态”,更是“真全能” * 1. 多模态处理:能“看、听、读、写”,还能“联动协作” * 2. 推理能力:复杂问题“会拆解、会纠错”,堪比专业助手 * 3. 代码能力:开发者的“全能帮手”,新手也能轻松上手 * 二、真实应用场景:这些领域,已经在用它提效了 * 1. 科研领域:帮研究员“节省时间”,专注核心工作 * 2. 内容创作:

By Ne0inhk

我让 AI 帮我写公众号,它连发布都包了

最近在折腾 AI Agent,我突发奇想:既然 AI 能帮我写文章,由于什么它不能顺手帮我发出去? 结果证明:它真能。 这篇文章就是想聊聊我是怎么把"写"和"发"打通的,以及这里面的核心——Skill(技能)。 啥是 Skill? 先不说虚的。现在的 AI(像 Claude 4.5、Gemini 3 这些)很强,但它们有个毛病:眼高手低。 它们像个读过万卷书的实习生,你问什么它都知道,但你让它"把这个文件传到后台去",它就傻眼了——它没手,也不懂具体怎么操作。 Skill 就是给 AI 找回双手,

By Ne0inhk

Mac Mini部署OpenClaw实战指南:打造7×24小时全天候AI数字管家

引言:当Mac Mini遇上OpenClaw 2026年,开源AI智能体OpenClaw已成为技术圈最炙手可热的话题——短短数月斩获超16.5万GitHub星标,社区成员突破8900人。这款原名Clawdbot/Moltbot的开源框架,能将大语言模型的推理能力转化为真实的系统操作力:管理文件、执行终端指令、调用本地应用、连接WhatsApp/Telegram/iMessage等数十种通信渠道。 而Mac Mini,特别是搭载M4芯片的最新款,凭借其极致能效、静音运行、统一内存架构三大特质,成为运行OpenClaw的公认理想宿主。本文将手把手教你将一台Mac Mini打造成全天候在线的AI数字管家,同时深入探讨安全加固、性能优化与生产级部署的最佳实践。 一、为什么是Mac Mini + OpenClaw? 1.1 OpenClaw:本地AI智能体的核心价值 OpenClaw并非简单的聊天机器人,而是一个具备系统级权限的自主执行框架: * 文件系统操作:创建、编辑、删除文件,整理目录结构 * 终端指令执行:运行Shell命令,调用系统工具 * 应用控制:操作浏览器、

By Ne0inhk
Trae IDE评测体验:通过 MCP Server - Figma AI Bridge 一键将 Figma 转为前端代码

Trae IDE评测体验:通过 MCP Server - Figma AI Bridge 一键将 Figma 转为前端代码

Trae IDE评测体验:通过 MCP Server - Figma AI Bridge 一键将 Figma 转为前端代码 在现代前端开发中,从设计稿到可用页面的交付往往需要大量重复劳动:切图、手写样式、布局调整……而借助 MCP Server - Figma AI Bridge,我们可以将 Figma 设计稿自动转换成整洁的 HTML/CSS/JS 代码,并立即生成可预览的网页。一键化、傻瓜式操作,让设计交付效率跃升。 先下载 Trae IDE,让我们一起开始吧! [立即免费获取 Trae]:https://trae.ai 演示环境 本文测试使用的系统环境如下: * Trae IDE 版本:0.

By Ne0inhk