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

“现在的AI就像1880年的笨重工厂!”微软CSO斯坦福泼冷水:别急着造神

“现在的AI就像1880年的笨重工厂!”微软CSO斯坦福泼冷水:别急着造神

大模型仍未对上商业的齿轮? 编译 | 王启隆 来源 | youtu.be/aWqfH0aSGKI 出品丨AI 科技大本营(ID:rgznai100) 现在的硅谷,空气里都飘着一股“再不上车就晚了”的焦躁感。 最近 OpenClaw 风头正旺,强势登顶 GitHub,终结了 React 神话,许多人更是觉得“AI 自己干活赚钱”的日子就在明天了。 特别是在斯坦福商学院(GSB)这种地方,台下坐着的都是成天琢磨怎么用下一个技术风口搞个独角兽出来的狠人。 微软的首席科学官(CSO)Eric Horvitz 被请到了这个几乎全美最想用 AI 变现的礼堂里。作为从上世纪 80 年代就开始搞 AI 的绝对老炮、也是微软技术底座的“扫地僧”,这位老哥并没有顺着台下的胃口,去吹捧下个月大模型又要颠覆什么行业,而是兜头给大家浇了一盆带点学术味的冷水。 他讲了一个挺有画面感的比喻:大家都在聊

By Ne0inhk
Godot被AI代码“围攻”!维护者崩溃发声:“不知道还能坚持多久”

Godot被AI代码“围攻”!维护者崩溃发声:“不知道还能坚持多久”

整理 | 郑丽媛 出品 | ZEEKLOG(ID:ZEEKLOGnews) 当大模型能在几秒钟内生成一段“看起来像那么回事”的补丁时,开源社区却开始付出另一种代价。 最近,开源游戏引擎 Godot 的核心维护团队公开吐槽:他们正被大量“AI 生成的低质量代码”淹没。那些代码往往结构完整、注释齐全、描述洋洋洒洒,但真正的问题是——提交者可能并不理解自己交上来的内容。 这件事,并不是简单的“有人偷懒用 AI 写代码”。它正在触及开源协作最核心的东西:信任。 一场悄无声息的“AI 洪水” 事情的导火索来自一条 Bluesky 讨论帖。 Godot 主要维护者之一、同时也是 Godot 商业支持公司 W4 Games 联合创始人的 Rémi Verschelde 表示,所谓的“AI slop”

By Ne0inhk
诺奖得主辛顿最新访谈:1 万个 AI 可以瞬间共享同一份“灵魂”,这就是为什么人类注定被超越

诺奖得主辛顿最新访谈:1 万个 AI 可以瞬间共享同一份“灵魂”,这就是为什么人类注定被超越

当宇宙级的“嘴炮”遇到降维打击。 编译 | 王启隆 来源 | youtu.be/l6ZcFa8pybE 出品丨AI 科技大本营(ID:rgznai100) 打开最新一期知名播客 StarTalk 的 YouTube 评论区,最高赞的一条留言是这样写的: “我长这么大,第一次看到尼尔·德葛司·泰森(Neil deGrasse Tyson)在一档节目里几乎全程闭嘴,像个手足无措的小学生一样乖乖听讲。” 作为全美最知名的天体物理学家,泰森平时的画风是充满激情、喋喋不休、用宇宙的宏大来震撼嘉宾。但这一次,坐在他对面的那位满头银发、带着温和英音的英国老人,仅仅用最平淡的语气,就让整个演播室陷入了数次令人窒息的沉默。 这位老人是 Geoffrey Hinton。深度学习三巨头之一,2024 年诺贝尔物理学奖得主,被公认为“AI 教父”。 对经常阅读 Hinton 演讲的我来说,这也是比较新奇的一幕—

By Ne0inhk
48小时“烧光”56万!三人创业团队濒临破产,仅因Gemini API密钥被盗:“AI账单远超我们的银行余额”

48小时“烧光”56万!三人创业团队濒临破产,仅因Gemini API密钥被盗:“AI账单远超我们的银行余额”

整理 | 苏宓 出品 | ZEEKLOG(ID:ZEEKLOGnews) 「仅过了 48 小时,一笔 8.2 万美元的天价费用凭空出现,较这家小型初创公司的正常月费暴涨近 46000%。」 这不是假设的虚幻故事,而是一家墨西哥初创公司正在经历的真实危机。 近日,一位名为 RatonVaquero 的开发者在 Reddit 发帖求助称,由于他的 Gemini API 密钥被盗用,原本每月仅约 180 美元(约 1242 元)的费用,在短短 48 小时内暴涨到 82,314.44 美元(约 56.8 万元)。对于这家只有三名开发者的小型创业团队来说,这笔突如其来的账单,几乎等同于灭顶之灾。 “我现在整个人都处在震惊和恐慌之中。”RatonVaquero

By Ne0inhk