【刷题记录】关于二叉树的OJ题

news2025/1/17 3:59:08

文章目录

  • 1.根据二叉树创建字符串
  • 2.二叉树的层序遍历
  • 3.二叉树的最近公共祖先
  • 4.二叉搜索树与双向链表
  • 5. 从前序与中序遍历序列构造二叉树
  • 6.二叉树的遍历

1.根据二叉树创建字符串

题目链接:606. 根据二叉树创建字符串 - 力扣(LeetCode)

题干:

给你二叉树的根节点 root ,请你采用前序遍历的方式,将二叉树转化为一个由括号和整数组成的字符串,返回构造出的字符串。

空节点使用一对空括号对 “()” 表示,转化后需要省略所有不影响字符串与原始二叉树之间的一对一映射关系的空括号对。

示例 1:

img
输入:root = [1,2,3,4]
输出:"1(2(4))(3)"
解释:初步转化后得到 "1(2(4)())(3()())" ,但省略所有不必要的空括号对后,字符串应该是"1(2(4))(3)" 。

示例 2:

img
输入:root = [1,2,3,null,4]
输出:"1(2()(4))(3)"
解释:和第一个示例类似,但是无法省略第一个空括号对,否则会破坏输入与输出一一映射的关系。

题目分析:

这是一道力扣简单题,显而易见使用先序遍历即可,但是这里需要注意到的点就是,用括号把子树括起来这一点,需要我们着重考虑一下。因为其中有的情况要省略空括号,有的情况不能省略。分析题目可知,当该节点左子树为nullptr,并且右子树有值时不能省略

代码实现:

class Solution {
public:
    string tree2str(TreeNode* root) {
		if(root == nullptr)//当树为空树的时候,遍历之后的结果是空字符串
            return string();
        string str;//创建字符串保存结果
        str += to_string(root->val);//线序遍历,首先把根节点的值放进str,这里需要把val的类型转换成string类型才能追加
        //先序遍历,首先走左子树,然后走右子树
        if(root->left)//左子树不为空时,先序遍历左子树,然后把左子树的字符串加上括号追加到str上
        {
            str += '(';
            str += tree2str(root->left);
            str += ')';
        }
        else if(root->right)//左子树为空,且右子树不为空时
        {
            //此时左子树的空括号不能省略
            str += "()";
        }
        if(root->right)//右子树不为空时,先序遍历右子树
        {
            str += '(';
            str += tree2str(root->right);
            str += ')';            
        }
        return str;
    }
};
image-20230506145824561

2.二叉树的层序遍历

题目链接:102. 二叉树的层序遍历 - 力扣(LeetCode)

题干:

给你二叉树的根节点 root ,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。

示例1

img
输入:root = [3,9,20,null,null,15,7]
输出:[[3],[9,20],[15,7]]

示例二

输入:root = [1]
输出:[[1]]

示例三

输入:root = []
输出:[]

题目分析:

二叉树的层序遍历,我们需要借助一下队列的数据结构,将每一层的节点放进队列中,然后需要在访问当前层节点的时候拿到当前节点的子节点,否则后面就找不到子节点了。所以思路就是首先非空节根节点入队列,然后每次出队列时,把该节点的子节点依次入队列,这要就能达到层序的目的,看起来很完美,但是注意一下题目示例,需要我们按照每层节点存放在一个单独的vector中,这就产生了一个问题:怎么判断当前节点是第几层的?仔细分析可知,队列中存放的节点只有两种可能:1.只有当前节点所在层数的节点;2.有当前节点所在层节点和下一层的部分节点,那我们采用一个变量存放当前层的节点个数,然后每当出一个节点就自减,直到为0时,队列中存放的节点个数就是下一层的节点个数,通过这个变量来控制存入第几个vector即可

代码实现:

class Solution {
public:
    vector<vector<int>> levelOrder(TreeNode* root) {
        vector<vector<int>> vv;
        queue<TreeNode*> q;
        int levelSize = 0;//存放当前层的节点个数
        if(root)
        {
            q.push(root);
            levelSize = 1;
        }
        while(!q.empty())
        {
            vector<int> v;//存放每一层的值,一层一层push_back到vv中
            while(levelSize--)
            {
                TreeNode* top = q.front();
                q.pop();
                v.push_back(top->val);
                if(top->left)
                    q.push(top->left);
                if(top->right)
                    q.push(top->right);
            }
            levelSize = q.size();
            vv.push_back(v);
        }
        return vv;
    }
};
image-20230506152450991

拓展与补充:107. 二叉树的层序遍历 II - 力扣(LeetCode)

这是上一道题的扩展,让我们自低向上层序遍历,这里有一个最简单的办法就是按照上述的代码走一遍,然后在return之前reverse一下即可。

3.二叉树的最近公共祖先

题目链接:236. 二叉树的最近公共祖先 - 力扣(LeetCode)

题干:

给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。

百度百科中最近公共祖先的定义为:“对于有根树 T 的两个节点 p、q,最近公共祖先表示为一个节点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节点也可以是它自己的祖先)。”

示例1

img

输入:root = [3,5,1,6,2,0,8,null,null,7,4], p = 5, q = 1
输出:3
解释:节点 5 和节点 1 的最近公共祖先是节点 3 。

示例2

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-CvrEncu5-1683507657217)(null)]

输入:root = [3,5,1,6,2,0,8,null,null,7,4], p = 5, q = 4
输出:5
解释:节点 5 和节点 4 的最近公共祖先是节点 5 。因为根据定义最近公共祖先节点可以为节点本身。

示例3

输入:root = [1,2], p = 1, q = 2
输出:1

题目分析:

通过分析题目可知,需要找的p、q节点在树种的存在情况只有:1.pq都在当前节点所在树的左子树中,在左子树中找pq的公共祖先即可;2.pq都在右子树中,在右子树中找pq的公共祖先即可;3pq分别在左右子树中,当前节点即是所求公共祖先节点;4.pq中有一个是当前节点,返回当前节点即可

代码实现:

class Solution {
public:
    bool isInTree(TreeNode* root, TreeNode* node)//判断node在不在树root中
    {
        if(root == nullptr)
            return false;
        //这里需要比较地址,而不是比较值
        return root == node || isInTree(root->left, node) || isInTree(root->right, node);
    }
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        //由于pq在树中,所以这里不需要考虑root为空的情况
        if(p == root || q == root)//pq有一个是当前节点
        {
            return root;
        }
        //判断pq的位置
        bool pInLeft = isInTree(root->left, p);
        bool pInRight = ! pInLeft;
        bool qInLeft = isInTree(root->left, q);
        bool qInRight = ! qInLeft;
        if((pInLeft && qInRight) || (pInRight && qInLeft))//pq在两侧
        {
            return root;
        }
        if(pInLeft && qInLeft)//都在左子树
        {
            return lowestCommonAncestor(root->left, p, q);
        }
        else//都在右子树
        {
            return lowestCommonAncestor(root->right, p, q);
        }
    }
};

image-20230506154625766

看一下结果,执行用时572ms,效率有点低啊,能不能优化一下嘞?分析一下我们的代码,我们的代码花费了太多时间在判断pq的位置上了,这里是一个普通二叉树,没办法很快的找到pq的位置,所以只能想想怎么优化掉这个过程了。

我们知道,pq节点肯定是在二叉树中的,那么他们在二叉树中就存在从根到节点的唯一路径,而且在这个路径中,绝对存在相同的部分,如果按照自低向上的看法,那么最终第一个相交的地方就是我们要找的公共祖先。那么做法思路就出来了:首先找到自低向上的节点路径,然后就可以类比链表相交的做法找到第一个相交的节点

代码实现:

class Solution {
public:
    bool GetPath(TreeNode* root, TreeNode* node, stack<TreeNode*>& path)
    {
        if(root == nullptr)
            return false;
        path.push(root);//首先当前节点入栈
        if(root == node)//当前节点就是所求节点时
            return true;
        //接下来需要在左右子树分别找,如果在左树中找到,就不需要去右树,否则还要去右树中找,所以这里需要判断有没有找到,所以这里把函数的返回值设置成bool
        if(GetPath(root->left, node, path))
        {
            return true;
        }
        //左树中没有,去右树种找
        if(GetPath(root->right, node, path))
        {
            return true;
        }
        //都没有时,代表当前根节点下边没有,所以pop掉
        path.pop();
        return false;
    }
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
		//在找pq的路径的时候,只能自顶向下找,所以找到之后需要逆置,而且后续操作中需要头删,所以这里干脆使用stack,就不需要中间的这些操作
        stack<TreeNode*> pPath;
        stack<TreeNode*> qPath;
        //这里不需要判断是否找到,因为题中已经确定pq在树种
        GetPath(root, p, pPath);
        GetPath(root, q, qPath);
        //让两个路径种长的先走,直到一样长的时候
        while(pPath.size() != qPath.size())
        {
            if(pPath.size() > qPath.size())
                pPath.pop();
            else
                qPath.pop();
        }
        //同时走,当节点相同时pop,返回相同节点即可
        while(pPath.top() != qPath.top())
        {
            pPath.pop();
            qPath.pop();
        }
        return pPath.top();
    }
};

image-20230506160945240

可以看到,效率提高显著。

4.二叉搜索树与双向链表

题目链接:二叉搜索树与双向链表_牛客题霸_牛客网 (nowcoder.com)

题干:

输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表。如下图所示

img

数据范围:输入二叉树的节点数 0≤n≤1000,二叉树中每个节点的值 0≤val≤1000
要求:空间复杂度O(1)(即在原树上操作),时间复杂度 O*(*n)

注意:

1.要求不能创建任何新的结点,只能调整树中结点指针的指向。当转化完成以后,树中节点的左指针需要指向前驱,树中节点的右指针需要指向后继
2.返回链表中的第一个节点的指针
3.函数返回的TreeNode,有左右指针,其实可以看成一个双向链表的数据结构

4.你不用输出双向链表,程序会根据你的返回值自动打印输出

输入描述:

二叉树的根节点

返回值描述:

双向链表的其中一个头节点。

示例1

输入:{10,6,14,4,8,12,16}
返回值:From left to right are:4,6,8,10,12,14,16;From right to left are:16,14,12,10,8,6,4;
说明:输入题面图中二叉树,输出的时候将双向链表的头节点返回即可。     

示例2

输入:{5,4,#,3,#,2,#,1}
返回值:From left to right are:1,2,3,4,5;From right to left are:5,4,3,2,1;
说明:               5
                  /
                4
              /
            3
          /
        2
      /
    1
树的形状如上图       

题目分析:

由于这是一个二叉搜索树,所以走中序遍历久是有序的结果,所以这里可以按照中序遍历的方式连接节点

代码实现:

class Solution {
public:
	void InOrderConvert(TreeNode* cur, TreeNode*& prev)//注意这里的参数类型设置,prev的参数类型要是引用,需要把prev的值带到其他栈帧中
	{
		if(cur == nullptr)//空树直接返回
			return;
		InOrderConvert(cur->left, prev);
		//中序操作
		cur->left = prev;//连接左结点
		if(prev)//连接右节点
			prev->right = cur;
		prev = cur;//交替向后走遍历所有节点
		InOrderConvert(cur->right, prev)
	}
    TreeNode* Convert(TreeNode* pRootOfTree) {
		TreeNode* prev = nullptr;
        InOrderConvert(pRootOfTree, prev);//重新连接节点

		TreeNode* head = pRootOfTree;
		while(head && head->left)//中序遍历的流程拿到最左节点
		{
			head = head->left;
		}
		return head;
    }
};
截屏2023-05-07 19.16.30

5. 从前序与中序遍历序列构造二叉树

题目链接: 105. 从前序与中序遍历序列构造二叉树

题干:

给定两个整数数组 preorder 和 inorder ,其中 preorder 是二叉树的先序遍历, inorder 是同一棵树的中序遍历,请构造二叉树并返回其根节点。

示例1:

img
输入: preorder = [3,9,20,15,7], inorder = [9,3,15,20,7]
输出: [3,9,20,null,null,15,7]

示例2

输入: preorder = [-1], inorder = [-1]
输出: [-1]

题目分析:

已知的条件是二叉树的前序和中续遍历的序列,那么通过前序序列可知根节点,在中序序列中,根节点把左右子树分开,就可以分别再构造左右子树,然后最后构造出整个二叉树

代码实现:

class Solution {
public:
    TreeNode* _buildTree(vector<int>& preorder, vector<int>& inorder, int& prei, int begin, int end)
    {
        if(begin > end)//当区间不存在时,直接return
        {
            return nullptr;
        }
        int i = 0;
        for(; i < inorder.size(); ++i)//找到当前的[begin,end]的中序序列里面根的位置
        {
            if(inorder[i] == preorder[prei])
                break;
        }
        TreeNode* root = new TreeNode(preorder[prei++]);//构造一个当前序列的根节点
				//分别构建左右子树
        // [begin,i-1] i [i+1, end]
        root->left = _buildTree(preorder, inorder, prei, begin, i-1);
        root->right = _buildTree(preorder, inorder, prei, i+1, end);
        return root;
    }
    TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {
        int i = 0;//创建先序序列中根的下标
        return _buildTree(preorder, inorder, i, 0, preorder.size()-1);
    }
};
截屏2023-05-07 21.13.31

拓展与补充: 106. 从中序与后序遍历序列构造二叉树 。

与上题类似,这里通过后续遍历序列来确定根节点,然后中序遍历序列中的根节点的位置来分别构造左右子树

6.二叉树的遍历

**题目链接 144. 二叉树的前序遍历 **

题干

给你二叉树的根节点 root ,返回它节点值的 前序 遍历。

示例1

img
输入:root = [1,null,2,3]
输出:[1,2,3]

示例2

输入:root = []
输出:[]

题目分析:

这个题使用递归的方式实现是非常轻松的,这里就不过多赘述,我们主要来理解迭代的思想使用迭代的方式解决此问题

改写迭代的重要性:递归的深度越深,消耗的资源越多,这个资源是在栈上的,而对于目前的操作系统来说,栈上的空间是很小的,大概只有8M左右,所以很容易出现栈溢出的问题,即使代码是没有问题的,也不一定能运行,相对于栈来说,堆上的空间就大很多了,所以一般将递归改写成迭代。

这里如果想要用迭代的方法遍历,就需要借助数据结构的栈来实现。我们来分析一下前序遍历的过程。对于任意一颗二叉树来说,可以分为左路节点和左路节点的右子树,首先就是根节点,然后是左子树的根,然后左子树的左子树,一直走下去,直到左子树为空然后走这个子树的右子树,右子树走完之后再回到上一个根的位置,再走右子树,直到回到整棵树的根节点为止。所以这里遵循后进先出的规则,因此我们借助栈来实现迭代的改写。

代码实现

class Solution {
public:
    vector<int> preorderTraversal(TreeNode* root) {
        vector<int> preorder;
        stack<TreeNode*> st;
        TreeNode* cur = root;
        while(cur || !st.empty())
        {
            //开始遍历一棵树,根节点为cur
            while(cur)//遍历左路节点并入栈
            {
                st.push(cur);
                preorder.push_back(cur->val);
                cur = cur->left;
            }
            //cur现在指向二叉树的最左节点,下一步出栈,并处理右子树
            TreeNode* top = st.top();
            st.pop();
            cur = top->right;//处理右子树
        }
        return preorder;
    }
};
截屏2023-05-08 08.02.22 **拓展1**:

94. 二叉树的中序遍历 递归解法

中序遍历与前序遍历类似,只是中序序列push_back的时候,不能先push根节点,而是在处理完左子树之后在push,所以代码示例如下

class Solution {
public:
    vector<int> inorderTraversal(TreeNode* root) {
        stack<TreeNode*> st;
        vector<int> inorder;
        TreeNode* cur = root;
        while(cur || !st.empty())
        {
            while(cur)
            {
                st.push(cur);
                cur = cur->left;
            }
            TreeNode* top = st.top();
            st.pop();
            inorder.push_back(top->val);//这里是在处理右子树之前push_back
            cur = top->right;
        }
        return inorder;
    }
};
截屏2023-05-08 08.26.29

拓展2

145. 二叉树的后序遍历 递归解法

同样的大思路,但是细节的地方还是要做一些修改。后续遍历要求访问完左右子树之后再访问根节点。分析过程可知,按照此思路走对于任意的右子树,将会访问两遍,分别是在此树访问右子树前和访问右子树后,这里要判断一下是否已经访问过了,如果访问过了就直接访问根,否则就访问右子树

class Solution {
public:
    vector<int> postorderTraversal(TreeNode* root) {
        stack<TreeNode*> st;
        vector<int> postorder;
        TreeNode* cur = root;
        TreeNode* prev = nullptr;//这里使用一个prev保存上一个访问的节点
        while(cur || !st.empty())
        {
            while(cur)
            {
                st.push(cur);
                cur = cur->left;
            }
            TreeNode* top = st.top();
//这里对右子树是否访问过的判断方法是判断上一个访问的节点是否是右子树,也可以使用其他方法,例如stack<pair<TreeNode*, bool>>
            if(top->right == nullptr || top->right == prev)//当右子树为空或者右子树已经访问过
            {
                st.pop();
                postorder.push_back(top->val);//访问根
                prev = top;//记录上一个访问节点
            }
            else//右子树没有被访问过,迭代访问右子树
                cur = top->right;
        }
        return postorder;
    }
};
截屏2023-05-08 08.49.48

这道题还有一个比较清奇的解法,就是按照类似前序遍历的方式,但是需要先遍历右子树,即根节点->右子树->左子树的顺序遍历一遍,也就是改写一下上述前序的方法,然后最终的结果reverse一下就是后序遍历序列,这里提供一下思路,有兴趣的可以试一下

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/501088.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

DS200TCQCG1BKG什么是控制模式,控制模式如何分类?

​ DS200TCQCG1BKG.什么是控制模式&#xff0c;控制模式如何分类&#xff1f; 控制回路的功能是在受控变量偏离该值时将其恢复到其设定值&#xff0c;从而将过程保持在所需条件下。实现这一点的动作称为控制模式。 控制方式分为两类 连续模式包括比例、积分和微分模式。 什么…

企业短信遭疯狂盗用,可能是没配置验证码

手机短信作为一种快捷的通讯方式被广泛应用。不仅在个人日常生活中&#xff0c;企业也习惯使用手机短信来进行验证和提醒&#xff0c;以保证业务的正常进行。随着数字化的发展&#xff0c;手机短信也成为了不法分子滥用的目标之一&#xff0c;给个人和企业带来不同经济损失。 个…

百度AI,和“吴文俊奖”同行的十二年、千丈山、万里路

今天&#xff0c;AI正作为一个科技发展周期的轴心&#xff0c;成为万众瞩目的焦点。与历史上数次技术革命和AI浪潮所不同的是&#xff0c;这次AI的全球领先阵营里&#xff0c;有了中国的身影。 从一个学术灵感&#xff0c;到一项全球领先的专利技术&#xff0c;从一篇顶会论文到…

MySQL基础(十)创建和管理表

1. 基础知识 1.1 一条数据存储的过程 存储数据是处理数据的第一步。只有正确地把数据存储起来&#xff0c;我们才能进行有效的处理和分析。否则&#xff0c;只能是一团乱麻&#xff0c;无从下手。 那么&#xff0c;怎样才能把用户各种经营相关的、纷繁复杂的数据&#xff0c…

关于如何对VS的C++项目进行完全重命名

很多人一个开始在VS编写C项目的时候&#xff0c;第一个项目名称都是系统默认名称或者HelloWorld这类的名字&#xff0c;一看就比较小白。 一段时间以后&#xff0c;项目已经进行了一段时间了&#xff0c;这时候想要对项目名称进行重命名。但是&#xff0c;偏偏VS的重命名功能做…

【homeassistant中ESPHome无法正常添加新设备指导操作】

【homeassistant中ESPHome无法正常添加新设备指导操作】 1. 在ESPHome添加设备1.1 问题显示1.2 添加NEW DEVICE1.3 烧录初始化固件2. 编辑主板的代码并录入3. 进行设备编译4. 编译完成后尝试亮灯5. ip地址的设置1. 在ESPHome添加设备 1.1 问题显示 点击添加设备,然后continu…

【网络】传输层协议-UDP协议

文章目录 传输层TCP/UDP预备知识:端口号的理解端口号的范围 关于端口号的相关问题netstat命令pidof命令 UDP协议所处的位置UDP协议格式UDP的特点UDP的缓冲区基于UDP的应用层协议 传输层TCP/UDP 回忆数据发送到网络的过程 之前在学习HTTP等应用层协议时为了方便理解:我们简单的认…

液晶显示控制驱动器HD61202介绍

液晶显示控制驱动器HD61202的特点 HD61202液晶显示控制驱动器是一种带有驱动输出的图形液晶显示控制器&#xff0c;它可直接与8位微处理器相连&#xff0c;它可与HD61203配合对液晶屏进行行、列驱动。HD61202是一种带有列驱动输出的液晶显示控制器&#xff0c;它可与行驱动器HD…

Docker 进阶实战:数据管理、网络

文章目录 Docker 进阶实战&#xff1a;数据管理、网络数据管理Volume创建数据卷挂载数据卷共享数据卷删除数据卷 Bind mountstmpfs mounts 网络端口映射容器互联Docker 内部网络Docker linkDocker Networking Docker 进阶实战&#xff1a;数据管理、网络 数据管理 默认情况下…

基于simulink使用麦克风阵列的声波束成形

一、前言 此示例演示如何对麦克风阵列接收到的信号进行波束化&#xff0c;以在嘈杂环境中提取所需的语音信号。 二、模型的结构 该模型模拟在 10 元件均匀线性麦克风阵列 &#xff08;ULA&#xff09; 上接收来自不同方向的三个音频信号。在接收器处添加热噪声后&#xff0c;应…

如何利用 Playwright 对已打开的浏览器进行爬虫!

大家好&#xff0c;我是安果&#xff01; 之前写过一篇关于如何利用 Selenium 操作已经打开的浏览器进行爬虫的文章 如何利用 Selenium 对已打开的浏览器进行爬虫&#xff01; 最近发现很多人都开始摒弃 Selenium&#xff0c;全面拥抱 Playwright 了&#xff0c;那如何利用 Pla…

经典文献阅读之--Point-LIO(鲁棒高带宽激光惯性里程计)

0. 简介 在我们之前接触的算法中&#xff0c;基本上都是要处理帧间雷达畸变的&#xff0c;类似于VSLAM系统&#xff0c;频率固定&#xff08;例如10Hz), 而实际上&#xff0c;激光雷达点是按照不同的时间瞬间顺序采样的&#xff0c;将这些点累积到帧中会引入人工运动畸变&…

3_机器学习数学基础知识

文章目录 一、偏导数二、目标函数&#xff08;损失函数&#xff09;求解方法2.1 梯度下降法2.2 坐标轴下降法2.2.1 坐标轴下降法算法公式2.2.2 坐标轴下降法算法过程 2.3 坐标轴下降法和梯度下降法的区别 三、概率3.1 大数定律、中心极限定理3.2 最大似然函数3.3 概率知识3.3.1…

windows下python下载及安装

下载python安装包 进入python官网&#xff1a;https://www.python.org/ 鼠标移动到“Downloads”->"Windows"上&#xff0c;可以看到最新版本是3.11.3版本 点击“Windows”按钮&#xff0c;可以去下载其他版本 标记为embeddable package的表示嵌入式版本&#x…

MyBatis:使用到的代码整理

文章目录 MyBatis&#xff1a;Day 04框架1. 依赖&#xff1a;pom.xml2. 外部配置文件&#xff1a;db.properties3. 核心配置文件&#xff1a;mybatis-config.xml4. 工具类&#xff1a;MybatisUtils.java5. 实体类6. 接口&#xff1a;xxxMapper.java7. 实现类&#xff1a;xxxMap…

[数据库系统] 三、简单查询

目录 第1关&#xff1a;简单查询 1.检索数据表所有的元组 2.检索符合条件的元组 educoder 第2关&#xff1a;多表查询 1.笛卡尔积 2.等值连接 3.自然连接 4.左外连接和右外连接的表示方法及转换为SQL educode 第3关&#xff1a;集合操作 1.集合并操作 2.集合交操作…

小学生护眼用什么样的台灯比较好用一点?小学生护眼台灯排行榜

孩子近视了&#xff0c;很多家长认为是数码产品导致的。其实除了数码产品&#xff0c;灯光也是一个非常重要的原因。或许你认为卧室的灯亮孩子写作业就没有问题&#xff0c;其实这种情况仍会出现灯下黑的现象。如果你想孩子写作业不受灯光的影响&#xff0c;那么一定要给他们配…

实验四 数据库设计

【实验目的】 1、掌握规范化数据库设计包括步骤及其任务、方法、结果等。 2、掌握数据库设计过程中关键文档的写法。 3、了解数据库辅助设计工具。 【实验内容】 请按下面大纲书写实验报告&#xff08;请自行附页&#xff09;。 &#xff08;1&#xff09;需求分析&#…

【MySQL】Mysql索引失效场景(15个必知)

文章目录 背景数据库及索引准备创建表结构初始化数据数据库版本及执行计划 1、联合索引不满足最左匹配原则2、 使用了select *3 、索引列参与运算4、 索引列参使用了函数5、 错误的Like使用6、 类型隐式转换7、使用OR操作8、 两列做比较9、 不等于比较10、 is not null11、 not…

ChatGPT实现stackoverflow 解释

stackoverflow 解释 ChatGPT 公开服务以来&#xff0c;程序员们无疑是最早深入体验和"测试"的一批人。出色的效果也引发了一系列知识产权上的争议。著名的 stackoverflow 网站&#xff0c;就宣布禁止用户使用 ChatGPT 生成的内容来回答问题&#xff0c;一经发现&…