【数据结构】基础知识总结
系列综述:
💞目的:本系列是个人整理为了数据结构
复习用的,由于牛客刷题发现数据结构方面和王道数据结构的题目非常像,甚至很多都是王道中的,所以将基础知识进行了整理,后续会将牛客刷题的错题一并整理到该文章中,可以考试复习或者找工作复习使用。
🥰来源:材料主要源于《王道数据结构考研复习指导》
进行的,每个知识点再考研中及复习至少思考过三遍,比较可靠。
🤭结语:如果有帮到你的地方,就点个赞和关注一下呗,谢谢🎈🎄🌷!!!
文章目录
- 数据结构
- 一、绪论
- 基本概念
- 数据结构三要素
- 算法
- 二、线性表
- 三、栈和队列
- 栈
- 队列
- 四、串
- 串的定义和实现
- 串的存储结构
- 五、树与二叉树
- 树
- 二叉树
- 树与森林
- 二叉排序树
- 平衡二叉树
- 六、图
- 基本概念
- 图的存储
- 图的遍历
- 图的应用
- 七、查找
- 基本概念
- 静态查找
- 动态查找
- 八、排序
- 插入排序
- 交换排序
- 选择排序
- 特殊排序
- 内部排序算法比较
- 参考博客
😊点此到文末惊喜↩︎
- 相关思维导图:https://wwb.lanzouy.com/iAZyi0d3qbyd
数据结构
一、绪论
基本概念
- 数据对象⊃数据元素⊃数据项数据对象 \supset 数据元素 \supset 数据项数据对象⊃数据元素⊃数据项
- 数据对象是具有
相同性质的数据元素的集合
- 数据元素是
数据的基本单位
,由若干数据项组成 - 据项是
构成数据元素不可分割最小单位
- 数据对象是具有
- 数据类型:是一个值的集合和定义在该集合上的一组操作
- 原子类型:值不可再分
- 结构类型:值可以再分
- 数据结构
- 定义:数据元素 + 相互关系
- 数据的运算实质就是算法的执行,由逻辑结构定义,由存储结构实现。不能过于考虑逻辑结构,而忽略存储结构是否容易实现,逻辑结构和存储结构都影响着算法的好坏。
数据结构三要素
- 逻辑结构
- 集合:数据元素
同属于一个集合
,其他无任何关系 - 线性结构:数据元素之间只存在
一对一
的关系,如栈、队列、串、数组、其他线性表 - 树状结构:数据元素间存在
一对多
的关系 - 图状结构:数据元素间存在
多对多
的关系
- 集合:数据元素
- 存储结构
- 顺序存储:逻辑相邻的数据元素存储在物理上相邻的位置。
- 优点:可以实现随机存取,占用空间小
- 缺点:可能产生空间碎片
- 链式存储:不同节点只要求逻辑上相邻,但是节点内要求物理连续
- 优点:无碎片现象,元素的增删容易
- 缺点:指针占用额外的空间,只能顺序存取
- 索引存储:存储元素的同时建立索引表
- 优点:查找速度快
- 缺点:索引表占用额外的空间,增删数据需要花费时间修改索引表
- 散列存储(哈希存储):将数据元素的值映射成存储地址
- 优点:增删改查都很快
- 缺点:出现哈希冲突会增大时空开销
- 顺序存储:逻辑相邻的数据元素存储在物理上相邻的位置。
算法
- 算法的特性
- 输入:有零个或多个输入
- 有穷性:每一步均可在有穷时间内完成
- 确定性:相同的输入只能获得相同的输出
- 可行性:算法可以通过计算机基本指令实现
- 输出:至少一个输出
- 好的算法
- 正确性:正确解决问题
- 健壮性:及时处理非法数据
- 可读性:易于人们理解
- 时空开销小
- 算法效率的度量
- 时间复杂度:最深层嵌套语句运算次数的数量级
- 空间复杂度:算法耗费的存储空间,原地存储指
O(1)
- O(1)<O(log2n)<O(n)<O(nlog2n)<O(n2)<O(2n)<O(n!)<O(nn)O(1)<O(log_{2}n)<O(n)<O(nlog_{2}n)<O(n^2)<O(2^n)<O(n!)<O(n^n)O(1)<O(log2n)<O(n)<O(nlog2n)<O(n2)<O(2n)<O(n!)<O(nn)
二、线性表
- 线性表:具有相同数据类型数据元素的有限序列
- 逻辑和物理的关系:是一种逻辑结构,表示数据元素一对一的相邻关系,即除表头元素其他元素只有一个直接前驱,除表尾元素其他元素只有一个直接后继
- 存储空间上:每个元素占用相同大小的存储空间
- 顺序表:线性表的顺序存储(逻辑结构和存储结构的结合)
- 逻辑和物理的关系:表中逻辑顺序和物理顺序相同
- 存储空间上:每个节点只存储数据元素,不需要浪费额外空间
- 增删改查:插入和删除需要移动大量元素,但是可以随机存取
- 静态分配,数组大小在编译阶段确定,执行时一旦超过可能发生溢出
- 动态分配,在程序执行时分配,一旦超过原空间,需要开辟更大的空间进行存储
- 单链表:线性表的链式存储
- 逻辑和物理的关系:表中逻辑顺序和物理顺序一般不相同
- 存储空间上:每个节点只存储数据元素,需要额外存储指针
- 增删改查:查找需要进行遍历,但是增删容易
- 头插法:链表元素的读入顺序和生成顺序时相反的
- 尾插法:链表元素的读入顺序和生成顺序时相同的
- 双链表:每个节点有两个指针prior和next,可以逆序遍历
- 循环单链表:尾指针指向头节点的链表,判空条件是头节点指针是否指向自己
- 循环双链表:判空条件是头节点的prior和next都等于头节点
- 静态链表:使用数组描述线性表的链式存储
- 节点具有数据域和指针域,指针域存放的是数组下标
- 结束标志是next == -1
- 静态链表的插入和删除只需要修改指针,不需要移动元素
- 可用于不支持指针的高级语言Basic
三、栈和队列
栈
- 栈的基本概念
- 栈:只允许在一端进行插入和删除操作的线性表,即后进先出
- 栈顶(Top):线性表允许进行插入删除的那一端
- 栈底(Bottom):线性表不允许插入和删除的那一端
- 空栈:不含任何元素的空线性表
- n个不同元素进栈,出栈元素不同排列的个数为1n+1C2nn\frac{1}{n+1}C_{2n}^{n}n+11C2nn
- 最后一个元素第一个出栈,则输出序列一定是逆序 - 判断出栈顺序是否正确:按序进栈并在栈中相邻的元素出栈一定是逆序的
- 顺序栈:采用顺序存储的栈(逻辑和物理存储顺序一致)
//数组 + 栈顶指针top #define MaxSize 50 typedef struct {Elemtype data[MaxSize];// 存放栈中元素,数组栈存在上溢的问题int top;// 栈顶指针 }SqStack;
- 栈顶指针:初始化时通常设置
S.top = -1
- 栈顶元素:
S.data[S.top]
- 进栈操作:栈不满时,栈顶指针先加1,再送值到栈顶元素
- 出栈操作:栈非空时,先取栈顶元素值,再将栈顶指针减1
- 栈空条件:
S.top == -1
- 栈满条件:
S.top == MaxSize-1
- 栈长:
S.top+1
- 栈顶指针:初始化时通常设置
- 共享栈
- 定义:栈底在共享数组两端,栈顶向共享空间中间延伸
- 栈空条件:
top0 = -1
时0号栈为空,top1=MaxSize
时1号栈为空 - 栈满条件:仅当两个栈顶指针相邻即
top1 - top0 = 1
- 链栈
typedef struct Linknode{Elemtype data;struct Linknode *next; }* LiStack;
- 定义:采用链式存储的栈
- 优点:不会栈满上溢,便于节点的插入和删除,但是需要前一个节点的辅助
队列
- 队列的基本概念
- 定义:只允许在线性表的一端进行插入,而在另一端进行删除的线性表,即先进先出
- 出队:在队头删除元素
- 入队:在队尾加入元素
- 空队列:不含任何元素的线性表
- 栈和队列都是操作受限的线性表,不可以随便读取栈或队列中间的某个数据
- 队列的顺序存储
#define MaxSize 50 typedef struct {Elemtype data[MaxSize];// 存放栈中元素,数组栈存在上溢的问题int front,rear; }SqQueue;
- 定义:分配一块连续的存储单元存放队列中的元素,队头指针front指向对头元素,队尾指针rear指向队尾元素的下一个位置
- 队空条件:Q.front == Q.rear==0
- 进队操作:队不满时,先送至到队尾元素,再将队尾指针加1
- 出队操作:队不空时,先取队头元素值,再将队头指针加1
- 假溢出:存在连续出队导致分配使用的内存不足,使用循环队列进行解决
- 循环队列
- 定义:将存储队列元素的表在逻辑上视为一个环
- 出队:
Q.front = Q.rear = 0
- 入队:
Q.rear = (Q.rear+1) % MaxSize
- 队列长度:
(Q.rear-Q.front+MaxSize)% MaxSize
- 将队尾指针指向队尾的下一个元素,牺牲了队尾指针指向的单元。将队头指针在队尾指针的下一位置作为队满的标志
- 队满条件:
(Q.rear+1) % MaxSize == Q.front
- 队空条件:
Q.front == Q.rear
- 链队列
typedef struct {ElemType data;struct LinkNode *next; }LinkNode; typedef struct{LinkNode *front,*rear; }LinkQueue;
- 定义:队列的链式表示,头指针指向队头节点,尾指针指向队尾节点的单链表
- 判空条件:
Q.front == NULL && Q.rear == NULL
- 设计:通常将链式队列设计成带头节点的单链表,统一插入和删除操作
- 优点:有利于元素数据变动,无溢出问题
- 双端队列
- 定义:允许前端和后端均可出入队操作的队列
- 输出受限的双端队列:允许在一端进行插入和删除,在另一端只允许插入的双端队列
- 输入受限的双端队列:允许在一段进行插入和删除,在另一端只允许删除的双端队列
- 特殊矩阵的压缩存储
- 压缩矩阵:为多个值相同的元素值分配一个存储空间,对零元素不分配存储空间
- 特殊矩阵:分布具有规律性的矩阵,例如上三角矩阵、对角矩阵、对称矩阵
- 稀疏矩阵:矩阵中非零元素相对于矩阵整体元素个数来说比较少,通常使用三元组存储
四、串
串的定义和实现
- 定义:由零个或多个字符组成有限序列
- 子串:串中任意个连续的字符组成的子序列
- 主串:包含子串的串,串的主要操作是以子串为对象进行的
- 空格串:由一个或多个空格组成的串
- 串相等:串的长度和每个位置对应的字符都相等
串的存储结构
-
定长顺序存储
#define MAXLEN 255 typedef struct{char ch[MAXLEN];// 字符串的最后一位是‘\0’int length; }SString;
-
堆分配存储表示
typedef struct{char *ch; // 按照串长分配存储区,ch指向串的基地址int length; }HString;
-
串长的两种表示方式
- length变量存储
- 结束标记法
五、树与二叉树
树
- 定义:树是n个节点的有限集合,当n=0时,称为空树
- 树的定义是递归的,具有以下特点
- 树的根节点没有前驱,除根节点外的所有结点都有且只有一个前驱
- 树中所有节点可以有零个或多个后继
- 树适合表示有层次结构的数据
- 在n个节点的树中有n-1条边
- 基本概念
- 祖先:根A到节点K的唯一路径上的任何节点,称为节点K的祖先
- 子孙:如果节点B是节点K的祖先,则节点K是节点B的子孙
- 兄弟节点:具有相同的父节点,根A是树中唯一没有双亲的结点
- 节点的度:某个节点的孩子的个数
- 树的度:树中节点的最大度数
- 分支节点:度大于0的节点
- 叶子节点:度为0的节点
- 节点的深度:从根节点开始自顶向下逐层累加的
- 节点的高度:从该节点分支的叶节点开始自底向上逐层累加的
- 树的高度(深度):树中节点的最大的层数
- 有序树:树中节点的各字数从左到右是有序的,不能互换
- 路径:自顶向下两个节点所经过的结点序列构成的
- 路径长度:路径上所经过的边的个数,同一个双亲之间的孩子不存在路径
- 森林:m棵互不相交的树的集合
- 一棵度为m的树
- 总节点数=n0+n1+n2+⋅⋅⋅总节点数 = n_0 + n_1+n_2+···总节点数=n0+n1+n2+⋅⋅⋅
总度数/总分支数=1∗n1+2∗n2+3∗n3+⋅⋅⋅总度数/总分支数 = 1*n_1+2*n_2+3*n_3+···总度数/总分支数=1∗n1+2∗n2+3∗n3+⋅⋅⋅
总节点数=总度数+1总节点数 = 总度数 +1总节点数=总度数+1 - 度为m的树中第i层上至多有mi−1m^{i-1}mi−1个节点
- 高度为h个m叉树至多有(mh−1)/(m−1)(m^h-1)/(m-1)(mh−1)/(m−1)个节点
- 总节点数=n0+n1+n2+⋅⋅⋅总节点数 = n_0 + n_1+n_2+···总节点数=n0+n1+n2+⋅⋅⋅
二叉树
- 特点
- 每个节点至多有两颗子树
- 二叉树是有序树,左右子树顺序不能颠倒
- 二叉树以递归形式定义
- 二叉树和度为2的有序树的区别
- 度为2的树至少有3个节点,而二叉树可以为空
- 度为2的树若某个节点只有一个孩子,则该孩子无须区分左右次序。有序树孩子左右次序是固定的
- 特殊的二叉树
- 满二叉树:一个高度为h,且含有2h−12^h-12h−1个节点的二叉树称为满二叉树,即树中每层都含有最多的节点。
- 完全二叉树:最后一层自左向右排列不满,其余均满。即度为1的节点只有左孩子。
- 二叉排序树:左子树上所有节点的关键字均小于根节点的关键字,右子树上的所有结点均大于根节点。左子树和右子树又各是一颗二叉排序树
- 平衡二叉树:书上的任一结点的左子树和右子树的深度之差不超过1
- 二叉树的性质
- 非空二叉树的叶子结点等于度为2的结点数加1,n0=n2+1n_0 = n_2+1n0=n2+1
- 结点数量为n,则边的数量为n-1
- 非空二叉树的第k层至多有2k−12^{k-1}2k−1个结点
- 高度为h的树至多有2h−12^h-12h−1个结点
- 完全二叉树和满二叉树可以自顶向下,自左向右进行顺序编号
- 顺序存储
- 定义:二叉树的顺序存储是指用一组地址连续的存储单元一次自上而下、自左向右存储完全二叉树上的结点
- 满二叉树和完全二叉树适合采用顺序存储,优点
- 节省存储空间
- 利用数组元素下标可以确定结点在二叉树中的位置
- 忽略数组第0个元素,从数组下标1开始存储的性质
- 当i为偶数时,其双亲的编号为i/2,是左孩子。
当i为奇数时,其双亲的编号为(i-1)/2,是右孩子。 - 结点i的左孩子编号为2i,否则无左孩子
结点i的右孩子编号为2i+1,否则无右孩子
- 当i为偶数时,其双亲的编号为i/2,是左孩子。
- 链式存储结构
typedef struct BiNode{ElemType data; // 数据域struct BiTNode *lchild, *rchild; // 左右孩子指针 }BiTNode, *BiTree;
- 在n个结点的二叉链表中,含有n+1个空链域
- 二叉树的遍历
- 定义:某条搜索路径访问树中的每个结点,使得每个结点均被访问一次
- 由遍历序列构造二叉树
- 由二叉树的先序和中序序列可以唯一确定一颗二叉树
- 由二叉树的中序和后序序列可以唯一确定一颗二叉树
- 有五种遍历方式,空间复杂度和时间复杂度均为O(n)
// 先序遍历:根左右void preOrder(BiTree T){if(T != NULL) {visit(T); // 遍历根结点preOrder(T->lchild); // 遍历左子树preOrder(T->rchild); // 遍历右子树}}// 中序遍历:左根右void preOrder(BiTree T){if(T != NULL) {preOrder(T->lchild); // 遍历左子树visit(T); // 遍历根结点preOrder(T->rchild); // 遍历右子树}}// 后序遍历:左右根void preOrder(BiTree T){if(T != NULL) {preOrder(T->lchild); // 遍历左子树preOrder(T->rchild); // 遍历右子树visit(T); // 遍历根结点}}// 层次遍历/广度优先遍历void levelOrder(BiTree T) {initQueue(Q); // 初始化辅助队列BiTree p;EnQueue(Q, T); // 将根节点入队while (!IsEmpty(Q)) { // 队列不空则循环DeQueue(Q, p); // 队头结点出队visit(p); // 访问队头结点if (p->lchile != NULL) EnQueue(Q, p->lchild); // 左子树不空,则左子树根节点入队if (p->rchild != NULL)EnQueue(Q, p->rchild); // 右子树不空,则右子树根节点入队}}
- 线索二叉树
- 目的:充分利用空指针域,加快查找结点前驱和后继的速度
- 方式
- 若无左子树,令lchile指向其前驱结点
若无右子树,令rchild指向其后继结点
- 若无左子树,令lchile指向其前驱结点
typedef struct ThreadNode{ElemType data; // 数据元素struct ThreadNode *lchildl, *rchild; // 左右孩子指针int ltag, rtag; // 左表示前驱,右表示后继,1表示前后,0表示孩子 }ThreadNode, *TheadTree;// 中序线索二叉树的建立 void InThread(ThreadTree T){if(T != NULL) {InThread(T->lchild);visit(T);InThread(T->rchild);} } void visit(ThreadNode *q){if(q->lchild == NULL){q->lchild = pre;q->ltag = 1;} if(pre != NULL && pre->rchild == NULL){pre->rchild = q;pre->rtag = 1;}pre = q;S }
树与森林
- 树的存储结构
-
双亲优先法
#define MAX_TREE_SIZE 100 // 树中最多的结点数 typedef struct {ElemType data; // 数据元素int parent; // 双亲数组下标 }PTNode; typedef struct { // 树的类型定义PTNode nodes[MAX_TREE_SIZE]; // 双亲定义int n; // 结点数 }PTree;
- 采用顺序存储,第一个是根节点,其双亲为-1
- 可很快得到双亲结点,但是求结点的孩子需要遍历整个结构
-
孩子优先法
typedef struct SNode{ElemType data;SNode *child; }SNode;SNode T[MAX_SIZE];// 双亲结点数组
- 使用邻接链表,数组存储存储所有结点表示双亲,链表元素表示双亲节点的孩子
- 寻找孩子结点非常直接,寻找双亲需要完整遍历
-
孩子兄弟表示法
typedef struct CSNode{ElemType data;struct CSNode *firstchild, *nextsibling; // 左指针为第一个孩子,右指针为孩子的兄弟 }CSNode, * CSTree;
- 易于查找孩子,但是难以查找双亲
-
- 树转化为二叉树
- 每个结点的左指针指向它的第一个孩子,右指针指向它在树中的相邻右兄弟
- 树的遍历
- 先根遍历:先根后顺序遍历相应子树,遍历序列和对应二叉树的先序序列相同
- 后根遍历:先顺序遍历子树后访问根,遍历序列和该树对应二叉树的中序序列相同
- 层次遍历:按每层结点顺序访问
二叉排序树
- BST的特性
- 左子树非空则左子树所有结点的值均小于根节点的值
- 右子树非空则右子树所有结点的值均大于根节点的值
- 左右子树分别各是一颗二叉排序树
- 重要:对二叉排序树结点中序遍历,可以得到一个递增的有序序列
- BST的非递归查找算法
BSTNode *BST_Search(BiTree T, ElemType key){while (T != NULL && key != T->data){if(key < T->data)T = T->lchild;elseT = T->rchild;}return T; }
- BST的插入和构造
int BST_Insert(BiTree &T, KeyType k) {if (T == NULL){T = (BiTree)malloc(sizeof(BSTNode));T->key = k;T->lchild = T->rchild = NULL;return l;}else if(k == T->key)return 0;else if(k < T->key)return BST_Insert(T->lchlde, k);elsereturn BST_Insert(T->lchild, k); } // 二叉排序树的构造 void Creat_BST(BiTree &T, keyType str[], int n){T = NULL;int i = 0;while (i < n) {BST_Insert(T, str[i]);i++;} }
- 二叉排序树的删除
- 若被删结点为叶结点,则直接删除
- 若被删结点只有一颗子树,则让该节点的子树成为其父亲结点的子树
- 若被删结点有左右子树,则从左子树中最大结点代替被删结点,从右子树中找最小结点代替被删结点
- 二叉排序树的查找效率
- 最好情况,二叉排序树是一个平衡二叉树,平均查找长度为O(log2n)O(log_2n)O(log2n)
- 最坏情况,输入序列是有序的,则形成一颗单支树,平均查找长度为O(log2n)O(log_2n)O(log2n)
- 二分查找的对象是有序顺序表
- 当有序表是静态查找表时,宜用顺序表作为存储结构
- 当有序表是动态查找表时,宜用二叉排序树作为逻辑结构
平衡二叉树
- 定义:任意结点的左右子树的高度差的绝对值不超过1
- 目的:为了避免树的高度增长过快,降低二叉排序树的性能
- 插入
- 调整对象:最小不平衡子树
- 四种平衡操作
- 平衡二叉最大深度和平均查找长度均为O(log2n)O(log_2n)O(log2n)
- 哈夫曼树的基本概念
- 带权路径长度:根到任意结点的路径长度(经历的边的数量)与该节点权值的乘积
- 树的带权路径长度:所有叶结点的带权路径长度之和
- 哈夫曼树:带权路径长度最小的二叉树
- 哈夫曼树的构造
- 给定n个带权结点序列,每次取出两颗权值最小的结点构成一颗二叉树,树的根节点为左右子树的结点之和,并将根节点加入到序列中
- 哈夫曼树的特点
- 每个初始结点都最终成为叶结点,权值越小的结点到根节点的路径长度越大
- 初始结点数量为n个,则哈夫曼树的结点总数2n-1
- 哈夫曼树中不存在度为1的结点
- 相同的序列可能构造不同的哈夫曼树,但是带权路径长度WPL相同且为最优
- 哈夫曼编码
- 固定长度编码:对每个字符用相等长度的二进制位表示
- 可变长度编码:允许对不同字符使用不等长的二进制位表示
- 哈夫曼树通常使用可变长度编码,对频率高的字符赋予更短的编码,启到压缩数据的目的
- 前缀编码:没有一个编码是这个编码的前缀
- 由哈夫曼树构建哈夫曼编码
- 初始结点的权值为该节点的频率
- 在哈夫曼树中对每个结点进行左0右1的标记
- 每个叶子结点从根节点开始形成的01序列即为编码
六、图
基本概念
- 图:由顶点集和边集组成,顶点的个数称为图的阶
- 线性表可以是空表,树可以是空树,但是图不可以是空图,顶点咋集一定非空
- 有向图:由顶点集和有向边集(也称为弧)组成,有向边使用<v,u> 序列表示,可以双向
- 无向图: 由顶点集和无向边集组成
- 简单图:不存在重复边和顶点到自身的边
- 多重图:允许两顶点间边数多于一条或顶点通过一条边和自己关联
- 完全图:顶点集合具有n个的无向图,具有n(n-1)/2条边
- 有向完全图:顶点集合具有n个的有向图,具有n(n-1)条边。在有向完全图中任意两个顶点之间都存在方向相反的两条弧
- G’是G的子图:G =(V,E)和G’=(V’ ,E’),若V’是V的子集,且E是E’的子集,其中E’的顶点必须在V’内?
- 生成子图:与主图顶点集相同的子图
- 连通图:图中任意两个顶点都有路径存在,边数小于n-1的必是非连通图
- 极大连通子图:包含主图所有边的连通子图,再加入一个顶点都会使其不连通
- 极小连通子图:保持图连通并使边数最少的子图
- 连通图的生成树:包含图中所有顶点的一个极小连通子图,边数必为n-1
- 强连通图:图中的每一对顶点都有双向的路径
- 顶点的度
- 无向图:该顶点边的条数。无向图全部顶点的度的和等于边数的两倍
- 有向图
- 入度:以顶点v为终点的有向边的数目
- 出度:以顶点v为起点的有向边的数目
- 有向图的全部顶点的入度之和与出度之和相等,且等于边数
- 带权图/网:边上带有权值的图
- 稀疏图、稠密图:使用边数的多少进行划分,一般∣E∣<∣V∣log∣V∣|E|<|V|log|V|∣E∣<∣V∣log∣V∣认为是稀疏图
- 路径:由边关联的相邻顶点组成的序列
- 路径长度:路径上边的数目
- 回路/环:第一个顶点和最后一个顶点相同的路径
- 简单路径:顶点不重复出现的路径
- 简单回路:除了第一个顶点和最后一个顶点外其余顶点不重复出现的回路
- 距离:两个顶点之间最短的路径
- 有向树:一个顶点的入度为0,其余顶点的入度均为1的有向图
图的存储
- 邻接矩阵
- 定义:存储顶点之间邻接关系二维数组
- 带权图中邻接矩阵存放着该边对应的权值
- 有向图和无向图的矩阵:n个节点生成n×n的矩阵,某行某列不为零表示行序号的节点到列序号的节点有边,带权图某行某列为其权值
- 结构体
#define MaxVertexNum 100 // 顶点数目的最大值 typedef char VertexType; // 顶点的数据类型 typedef int EdgeType; // 带权图中边上给权值的数据类型 typedef struct{VertexType Vex[MaxVertexNum]; // 顶点表EdgeType Edge[MaxVertexNum][MaxVertexNum]; // 邻接矩阵、边表int vexnum, arnum; // 图的当前顶点数和弧数 }MGraph;
- 特点:
- 无向图的邻接矩阵是对称矩阵,可以进行压缩操作,只存储半三角
- 邻接矩阵表示法的空间复杂度为O(n2)O(n^2)O(n2),其中n为图的顶点数量|V|
- 无向图的邻接矩阵的第i行(或第i列)非零元素(或非无穷元素)的个数正好是第i个顶点的度数
- 有向图第i行全部的有值元素之和为顶点i的出度,第i列全部的有值元素之和为顶点i的入度
- 邻接矩阵容易确定顶点之间是否有边相连,但确定边数花费时间比较大
- 稠密图适合使用邻接矩阵存储
- 邻接表法
- 定义:顶点表使用顺序存储,边表使用链式存储
- 结构体
#define MaxVertexNum 100 // 图中顶点的最大数量 // 边表节点 typedef struct ArcNode{int adjvex; // 该弧所指向的顶点位置struct ArcNode *next; // 指向下一条弧的指针//InfoType info; // 网的边权值 }ArcNode; // 顶点表节点 typedef struct VNode{VertexType data; // 顶点数据ArcNode *first; // 指向第一条依附该顶点的弧线的指针 }VNode,AdjList[MaxVertexNum]; // 邻接表 typedef struct{AdjList vertices; // 图的顶点数和弧数int vexnum, arcnum; // AlGraph是邻接表存储的图 }ALGraph;
- 特点:
- G为无向图,则所需的存储空间为O(∣V∣+2∣E∣)O(|V|+2|E|)O(∣V∣+2∣E∣)。G为有向图,则所需的存储空间为O(∣V∣+∣E∣)O(|V|+|E|)O(∣V∣+∣E∣)。
- 邻接链表适合存储稀疏图
- 给定顶点,通过邻接链表可以很容易的找到它的所有邻边和顶点
- 有向图的邻接链表容易计算某个顶点的出度,但是不容易计算入度,需要遍历全部邻接链表
- 图的邻接链表不是唯一的,各边节点的连接次序可以任意的
- 十字链表
- 定义:有向图的一种链式存储结构,由顶点结构体和边结构体组成
- 常用于存储稀疏图,容易求得顶点的出度和入度
- 邻接多重表
- 定义:无向图的一种链式存储,由顶点结构体和边结构体组成
- 相比于邻接表同一条边需要两个节点表示,在邻接多重表中只使用一个节点
图的遍历
- 定义:从图中某一个顶点出发,对图中所有顶点访问一次且仅访问一次,
- 广度优先搜索BFS
bool visited[MAX_VERTEX_NUM]; // 遍历所有节点 void BFSTraverse(Graph G){for(i = 0; i < G.vexnum; ++i)visited[i] = FALSE;InitQueue(Q);for(i=0; i<G.vexnum; ++i)if(!visited[i])BFS(G, i);} // 遍历输入节点的所有子节点 void BFS (Graph G, int v){visit(v);visited[v] = TRUE;Enqueue(Q, v);while(!isEmpty(Q)){DeQueue(Q,v);for (w=FirstNeighbor(G,v); w>=0; w=NextNeighbor(G, v, w))if(!visited[w]){visit(w);visited[w] = TRUE;EnQueue(Q, w);}} }
- 算法时间复杂度
- 邻接表存储为O(∣V∣+∣E∣)O(|V|+|E|)O(∣V∣+∣E∣)
- 邻接矩阵存储为O(∣V2∣)O(|V^2|)O(∣V2∣)
- 广度优先遍历过程可以得到一颗广度优先生成树,广度优先生成树的唯一性和邻接矩阵存储表示的唯一性一致
- 算法时间复杂度
- 深度优先搜索DFS
bool visited[MAX_VERTEX_NUM]; void DFSTraverse(Graph G){for(v=0; v<G.vexnum; ++v)visited[v]=FALSE;for(v=0; v<G.vexnum; ++v)if(!visited[v])DFS(G,v); }void DFS(Graph G, int v){visit(v);visited[v] = TRUE;for(w = FirstNeighbor(G, v); w>=0; w=NextNeighor(G,v,w))if(!visited[w]){DFS(G,w);} }
- DFS是一个递归算法,空间复杂度为O(∣V∣)O(|V|)O(∣V∣)
- 时间复杂度
- 邻接表存储为O(∣V∣+∣E∣)O(|V|+|E|)O(∣V∣+∣E∣)
- 邻接矩阵存储为O(∣V2∣)O(|V^2|)O(∣V2∣)
图的应用
- 最小生成树
- 定义:包含连通图的所有顶点 && 只含尽可能少的边的生成树 && 边的权值之和最小
- 性质
- 不一定唯一,图中的各边权互不相等时,G的最小生成树是唯一的
- 无向连通图G的边数比顶点数小1,即G本身是一颗树时,则G的最小生成树就是它本身
- 最小生成树的边的权值之和总是唯一的,且是最小的
- 最小生成树的边数为顶点数减1
- 生成算法1: Prim算法
- 过程:开始从图中任取一顶点,加入树T中,之后选择一个与当前T中顶点集合距离最近且加入后不生成环的顶点加入集合,再重复以上过程。
- 特点:时间复杂度为O(∣V∣2)O(|V|^2)O(∣V∣2),适用于求解边稠密图的最小生成树
- 生成算法2: Kruskal算法
- 过程:将边的权值由小到大进行排序,不断选取当前未被选取的权值最小的边且加入后不会构成环
- 特点:主要由排序时间决定,适合顶点多的图
- 最短路径
- 定义:带权图中两顶点长度最短的那条路径
- 最短路径问题分类
- Dijkstra算法:单源最短路径问题,即求图中某一个顶点到其他各顶点的最短路径
特点:1.基于贪心策略的 2.时间复杂度均为O(∣V∣2)O(|V|^2)O(∣V∣2) 3.不适用边上有负权值的 - Floyd算法:求顶点对之间的最短路径
特点:1.时间复杂度为O(∣V∣3)O(|V|^3)O(∣V∣3) 2. 不允许带有负权值边组成的回路
- Dijkstra算法:单源最短路径问题,即求图中某一个顶点到其他各顶点的最短路径
- 有向无环图:不存在环的有向图,简称DAG图
- 用于描述含有公共子式的共享,从而节省存储空间
- 拓扑排序
- AOV网:使用顶点表示前后关系的有向无环图
- 定义:有一个有向无环图组成的序列,且满足以下关系
- 每个顶点出现且只出现一次
- 在序列中靠后的顶点不存在到前面顶点的路径
- 每个AOV网中存在一个到多个拓扑排序序列
- 对AOV网进行拓扑排序
- 从AOV网中选择入度为0的顶点并输出
- 从网中删除该顶点和所有以它为起点的有向边
- 重复前两步知道当前AOV网络为空或网中不存在无前驱的顶点
- 特点:
- 时间复杂度为O(∣V∣+∣E∣)O(|V|+|E|)O(∣V∣+∣E∣)
- 有向无环图的拓扑序列唯一不能唯一确定该图
- 若不存在拓扑序列则图中必有回路
- 关键路径
- AOE网:在带权有向图中,以顶点表示事件,以有向边表示活动,边上的权值表示完成活动的开销
- 概念:
- 开始顶点(源点 ):AOE网中仅有的一个入度为0的顶点,表示整个工程的开始
- 结束顶点(汇点):网中仅有的一个出度为0的顶点,表示整个工程的结束
- 关键路径:从源点到汇点的所有路径中,具有最大路径长度的路径。这个路径决定了工程最短结束事件
- 性质
- 只有在某顶点所代表的事件发生后,从该顶点出发的各有向边所代表的活动才能开始
- 只有在进入某个顶点的各有向边所代表的活动都结束,该顶点所代表的事件才能发生
- 网络中的关键路径可能不唯一,且关键路径上的所有活动都是关键活动
- 缩短所有关键路径上共有的任意一个关键活动
七、查找
基本概念
- 查找:在数据集合(查找表)中寻找满足某种条件的数据元素
- 分类
- 静态查找:顺序查找、折半查找、散列查找
- 动态查找:散列查找、二叉平衡树查找、B树查找
- 关键字:数据元素中唯一表示该元素的某个数据项的值
- 平均查找长度:所有查找过程中进行关键字的比较次数的平均值
- 增加查找效率的数据结构都是对于查找表进行规则约束而形成的
静态查找
- 顺序查找
- 定义:在线性表中遍历查找
- 平均查找长度
- ASL成功=(n+1)/2ASL_{成功} =(n+1)/2ASL成功=(n+1)/2
- ASL失败=n+1ASL_{失败} =n+1ASL失败=n+1
- 特点
- 当n平均查找长度较大,效率低
- 对数据元素的存储和有序性没有要求
- 折半查找
- 适用范围:二分查找适用于有序顺序表(存储结构必须支持随机查找)
- 时间复杂度:O(log2n)O(log_2n)O(log2n)
- 分块查找
- 特点
- 块内元素无序,块之间元素有序
- 第一个块中的最大关键小于第二个块中的所有记录的关键字
- 索引表中的每个元素中含有各块中的最大关键字和各块中的第一个元素的地址
- 理想块长度=查找表长度理想块长度 = \sqrt{查找表长度}理想块长度=查找表长度
- 特点
动态查找
- B树
- 定义:是一颗多路平衡查找树,m叉的B树有以下特性
- 绝对平衡:任一层的所有子树均高度相同
- 根节点:子树数量[2,m],关键字数[1,m-1]
- 其他节点子树:[m/2,m],关键字数[(m/2)-1,m-1]
- 每个节点内的关键字有序,左小右大
- 叶子节点均在最后一层
- 节点中孩子个数等于该节点中关键字个数加1
- B树叶子节点中对应查找失败的情况,n个关键字中有n+1中失败的可能性
- B树的高度和磁盘存取顺序成正比,不包括没有信息的叶节点那层
- 基本操作
- 在B树中找节点(通常存储在磁盘中)
- 在节点内找关键字(通常存储在内存中)
- 插入操作
- 插入后节点的关键字个数小于m,可以直接插入
- 插入后检查被插入节点内的关键字个数,当插入后的节点关键字个数大于m-1时,必须对节点进行分裂,左部分放在原节点中,中间位置元素插入到原节点的父节点,右部分放到新节点中
- 删除操作
- 直接删除:所在节点的关键字个数大于等于m/2
- 兄弟够借:被删除节点由父顶点代替,父顶点由被借顶点代替
- 兄弟不够借:父亲顶点下来与兄弟合并
- B+树
- 具有n个关键字的节点只含有n棵子树,即每个关键字对应一颗子树
- 内部节点关键字个数n的范围是m/2≤n≤mm/2\le n \le mm/2≤n≤m,根节点为1≤n≤m1\le n \le m1≤n≤m
- 只有叶子节点包含信息(全部节点),非叶节点只是索引作用,存盘存储块小,读入内存快
- 查找过程,无论成功与否,每次查找都是一条从根节点到叶子节点的路径
- 定义:是一颗多路平衡查找树,m叉的B树有以下特性
- 散列表(哈希表)
- 散列函数:将查找表中关键字映射成对应地址的函数,但是可能将多个关键字映射到同一个地址发生冲突
- 哈希表:建立关键字和存储地址直接的直接映射关系
- 常用散列函数
- 直接定址法:简单算数运算进行映射,简单不会产生冲突,只适合关键字分布基本连续的情况,空位较多会造成存储空间的浪费
- 除留余数法:选取小于等于元素个数的最大质数,进行取余运算。冲突主要取决于选取的余数
- 数字分析法:选取数码分布较为均匀的其中几位作为散列地址(手机号)
- 平方取中法:取关键字平方值的中间几位作为散列地址
- 开放定址法处理冲突的方式
定义:存放新表项的空闲地址既向它的同义词表项开放,又向它的非同词表项开放
缺点:开放定址法不能随便删除元素,可以给要删除的元素做一个标记,进行逻辑删除,防止中断查找路径- 线性探测法:冲突发生时,顺序查看下一个表中的单元直到找到一个空闲单元。但是可能造成相邻散列地址的堆积,导致查找效率下降
- 平方探测法:每次查找的步数呈指数增长,不能探测所有单元,但是可以探测一半
- 再散列法:使用两个散列函数,当通过第一个散列函数得到的地址发生冲突时,再利用第二个散列函数
- 伪随机序列法:出现碰撞时使用生成的伪随机序列作为增量
- 拉链法
顺序数组作为散列后的地址,数组中的每个数据元素指向发生冲突的同义词链表
八、排序
- 算法的稳定性:关键字相同的两个元素在排序后顺序不变
插入排序
- 直接插入排序
- 基本思想
对于n个元素的序列,分为有序表和无序表,每次从后部的无序表中选择第一个元素和有序表中的元素从后向前进入比较,若有序表中选择元素比无序表中选择元素大,则有序表元素后移,直到找到无序表中选择元素要插入的位置 - 哨兵的作用:在进行大量数据进行比较时,在数据边界放置特别的元素标注,避免每次比较是否越界而浪费时间
- 空间复杂度:O(1) 时间复杂度:O(n2)O(n^2)O(n2)
- 基本思想
- 折半插入排序
- 基本思想
适合整体有序的顺序表,使用折半查找确定插入位置,从而统一移动元素 - 时间复杂度仍然为O(n2)O(n^2)O(n2),但是对于数据量不大的排序表往往可以表现出较好的性能
- 基本思想
- 希尔排序(缩小增量排序)
- 基本思想
取一个小于n的步长d,把表中的全部记录分成d组,将待排序表中d的倍数的数据元素放在同一组,在组内进行直接插入排序。再取更小的步长进行排序,直到所有记录放在同一组中。 - 空间复杂度:O(1) 时间复杂度:O(n1.3)O(n^{1.3})O(n1.3),但仅适用顺序存储
- 基本思想
交换排序
- 冒泡排序
基本思想:按逆序或顺序两两比较相邻的元素,按升序或降序进行交换- 产生的子序列是全局有序的,即每一次排序会产生一个最终结果元素
- 空间复杂度:O(1) 时间复杂度:O(n2)O(n^2)O(n2)
- 快速排序
- 基本思想
- 选择基准:在待排序列中,按照某种方式挑出一个元素,作为 “基准”(pivot)
- 分割操作:以该基准在序列中的实际位置,把序列分成两个子序列。此时,在基准左边的元素都比该基准小,在基准右边的元素都比基准大
- 递归地对两个序列进行快速排序,直到序列为空或者只有一个元素。
- 特点
- 不产生有序子序列,但每次排序后会将基准元素放到最终位置上
- 每次排序划分子区间越相近越能发挥快排优势
- 空间复杂度:O(log2n)O(log_2n)O(log2n) 时间复杂度:O(nlogn)O(nlogn)O(nlogn)
- 基本思想
选择排序
- 简单选择排序
- 基本思想:分成有序区和无序区,每次从无序区中找一个最小的放到有序区的最后一个
- 空间复杂度:O(1) 时间复杂度:O(n2)O(n^2)O(n2)
- 堆排序
- 堆分类
- 大根堆:在完全二叉树中,所有根均大于左右子树的值
- 小根堆:在完全二叉树中,所有根均小于左右子树的值
- 堆的建立
- 检查所有非终端节点,顺序存储前i,i≤n/2i \le n/2i≤n/2
- 从最大分支节点第i个开始,与其顺序表中左孩子2i和右孩子2i+1进行比较,交换最大值,使该分支节点满足大根堆
- 指针前移指向下一个分支节点,重复上述过程,时间复杂度为O(n)
- 堆的插入
先将新节点放在堆末端,再对这个新节点执行调整操作,与父节点不断比较向上调整到最后位置 - 堆的删除
用堆底部元素进行填充,然后该元素进行下坠比较 - 堆的应用
- 可以用堆实现优先队列
- 算法性能
- 空间复杂度:O(1)O(1)O(1) 时间复杂度:O(nlog2n)O(nlog_2n)O(nlog2n)
- 堆分类
特殊排序
- 归并排序
- 定义:将n个有序表组合在一起形成新的有序表,通常n为2称为2路归并算法
- 算法性能
- 空间复杂度:O(n)O(n)O(n) 时间复杂度:O(nlog2n)O(nlog_2n)O(nlog2n)
- 基数排序
- 不基于比较和移动,而是基于关键字各位的大小进行排序
- 算法性能
- 空间复杂度:O(r)O(r)O(r) ,r表示辅助队列个数
- 时间复杂度:O(d(n+r))O(d(n+r))O(d(n+r)),共接进行d躺分配和收集,一趟分配需要O(n),一趟收集需要O®
内部排序算法比较
- 直冒简希,快堆二基
算法种类 | 最好情况 | 平均情况 | 最坏情况 | 空间复杂度 | 是否稳定 |
---|---|---|---|---|---|
直接插入排序 | O(n)O(n)O(n) | O(n2)O(n^2)O(n2) | O(n2)O(n^2)O(n2) | O(1)O(1)O(1) | 是 |
冒泡排序 | O(n)O(n)O(n) | O(n2)O(n^2)O(n2) | O(n2)O(n^2)O(n2) | O(1)O(1)O(1) | 是 |
简单选择排序 | O(n2)O(n^2)O(n2) | O(n2)O(n^2)O(n2) | O(n2)O(n^2)O(n2) | O(1)O(1)O(1) | 否 |
希尔排序 | 适合大规模数据 | O(n1.3)O(n^{1.3})O(n1.3) | O(n2)O(n^2)O(n2) | O(1)O(1)O(1) | 否 |
快速排序 | O(nlog2n)O(nlog_2n)O(nlog2n) | O(nlog2n)O(nlog_2n)O(nlog2n) | O(n2)O(n^2)O(n2) | O(log2n)O(log_2n)O(log2n) | 否 |
堆排序 | O(nlog2n)O(nlog_2n)O(nlog2n) | O(nlog2n)O(nlog_2n)O(nlog2n) | O(nlog2n)O(nlog_2n)O(nlog2n) | O(1) | 否 |
2路归并排序 | O(nlog2n)O(nlog_2n)O(nlog2n) | O(nlog2n)O(nlog_2n)O(nlog2n) | O(nlog2n)O(nlog_2n)O(nlog2n) | O(n)O(n)O(n) | 是 |
基数排序 | O(d(n+r))O(d(n+r))O(d(n+r)) | O(d(n+r))O(d(n+r))O(d(n+r)) | O(d(n+r))O(d(n+r))O(d(n+r)) | O(r)O(r)O(r) | 是 |
- 当n比较小或关键字基本有序的时候,使用直接插入排序。
- 当n比较大,使用O(nlog2n)O(nlog_2n)O(nlog2n)的排序算法
- 快速排序,排序关键字随机分布时,性能表现最好
- 堆排序,辅助空间少,不会出现最坏情况
- 归并排序,排序稳定,不交换相等数据元素
- 数据元素本身信息量比较大,可以使用简单选择排序,减少数据元素的移动次数。或者使用链表作为存储结构。
- n很大且关键字位数少可以分解,可以采用基数排序
🚩点此跳转到首行↩︎
参考博客
- 王道考研数据结构
相关文章:
【数据结构】基础知识总结
系列综述: 💞目的:本系列是个人整理为了数据结构复习用的,由于牛客刷题发现数据结构方面和王道数据结构的题目非常像,甚至很多都是王道中的,所以将基础知识进行了整理,后续会将牛客刷题的错题一…...
宣布推出 .NET 社区工具包 8.1!
我们很高兴地宣布 .NET Community Toolkit 8.1 版正式发布!这个新版本包括呼声很高的新功能、bug 修复和对 MVVM 工具包源代码生成器的大量性能改进,使开发人员在使用它们时的用户体验比以往更好! 就像在我们之前的版本中一样,我…...
ChatGPT解开了我一直以来对自动化测试的疑惑
目录 前言 与ChatGPT的对话 什么是自动化测试,我该如何做到自动化测试,或者说需要借助什么工具可以做到自动化测试? 自动化测试如何确保数据的准确性 自动化测试是怎么去验证数据的 如何通过断言验证数据 自动化测试有哪些验证工具可以验证数据 总结 前言…...
十大经典排序算法(上)
目录 1.1冒泡排序 1. 算法步骤 3.什么时候最快 4. 什么时候最慢 5.代码实现 1.2选择排序 1. 算法步骤 2. 动图演示 3.代码实现 1.3 插入排序 1. 算法步骤 2. 动图演示 3. 算法实现 1.4 希尔排序 1. 算法步骤 2. 动图演示 3.代码实现 1.5 归并排序 1. 算法步骤 2…...
如何从 MySQL 读取 100w 数据进行处理
文章目录 场景常规查询流式查询MyBatis 流式查询接口非流式查询和流式查询区别游标查询场景 大数据量操作的场景大致如下: 1、 数据迁移; 2、 数据导出; 3、 批量处理数据; 在实际工作中当指定查询数据过大时,我们一般使用分页查询的方式一页一页的将数据放到内存处理。…...
【数据降维-第2篇】核主成分分析(KPCA)快速理解,及MATLAB实现
一篇介绍了PCA算法的快速理解和应用,本章讲一下KPCA。KPCA方法与PCA方法一样,是有着扎实的理论基础的,相关理论在论文上以及网络上可以找到大量的材料,所以这篇文章还是聚焦在方法的快速理解以及应用上,此外还会对同学…...
Python+ChatGPT实战之进行游戏运营数据分析
文章目录一、数据二、目标三、解决方案1. DAU2. 用户等级分布3. 付费率4. 收入情况5. 付费用户的ARPU最近ChatGPT蛮火的,今天试着让ta写了一篇数据分析实战案例,大家来评价一下!一、数据 您的团队已经为您提供了一些游戏数据,包括…...
Java每日一练(20230313)
目录 1. 字符串统计 ★ 2. 单词反转 ★★ 3. 俄罗斯套娃信封问题 ★★★ 🌟 每日一练刷题专栏 C/C 每日一练 专栏 Python 每日一练 专栏 Java 每日一练 专栏 1. 字符串统计 编写一个程序,对于输入的一段英语文本,可以统计&#…...
国内ChatGPT日趋成熟后,可以优先解决的几个日常小问题
现在ChatGPT的发展可谓如日中天,国内很多大的公司例如百度、京东等也开始拥抱新技术,推出自己的应用场景,但可以想象到的是,他们必定利用这个新技术在巩固自己的现有应用场景,比如某些客服,你都不用想&…...
业内人士真心话,软件测试是没有前途的,我慌了......
我在测试行业爬模滚打7年,从点点点的功能测试到现在成为高级测试,工资也翻了几倍。个人觉得,测试的前景并不差,只要自己肯努力。 我刚出来的时候是在鹅厂做外包的功能测试,天天点点点,很悠闲,点…...
哈佛与冯诺依曼结构
1. 下图是典型的冯诺依曼结构 2. CPU分为三部分:ALU运算单元,CU控制单元,寄存器组。 3. 分析51单片机为何能使用汇编进行编程 51指令集(Instruction Set)是单片机CPU能够执行的所有指令的集合。在编写51单片机程序时&a…...
传输安全HTTPS
为什么要有 HTTPS 为什么要有 HTTPS?简单的回答是:“因为 HTTP 不安全”。HTTP 怎么不安全呢? 通信的消息会被窃取,无法保证机密性(保密性):由于 HTTP 是 “明文” 传输,整个通信过…...
Docker--(六)--Docker资源限制
前言系统压力测试Cpu资源限制Mem资源限制IO 资源限制【扩展】 1.前言 在使用 Docker 运行容器时,一台主机上可能会运行几百个容器,这些容器虽然互相隔离,但是底层却使用着相同的 CPU、内存和磁盘资源。如果不对容器使用的资源进行限制&#x…...
消息队列总结及案例
文章目录python内置队列先进先出的队列Queue分布式队列rabbitmqrocketmqredis list 队列python内置队列 标准库queue提供Queue队列、LifoQueue栈、PriorityQueue优先级队列用于单机的生产者、消费者缓冲队列; 生产者,生产消息的进程或线程;…...
通过WiFi连接adb调试
通过WiFi连接adb调试 解决 cannot connect to 192.168.1.136:5555: 由于目标计算机积极拒绝,无法连接。 (10061) 解决办法1 (Windows下cmd环境执行) 1.连接USB数据线,打开USB调试 使用windows的“运行”命令行方式:&a…...
【蓝桥杯-筑基篇】常用API 运用(1)
🍓系列专栏:蓝桥杯 🍉个人主页:个人主页 目录 🍍1.输入身份证,判断性别🍍 🍍2.输入英语句子,统计单词个数🍍 🥝3.加密解密🥝 🌎4.相邻重复子串…...
想要成为高级网络工程师,只需要具备这几点
首先,成为高级网络工程师的目的,就是为了搞钱。高级网络工程师肯定是不缺钱的,但成为高级网络工程师你一定要具备以下几点:第一 心态作为一个高级网工,首先你必须情绪要稳定,在碰到重大故障的时候不慌&…...
c++ 每日十问3-处理数据
1.为什么 C有多种整型? 解析: C语言中包含多种整数类型,主要包括 short、int、long 和 long long 这4种,每一种还分别包含有符号类型和无符号类型(unsigned)。此外,char 类型也可以看作一种小整数类型。C语言中这些整数类型的主要区别在于存…...
【MySQL】实验一 数据定义
目录 1. 表定义:创建工程项目表 2. 表定义:创建供应商表 3. 表定义:创建供应情况表 4. 表定义:创建零件表 5. 表定义:创建student表 6. 表定义:创建course表 7. 表定义:创建sc表 8.…...
17.电话号码的字母组合(深度递归遍历解决经典老题)
前文C深度递归遍历解决"电话号码的字母组合问题",本题考察的比较全面,考察到vector的使用,深度遍历以及递归的熟练度,希望能对铁子们有所帮助一,题目链接:https://leetcode.cn/problems/letter-c…...
Python 基础教程【1】:Python介绍、变量和数据类型、输入输出、运算符
本文已收录于专栏🌻《Python 基础》文章目录1、Python 介绍2、变量和数据类型2.1 注释的使用2.2 变量以及数据类型2.2.1 什么是变量?2.2.2 怎么给变量起名?2.2.3 变量的类型🎨 整数 int🎨 浮点数(小数&…...
【RPC】Apache Thrift系列详解 - 概述与入门
文章目录前言正文Thrift的技术栈Thrift的特性(一) 开发速度快(二) 接口维护简单(三) 学习成本低(四) 多语言/跨语言支持(五) 稳定/广泛使用Thrift的数据类型Thrift的协议Thrift的传输层Thrift的服务端类型Thrift入门示例(一) 编写Thrift IDL文件(二) 新建Maven工程总结前言 Th…...
class03:MVVM模型与响应式原理
目录一、MVVM模型二、内在1. 深入响应式原理2. Object.entries3. 底层搭建一、MVVM模型 MVVM,即Model 、View、ViewModel。 Model > data数据 view > 视图(vue模板) ViewModel > vm > vue 返回的实例 > 控制中心, 负责监听…...
[Spring学习]08 @Resource和@Autowired注解的区别
目录前言一、Resource和Autowired注解的身世1、Resource注解2、Autowired注解3、常见的三种依赖注入方式及区别1. Filed注入2. Setter注入3. Constructor注入4. 三种依赖注入方式的区别二、Resource和Autowired注解的区别三、Resource和Autowired注解的推荐用法前言 当我们在属…...
前端开发神器VS Code安装教程
✅作者简介:CSDN一位小博主,正在学习前端 📃个人主页:白月光777的CSDN博客 💬个人格言:但行好事,莫问前程 安装VS CodeVS Code简介VS Code安装VS Code汉化结束语💡💡&…...
【Hive进阶】-- Hive SQL、Spark SQL和 Hive on Spark SQL
1.Hive SQL 1.1 基本介绍概念Hive由Facebook开发,用于解决海量结构化日志的数据统计,于2008年贡献给 Apache 基金会。Hive是基于Hadoop的数据仓库工具,可以将结构化数据映射为一张表,提供类似SQL语句查询功能本质:将Hi…...
搭建自己的直播流媒体服务器SRS,以及SRS+OBS直播推拉流使用及配置
一、前言 目前,全球直播带货什么的,成为主流,那如何自己搭建一个直播服务器呢。首先需要一个流媒体服务器,搭建流媒体有很多种方式,如下: 流媒体解决方案 Live555 (C)流媒体平台框…...
Node.js-----使用express写接口
使用express写接口 文章目录使用express写接口创建基本的服务器创建API路由模块编写GET接口编写POST接口CROS跨域资源共享1.接口的跨域问题2.使用cros中间件拒绝跨域问题3.什么是cros4.cros的注意事项5.cros请求的分类JSONP接口1.回顾jsonp的概念和特点2.创建jsonp接口的注意事…...
【Linux修炼】16.共享内存
每一个不曾起舞的日子,都是对生命的辜负。 共享内存一.共享内存的原理二.共享内存你的概念2.1 接口认识2.2演示生成key的唯一性2.3 再谈key三.共享资源的查看3.1 如何查看IPC资源3.2 IPC资源的特征3.3 进程之间通过共享内存进行关联四.共享内存的特点五.共享内存的内…...
JAVA进阶 —— Stream流
目录 一、 引言 二、 Stream流概述 三、Stream流的使用步骤 1. 获取Stream流 1.1 单列集合 1.2 双列集合 1.3 数组 1.4 零散数据 2. Stream流的中间方法 3. Stream流的终结方法 四、 练习 1. 数据过滤 2. 数据操作 - 按年龄筛选 3. 数据操作 - 演员信息要求…...
宁波网站怎么建设/搜索引擎收录查询工具
大家好,我是小鸭酱,博客地址为:http://www.cnblogs.com/xiaoyajiang pyAudioAnalysis是一个音频分析python库,用于Feature Extraction, Classification, Segmentation 和Applications,其github见 https://github.com/t…...
在家帮别人做网站赚钱/外包公司是什么意思
文件系统类型 在windows中我们常见的磁盘格式有fat16、fat32和ntfs。但是windows的文件管理显得有些赘余,为打开一个文件需要打开n个地方,在一个角落里找。而且windows本身对于其他系统的文件格式就更差了,没有听说在windows里打开ext3或者ma…...
酒店为什么做网站/网站快速优化排名app
基础性质概念1)历史发展久,与java同期2)版本3.X 2.7lib多 常用3)脚本语言,不需要编译4)解释性语言,读一行解释一行5) 运行性能较差,没有ruby差 哈哈哈哈6) lib(库文件)多,声音视频,数据挖掘,…...
云南人才招聘网/网站seo如何优化
几何着色器,也就是根据顶点着色器输出的数据进行处理,比如让他们衍生啥的,反正就是处理 关于几何链接着色器 geometryShader glCreateShader(GL_GEOMETRY_SHADER); glShaderSource(geometryShader, 1, &gShaderCode, NULL); glCompileSh…...
东莞建设网站推广公司地址/各种推广平台
http://blog.csdn.net/jay14/article/details/54074553 TLB原理 转载于:https://www.cnblogs.com/wangdgy/p/8465574.html...
wordpress 1 s/seo是一种利用搜索引擎的
1、修改忽略文件权限 git config core.filemode false 2、查看配置 git config --list 发现core.filemodefalse表明成功...