DS:二叉树的链式结构及实现

                                                    创作不易,友友们给个三连吧!!

一、前言

       前期我们解释过二叉树的顺序结构(堆)为什么比较适用于完全二叉树,因为如果用数组来实现非完全二叉树,那么数组的中间部分就可能会存在大量的空间浪费

      而二叉树的链式结构即用链表结构来存储二叉树,这里就没有什么限制了,所有的二叉树都可以用链式结构来存储,因为链式结构存在两个指针分别指向自己的左右孩子,无论是少了左孩子还是少了右孩子,只需要让相应的指针指向NULL就可以了。

       虽然链式结构可以表示所有类型的二叉树,但是由于二叉树本身存储数据的价值并不大(链表、顺序表远远优于二叉树),所以实现增删查改是没有太大意义的,学习链式二叉树真正的意义是学会如何去控制、遍历二叉树的结构,为我们后期学习搜索二叉树做好铺垫。而以下的学习中要重点理解二叉树中的递归思想和分治思想 !

二、链式二叉树的实现

2.1 节点结构体的创建

      创建的方式和单链表的很相似,唯一的区别就是要有两个指针,一个指向自己的左孩子,一个指向自己的右孩子!

typedef int BTDataType;//方便我们后面要存储其他类型的数据的时候方便修改!
typedef struct BinaryTreeNode
{
	BTDataType data;//二叉树节点的数据域
	struct BinaryTreeNode* left;//左孩子
	struct BinaryTreeNode* right;//右孩子
}BTNode;

2.2 二叉树节点的创建

      二叉树节点构建方式和单链表节点的构建方式相同!

BTNode* BuyNode(BTDataType x)
{
	BTNode* newnode = (BTNode*)malloc(sizeof(BTNode));
	if (newnode == NULL)//如果创建不成功
	{
		perror("malloc fail");
		exit(1);
	}
	//如果创建成功
	newnode->data = x;
	newnode->left = newnode->right = NULL;
	return newnode;
}

2.3 前序遍历

为了能够深入前中后续遍历,我们先手动创建一个二叉树,后面都按照这个二叉树来研究

 下面我先给出前中后续的遍历结果,再逐个分析,要注意,没有箭头默认指向空!

 分析前序:

       前序的顺序是根、左子树、右子树:首先1是根,接着访问1的左子树2,2也是根,再访问2的左子树3,3也是根,再访问3的左子树N,接着访问3的右子树N,,回归到2,对于2来说,他的左子树已经访问完了,然后访问右子树N,回归1,对于1来说,他的左子树已经访问完了,该访问他的右子树4了,4是根,接着访问4的左子树5,5也是根,接着访问5的左子树N,再访问5的右子树N,然后回归5,对于4来说,左子树访问完了,接着访问6,6的左子树是N,然后访问6的右子树N。此时所有节点都访问完了。

      大家可以看看上面的图我画的框框,1这个根右边的2 3 N N N  和4 5 N N 6 N N 分别表示1的左右子树,而对于2这个根来说,3 N N  和  N分别代表2的左右子树,对于4这个根来说,5 N N和6 N N分别代表4的左右子树,对于3、5、6 这三个根来说,他们的左右子树都是N和N。

       那么怎么写前序遍历的代码呢?根据上一段的思路我们可以发现,对于根1来说,他的左子树是以2为根的树,他的右子树是以4为根的树,而对于2来说,他的左子树是以3为根的树,右子树是N,对于4来说,他的左子树是以5为根的树,他的右子树是以6为根的树, 而对于3、5、6来说,他们其实也是根,左右子树都是N。所以将问题转化为递归问题。

void PreOrder(BTNode* root)
{
	if (root == NULL)
	{
		printf("N ");
		return;
	}
	printf("%d ", root->data);
	PreOrder(root->left);
	PreOrder(root->right);
}

画一下递归展开图:

时间复杂度是O(N)  因为每个节点都遍历了一次

空间复杂度也是O(h) 通过上图可以看出,当开辟h个的函数栈帧后,后续的空间都是在前一个空间释放后复用的,所以最多同时只有h个栈帧空间被开辟,根据空间可以重复利用的特点,空间复杂度是o(N)!

中序遍历和后序遍历的方法是一样的,后面就不分析了

2.4 中序遍历

void InOrder(BTNode* root)
{
	if (root == NULL)
	{
		printf("N ");
		return;
	}
	InOrder(root->left);
	printf("%d ", root->data);
	InOrder(root->right);
}

2.5 后序遍历

void PostOrder(BTNode* root)
{
	if (root == NULL)
	{
		printf("N ");
		return;
	}
	PostOrder(root->left);
	PostOrder(root->right);
	printf("%d ", root->data);
}

2.6 层序遍历

       层序遍历是一层一层遍历,之前无论是前序、中序、还是后序遍历,都是根据父子关系(父亲会指向自己的左右孩子)转化成递归问题去遍历的, 但是链式二叉树的指针指向并不指向兄弟节点,所以这边如果要一层一层遍历,需要用到队列。

     选择队列的原因是利用队列队头出队尾进的特点,下面进行分析:

Queue.h

void LevelOrder(BTNode* root)
{
	Queue q;
	QueueInit(&q);
	if (root)//如果不为NULL,就进队列
		QueuePush(&q, root);
	while (!QueueEmpty(&q))//队列为空,就是遍历完成
	{
		BTNode* front = QueueFront(&q);//让每一次循环过来的节点变成根部,再去访问左右子树
		QueuePop(&q);//记录完一个元素就出队列
		printf("%d ", front->data);
		if(front->left)
			QueuePush(&q, front->left);//左节点不为空,进队列
		if (front->right)
			QueuePush(&q, front->right);//右节点不为空,进队列
	}
	printf("\n");
	//遍历完了就可以销毁队列了
	QueueDestory(&q);
}

2.7 二叉树节点个数

我们先按照按照军棋里的模式来解释分治思想:

       假设军长要统计总人数,有两个方法,一个是军长一级一级去视察数人,这显然是效率比较低的, 而另一个方法就是分而治之——利用自己的职权将任务移交给下级,军长把任务分配给两个旅长,然后旅长统计过来的人数加上自己就是全军人数,旅长又分配给连长,将两个连长统计的人数加上自己就是全旅人数,以此类推……

     

int BinaryTreeSize(BTNode* root)
{
	return root == NULL ? 0 : BinaryTreeSize(root->left) + BinaryTreeSize(root->right) + 1;
}

2.8 二叉树叶子节点个数

    还是利用分治思想,叶子节点的特征是左右子树都为NULL,我们还是按照分治思想将这个任务拆分下去

int BinaryTreeLeafSize(BTNode* root)
{
	if (root == NULL)
		return 0;
	if (root->left == NULL && root->right == NULL)
		return 1;
	return BinaryTreeLeafSize(root->left) + BinaryTreeLeafSize(root->right);
}

2.9 二叉树第k层节点

还是利用分治思想,将求第k层节点转化为求左子树的k-1层节点加上右子树的k-1层节点。

int BinaryTreeLevelKSize(BTNode* root,int k)
{
	if (root == NULL)
		return 0;
	if (k == 1)
		return 1;
	return BinaryTreeLevelKSize(root->left,k-1) + BinaryTreeLevelKSize(root->right,k-1);
}

2.10 二叉树的高度

还是利用分治思想,求高度转化为比较左右子树的大小再+1

​​int BinaryTreeHeight(BTNode* root)
{
	if (root == NULL)
		return 0;
	return BinaryTreeHeight(root->left) > BinaryTreeHeight(root->right) ? BinaryTreeHeight(root->left) + 1 : BinaryTreeHeight(root->right) + 1;
}

 其实这样写是有问题的!!!因为每次我递归比较完后,并没有记录最大值,所以导致每次递归的时候最大值都得再重新递归一次!!如果树节点高度特别高的话,就很可能会出现这样的问题。

改进版:

int BinaryTreeHeight(BTNode* root)
{
	if (root == NULL)
		return 0;
	int LeftHeight = BinaryTreeHeight(root->left);
	int RightHeight = BinaryTreeHeight(root->right);
	return LeftHeight > RightHeight ? LeftHeight + 1 : RightHeight + 1;
}

 所以我们在递归时使用三目表达式要注意:如果比较的过程已经递归一次了,比较的结果就不要再去递归了!!可以及时的记录值!!

2.11 二叉树查找值为x的节点

还是利用分治思想,转化为在左子树和右子树中找

BTNode* BTFind(BTNode* root, BTDataType x)
{
	if (root == NULL)
		return NULL;
	if (root->data == x)
		return root;
	BTNode* ret1 = BTFind(root->left, x);
	if (ret1)
		return ret1;
	BTNode* ret2 = BTFind(root->right, x);
	if (ret2)
		return ret2;
	return NULL;
}

2.12 通过前序遍历数组构建二叉树

     之前我们构建二叉树是一个节点一个节点去手动连接,这次我们尝试自己利用前序遍历数组去构建二叉树,比如abc##de#g##f###

BTNode* BTCreate(char*arr,int*pi)
{
	if (arr[*pi] == '#')
	{
		(*pi)++;
		return NULL;
	}
	BTNode* root = BuyNode(arr[*pi]);
	(*pi)++;
	root->left = BTCreate(arr, pi);
	root->right = BTCreate(arr, pi);
	return root;
}

 递归展开图:

2.13 二叉树销毁

       二叉树要销毁,就要遍历每个节点,因为我们如果先销毁根,那么就很可能找不到他的左子树和右子树了,除非销毁根之前记录一下,但是这样比较麻烦,所以我们选择后序遍历,先销毁左子树,再销毁右子树,再销毁根。

void BTDestory(BTNode* root)
{
	if (root == NULL)
		return;
	BTDestory(root->left);
	BTDestory(root->right);
	free(root);
	//root = NULL;没用
}

注意:这里的参数是一级指针,而root本身也是指针,所以在这里对root置NULL,是没有意义的,所以要使用的话,要在main函数中手动置空(这里不用二级指针的原因是为了保持接口一致性)!!

2.14 判断二叉树是否是完全二叉树

         完全二叉树的特点就是:前h-1层是满的,最后一层从前往后要到最后才会访问到NULL,所以们可以分析一下他的特点:

bool BTComplete(BTNode* root)
{
	Queue q;
	QueueInit(&q);
	if (root)
		QueuePush(&q, root);
	while (!QueueEmpty(&q))
	{
		BTNode* front = QueueFront(&q);//记录该结果,然后再pop出去
		QueuePop(&q);

		if (front == NULL)
			break;
		QueuePush(&q, root->left);
		QueuePush(&q, root->right);
	}
	//跳出循环后,检查后面的节点是不是还有非空节点,有的话就是非完全二叉树
	while (!QueueEmpty(&q))
	{
		BTNode* front = QueueFront(&q);//记录该结果,然后再pop出去
		QueuePop(&q);
		if (front)
		{
			QueueDestory(&q);//队列一定别忘了销毁
			return false;
		}
		QueueDestory(&q);
		return true;
	}
}

2.15 判断两颗二叉树是否完全相同

我们还是用分治思想,前序遍历比较

bool IsBTSame(BTNode* root1, BTNode* root2)
{
	if (root1 == NULL && root2 == NULL)
		return true;
	if (root1 == NULL || root2 == NULL)
		return false;
	//此时节点肯定不为NULL了,可以解引用找到val
	if (root1->data != root2->data)
		return false;
	//不相等的话只能去找自己的左右子树,左右子树都返回true最后结果才能返回true
	return IsBTSame(root1->left, root2->left) && IsBTSame(root1->right, root2->right);
}

三、链式二叉树实现的全部代码

前两个文件只是因为层序遍历和判断完全二叉树会用到,重点还是看后面两个文件

3.1 queue.h

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>
#include<assert.h>
struct BinaryTreeNode;//为了防止嵌套调用,这里声明一下
typedef struct BinaryTreeNode* QDatatype;//方便后面修改存储数据的数据类型
typedef struct QueueNode//队列结点的数据结构
{
	QDatatype data;//存储数据
	struct QueueNode* next;
}QNode;

typedef struct Queue
{
	QNode* phead;//指向队头,用于出队(头删)
	QNode* ptail;//指向队尾,用于入队(尾插)
	int size;//记录有效元素个数
}Queue;//创建一个队列相关结构体
void QueueInit(Queue* pq);//队列的初始化
void QueuePush(Queue* pq, QDatatype x);//队列的入队(尾插)
void QueuePop(Queue* pq);//队列的出队(头删)
QDatatype QueueFront(Queue* pq);//获取队列头部元素
QDatatype QueueBack(Queue* pq);//获取队列尾部元素
int QueueSize(Queue* pq);//获取队列中有效元素个数
bool QueueEmpty(Queue* pq);//判断队列是否为空
void QueueDestory(Queue* pq);//队列的销毁

3.2 queue.c

#include"Queue.h"
void QueueInit(Queue* pq)
{
	assert(pq);//判断传的是不是空指针
	pq->phead = pq->ptail = NULL;
	pq->size = 0;//因为队列不像栈一样,有一个top表示栈顶元素的下标
	//所以如果我们想知道这个队列的有效数据个数,就必须遍历队列
	//由于其先进先出的特性,我们默认只能访问到头元素和尾元素
	//所以必须访问一个头元素,就出队列一次,这样才能实现遍历
	//但是这样的代价太大了,为了方便,我们直接用size
}

void QueuePush(Queue* pq, QDatatype x)
{
	assert(pq);
    //入队必须从队尾入!
	QNode* newnode = (QNode*)malloc(sizeof(QNode));//创建一个新节点
	if (newnode==NULL)//如果新节点申请失败,退出程序
	{
		perror("malloc fail");
	}
	//新节点创建成功,给新节点初始化一下
	newnode->data = x;
	newnode->next = NULL;
	//开始入队
	//如果直接尾插的话,由于会用到ptail->next,所以得考虑队列为空的情况
	if (pq->ptail== NULL)//如果为空,直接把让新节点成为phead和ptail
	{
		//按道理来说,如果ptail为空,phead也应该为空
		// 但是有可能会因为我们的误操作使得phead不为空,这个时候一般是我们写错的问题
		//所以使用assert来判断一下,有问题的话会及时返回错误信息
		assert(pq->phead == NULL);
		pq->phead = pq->ptail = newnode;
	}
	else
	{
		pq->ptail->next = newnode;
		pq->ptail = newnode;
	}
	pq->size++;
}

void QueuePop(Queue* pq)
{
	assert(pq);
	//如果队列为空,没有删除的必要
	assert(!QueueEmpty(pq));
	//队列中的出队列相当于链表的头删
	//如果直接头删,那么如果队列只有一个有效数据的话,那么我们将phead的空间释放掉,但是没有将ptail给置空
	//这样会导致ptail成为一个野指针,所以我们需要考虑只有一个节点多个节点的情况
	if (pq->phead->next == NULL)//一个节点的情况,直接将这个节点释放并置空即可
	{
		free(pq->phead);
		pq->phead = pq->ptail = NULL;//置空,防止野指针
	}
	else//多个节点的情况,直接头删

	{
		QNode* next = pq->phead->next;//临时指针记住下一个节点
		free(pq->phead);
		pq->phead = next;//让下一个节点成为新的头
	}
	pq->size--;
}

QDatatype QueueFront(Queue* pq)
{
	assert(pq);
	assert(!QueueEmpty(pq));//队列如果为空,则不可能找得到队列头元素
	//队列不为空的时候,直接返回phead指向的数据
	return pq->phead->data;
}

QDatatype QueueBack(Queue* pq)
{
	assert(pq);
	assert(!QueueEmpty(pq));//队列如果为空,则不可能找得到队尾元素
	//队列不为空的时候,直接返回ptail指向的数据
	return pq->ptail->data;
}

int QueueSize(Queue* pq)
{
	assert(pq);
	return pq->size;
}

bool QueueEmpty(Queue* pq)//链表为空的情况,可以根据容量,也可以根据ptail==NULL&&phead==NULL
{
	assert(pq);
	return pq->ptail == NULL && pq->phead == NULL;
}

void QueueDestory(Queue* pq)
{
	assert(pq);//判断传的是不是空指针
	//要逐个节点释放
	QNode* pcur = pq->phead;
	while (pcur)
	{
		QNode* next = pcur->next;
		free(pcur);
		pcur = next;
	}
	pq->phead = pq->ptail = NULL;
	pq->size = 0;
}

3.3 BT.h

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
#include"Queue.h"
typedef int BTDataType;//方便我们后面要存储其他类型的数据的时候方便修改!
typedef struct BinaryTreeNode
{
	BTDataType data;
	struct BinaryTreeNode* left;
	struct BinaryTreeNode* right;
}BTNode;

BTNode* BuyNode(BTDataType x);
void PreOrder(BTNode* root);//前序遍历
void InOrder(BTNode* root);//中序遍历
void PostOrder(BTNode* root);//后序遍历
void LevelOrder(BTNode* root);//层序遍历
int BTSize(BTNode* root);//二叉树节点个数
int BTLeafSize(BTNode* root);//二叉树的叶子节点个数
int BTLevelKSize(BTNode* root, int k);//二叉树第k层的节点个数
int BTHeight(BTNode* root);//二叉树的高度
BTNode* BTFind(BTNode* root, BTDataType x);//二叉树找值为x的点
BTNode* BTCreate(char* arr,int*pi);//根据前序数组创建二叉树
void BTDestory(BTNode* root);//销毁二叉树
bool BTComplete(BTNode* root);//判断是否完全二叉树
bool IsBTSame(BTNode* root1, BTNode* root2);//判断两个二叉树是否完全相等

3.4 BT.c

#include"BTtree.h"

BTNode* BuyNode(BTDataType x)
{
	BTNode* newnode = (BTNode*)malloc(sizeof(BTNode));
	if (newnode == NULL)//如果创建不成功
	{
		perror("malloc fail");
		exit(1);
	}
	//如果创建成功
	newnode->data = x;
	newnode->left = newnode->right = NULL;
	return newnode;
}

void PreOrder(BTNode* root)
{
	if (root == NULL)
	{
		printf("N ");
		return;
	}
	printf("%d ", root->data);
	PreOrder(root->left);
	PreOrder(root->right);
}

void InOrder(BTNode* root)
{
	if (root == NULL)
	{
		printf("N ");
		return;
	}
	InOrder(root->left);
	printf("%d ", root->data);
	InOrder(root->right);
}

void PostOrder(BTNode* root)
{
	if (root == NULL)
	{
		printf("N ");
		return;
	}
	PostOrder(root->left);
	PostOrder(root->right);
	printf("%d ", root->data);
}

void LevelOrder(BTNode* root)
{
	Queue q;
	QueueInit(&q);
	if (root)//如果不为NULL,就进队列
		QueuePush(&q, root);
	while (!QueueEmpty(&q))//队列为空,就是遍历完成
	{
		BTNode* front = QueueFront(&q);//让每一次循环过来的节点变成根部,再去访问左右子树
		QueuePop(&q);//记录完一个元素就出队列
		printf("%d ", front->data);
		if(front->left)
			QueuePush(&q, front->left);//左节点不为空,进队列
		if (front->right)
			QueuePush(&q, front->right);//右节点不为空,进队列
	}
	printf("\n");
	//遍历完了就可以销毁队列了
	QueueDestory(&q);
}

int BTSize(BTNode* root)
{
	return root == NULL ? 0 : BTSize(root->left) + BTSize(root->right) + 1;
}

int BTLeafSize(BTNode* root)
{
	if (root == NULL)
		return 0;
	if (root->left == NULL && root->right == NULL)
		return 1;
	return BTLeafSize(root->left) + BTLeafSize(root->right);
}

int BTLevelKSize(BTNode* root,int k)
{
	if (root == NULL)
		return 0;
	if (k == 1)
		return 1;
	return BTLevelKSize(root->left,k-1) + BTLevelKSize(root->right,k-1);
}

int BTHeight(BTNode* root)
{
	if (root == NULL)
		return 0;
	int LeftHeight = BTHeight(root->left);
	int RightHeight = BTHeight(root->right);
	return LeftHeight > RightHeight ? LeftHeight + 1 : RightHeight + 1;
}

BTNode* BTFind(BTNode* root, BTDataType x)
{
	if (root == NULL)
		return NULL;
	if (root->data == x)
		return root;
	BTNode* ret1 = BTFind(root->left, x);
	if (ret1)
		return ret1;
	BTNode* ret2 = BTFind(root->right, x);
	if (ret2)
		return ret2;
	return NULL;
}

BTNode* BTCreate(char*arr,int*pi)
{
	if (arr[*pi] == '#')
	{
		(*pi)++;
		return NULL;
	}
	BTNode* root = BuyNode(arr[*pi]);
	(*pi)++;
	root->left = BTCreate(arr, pi);
	root->right = BTCreate(arr, pi);
	return root;
}

void BTDestory(BTNode* root)
{
	if (root == NULL)
		return;
	BTDestory(root->left);
	BTDestory(root->right);
	free(root);
	//root = NULL;没用
}

bool BTComplete(BTNode* root)
{
	Queue q;
	QueueInit(&q);
	if (root)
		QueuePush(&q, root);
	while (!QueueEmpty(&q))
	{
		BTNode* front = QueueFront(&q);//记录该结果,然后再pop出去
		QueuePop(&q);

		if (front == NULL)
			break;
		QueuePush(&q, root->left);
		QueuePush(&q, root->right);
	}
	//跳出循环后,检查后面的节点是不是还有非空节点,有的话就是非完全二叉树
	while (!QueueEmpty(&q))
	{
		BTNode* front = QueueFront(&q);//记录该结果,然后再pop出去
		QueuePop(&q);
		if (front)
		{
			QueueDestory(&q);//队列一定别忘了销毁
			return false;
		}
		QueueDestory(&q);
		return true;
	}
}

bool IsBTSame(BTNode* root1, BTNode* root2)
{
	if (root1 == NULL && root2 == NULL)
		return true;
	if (root1 == NULL || root2 == NULL)
		return false;
	//此时节点肯定不为NULL了,可以解引用找到val
	if (root1->data != root2->data)
		return false;
	//不相等的话只能去找自己的左右子树,左右子树都返回true最后结果才能返回true
	return IsBTSame(root1->left, root2->left) && IsBTSame(root1->right, root2->right);
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

二叉树习题

路径和&#xff1a;不能将叶节点向下扩展一层nullptr来标记这个节点是叶节点 struct TreeNode {int val;TreeNode *left;TreeNode *right;TreeNode() : val(0), left(nullptr), right(nullptr) {}TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}TreeNode(int x, T…

【计算机网络】电子邮件

用户代理 user agent邮件服务器 mail server简单邮件传输协议 SMTP 使用TCP与HTTP对比 HTTP&#xff1a;web服务器向web客户传输文件 SMTP&#xff1a;邮件服务器向另一个邮件服务器传输文件 持续链接 持续链接 拉协议&#xff08;pull protocol&#xff09;由想接收发起 …

顺序表(上)

1.顺序表的概念 顺序表&#xff08;Sequential List&#xff09;是一种基本的数据结构&#xff0c;它是一种线性表的存储结构。线性表是一种数据元素的有限序列&#xff0c;元素之间存在顺序关系。 线性表&#xff1a;线性表&#xff08; linearlist &#xff09;是n个具有相…

StringBuilder

StringBuilder代表可变字符串&#xff0c;相当于一个容器&#xff0c;里面的字符串可以改变&#xff0c;用来操作字符串。此类设计用作StringBuffer替代品。 构造方法&#xff1a; StringBuilder() StringBuilder(String str) 操作方法&#xff1a; 1. append()&#xff1…

爬爬爬——qq模拟登录,古诗文网模拟登录并爬取内容(cookie)

cookie——可以理解为&#xff0c;记录为登录状态。如果在登录一个网站之后&#xff0c;想拿到信息发现404了&#xff0c;就是没有加cookie在这个header里。 下图加了cookie和没有加的对比&#xff08;我是用了selenuim自动化登录的&#xff09;&#xff1a; 下面是加了的 这个…

【C语言】指针的入门篇2,深入理解指针和数组的关系

欢迎来CILMY23的博客喔&#xff0c;本期系列为【C语言】指针的入门篇2&#xff0c;深入理解指针和数组的关系&#xff0c;图文讲解指针和数组关系的知识&#xff0c;带大家理解指针和数组的关系&#xff0c;以及指针数组的用法&#xff0c;感谢观看&#xff0c;支持的可以给个赞…

书生浦语大模型实战营-课程笔记(2)

介绍了一下InternLm的总体情况。 InternLm是训练框架&#xff0c;Lagent是智能体框架。 这个预训练需要这么多算力&#xff0c;大模型确实花钱。 Lagent是智能体框架&#xff0c;相当于LLM的应用。 pip设置 开发机的配置 pip install transformers4.33.1 timm0.4.12 sente…

数据结构——5.5 树与二叉树的应用

5.5 树与二叉树的应用 概念 结点的权&#xff1a;大小可以表示结点的重要性 结点的带权路径长度&#xff1a;从树的根到该结&#xff0c;的路径长度&#xff08;经过的边数&#xff09;与该结点权的乘积 树的带权路径长度&#xff1a;树中所有叶结点的带权路径长度之和(WPL) …

C语言函数(四):递归

目录 1.什么是递归2.递归的限制条件3.递归举例3.1 举例一&#xff1a;求n的阶乘 4.递归与迭代4.1 求第n个斐波那契数 5.递归与循环的选择 1.什么是递归 在学习函数这一章节&#xff0c;递归是每个计算机语言绕不开的知识点&#xff0c;那什么是递归呢&#xff1f; 递归就是一种…

Java入门高频考查基础知识9(银盛15问万字参考答案)

JAVA刷题专栏&#xff1a;http://t.csdnimg.cn/9qscL 目录 一、Springcloud的工作原理 三、注册中心心跳是几秒 四、消费者是如何发现服务提供者的 五、多个消费者调⽤用同⼀接口&#xff0c;eruka默认的分配⽅式是什么 六、springboot常用注解&#xff0c;及其实现 七、…

【C语言】指针的入门篇,深入理解指针和指针变量

欢迎来sobercq的博客喔&#xff0c;本期系列为【C语言】指针的入门篇&#xff0c;深入理解指针和指针变量 图文讲解指针的知识&#xff0c;带大家理解指针和内存的关系&#xff0c;以及指针的用法&#xff0c;感谢观看&#xff0c;支持的可以给个赞哇。 目录 一、内存和地址 二…

【使用IDEA总结】01——新增作者信息、方法参数返回值

[TOC](目录) 1.类新增作者信息 打开IDEA的Settings&#xff0c;Editor->Code Style->File and Code Templates->Includes->File Header&#xff0c;输入以下作者信息&#xff0c;作者名更换为自己的即可&#xff0c;操作如下图所示 /*** Author Linhaipeng* Date…

实现表达式语言

实现表达式语言 考虑使用大量Scriplet代码嵌入Java代码的JSP页面。过度使用Scriptlet代码使JSP页面变得混乱。因此。开发人员难以阅读和调试页面。另外,网页设计师在编辑表示代码时也会遇到问题。为了解决此类问题,开发无脚本的JSP页面受到推崇。 无脚本的代码使JSP页面易于…

Uipath 实现Excel 文件合并

场景描述 某文件夹下有多个相同结构(标题列相同)的Excel 文件&#xff0c;需实现汇总到一个Excel文件。 常见场景有销售明细汇总&#xff0c;订单汇总等。 解决方案 对于非IT 人员则可使用Uipath 新式Excel活动&#xff0c;通过拖拉实现。也可以通过内存表或使用VB脚本&…

【动态规划初识】不同路径问题

每日一道算法题之不同路径问题 一、题目描述二、思路三、C++代码一、题目描述 题目来源:LeetCode 一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为 “Start” )。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为 “Finis…

CTFshow web(php文件上传155-158)

web155 老样子&#xff0c;还是那个后端检测。 知识点&#xff1a; auto_append_file 是 PHP 配置选项之一&#xff0c;在 PHP 脚本执行结束后自动追加执行指定的文件。 当 auto_append_file 配置被设置为一个文件路径时&#xff0c;PHP 将在执行完脚本文件的所有代码后&…

opencv通道分离与合并

void QuickDemo::channels_demo(Mat & image) {std::vector<Mat>mv;//通道分离合并split(image,mv);//原图 指针(Mat)imshow("蓝色", mv[0]);imshow("绿色", mv[1]);imshow("红色", mv[2]); } split(image,mv);//原图 指针(Mat) 这里…

华为OD机试 - 分配土地( Python C C++ JavaGo JS PHP)

题目描述 从前有个村庄&#xff0c;村民们在各种田地上插上小旗子&#xff0c;每个旗子上都标识了一个数字。现在&#xff0c;村民们想要找出一个包含相同数字的最小矩形区域&#xff0c;并将这块土地分配给对村庄做出巨大贡献的村民。我们需要找出这个矩形区域的最大面积。 …

网络原理(3)--以太网协议,DNS

&#x1f495;"Echo"&#x1f495; 作者&#xff1a;Mylvzi 文章主要内容&#xff1a;网络原理(3)–以太网协议,DNS 在网络原理(2)中介绍了网络层中的一个重要的协议–ip协议,网络层关注的通信时的起点和终点,而数据链路层更加"底层"一些,关注的是传输过程…

嵌入式软件设计入门:从零开始学习嵌入式软件设计

&#xff08;本文为简单介绍&#xff0c;个人观点仅供参考&#xff09; 首先,让我们了解一下嵌入式软件的定义。嵌入式软件是指运行在嵌入式系统中的特定用途软件,它通常被用来控制硬件设备、处理实时数据和实现特定功能。与桌面应用程序相比,嵌入式软件需要具备更高的实时性、…