学习笔记---更进一步的双向链表专题~~

这篇具有很好参考价值的文章主要介绍了学习笔记---更进一步的双向链表专题~~。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


目录

1. 双向链表的结构🦊

2. 实现双向链表🐝

2.1 要实现的目标🎯

2.2 创建+初始化🦋

2.2.1 List.h

2.2.2 List.c

2.2.3 test.c

2.2.4 代码测试运行

2.3 尾插+打印+头插🪼

思路分析

2.3.1 List.h

2.3.2 List.c

2.3.3 test.c

2.3.4 代码测试运行

2.4 尾删+头删🐊

2.4.0 思路分析

2.4.1 List.h

2.4.2 List.c

2.4.3 test.c

2.4.4 代码测试运行

2.5 查找数据+pos节点后插入+删除pos节点🦩

2.5.0 思路分析

2.5.1 List.h

2.5.2 List.c

2.5.3 test.c

2.5.4 代码测试运行

2.6 销毁☄️

2.6.0思路分析

1. 一级指针

2.6.1 List.h

2.6.2 List.c

2.6.3 test.c

2.6.4 代码测试运行

2. 二级指针

2.6.1 List.h

2.6.2 List.c

2.6.3 test.c

2.6.4 代码测试运行

2.7 完整代码💦

2.7.1 List.h

2.7.2 List.c

2.7.3 test.c

3. 顺序表和双向链表的分析🍻


1. 双向链表的结构🦊

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


这里的双向链表,准确的说是:带头双向循环链表

这里的“头节点”指的是“哨兵位”哨兵位节点不存储任何有效元素,只是站在这⾥“放哨

的”。

“哨兵位”存在的意义:遍历循环链表避免死循环

注意⚠️

双向链表的每一个节点存储一个有效数据+下一个节点的地址+上一个节点的地址

头节点和尾节点有些特殊:头节点指向的上一个节点的地址是尾节点,尾节点指向的下一个节点的地址是头节点


2. 实现双向链表🐝

2.1 要实现的目标🎯

我们需要多个接口帮助我们实现:创建、一系列具体操作、销毁

具体操作包括:头部/尾部插入数据、头部/尾部删除数据、打印出双向链表、指定节点之后插入数据、删除指定节点的数据、查找指定节点

2.2 创建+初始化🦋

2.2.1 List.h

#include<assert.h>
#include<string.h>
#include<stdbool.h>

typedef int LTDataType;
//创建双向链表的结构体
typedef struct ListNode {
	LTDataType data;
	struct ListNode* prev;
	struct ListNode* next;
}ListNode;

//初始化
ListNode* LTInit();//不用传入参数,直接调用接口返回一个头节点

2.2.2 List.c

#include"List.h"
//初始化
ListNode* LTInit()//不用传入参数,直接调用接口返回一个头节点
{
	//为头节点申请空间
	ListNode* phead = (ListNode*)malloc(sizeof(ListNode));
	//判断开辟是否成功
	if (phead == NULL)
	{
		perror("malloc error!\n");
		return;
	}
	//开辟成功--->初始化头节点
	phead->data = -1;//头节点不存储有效数据,可以任意赋值
	//只有哨兵位的时候,要实现双向链表,不能指向NULL,否则无法双向循环,所以我们指向自己
	phead->prev = phead->next = phead;
	return phead;
}

2.2.3 test.c

#include"List.h"
void ListTest()
{
	ListNode* plist = LTInit();
}
int main()
{
	ListTest();
	return 0;
}

2.2.4 代码测试运行

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


2.3 尾插+打印+头插🪼

思路分析

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


2.3.1 List.h

//在双向链表中不会改变哨兵位,所以这里都可以传一级指针
//尾插
void LTPushBack(ListNode* phead, LTDataType x);

//打印
void LTPrint(ListNode* phead);

//头插
void LTPushFront(ListNode* phead, LTDataType x);

2.3.2 List.c

//在双向链表中不会改变哨兵位,所以这里都可以传一级指针
// 只改变数据,不改变地址

//开辟空间
ListNode* ListBuyNode(LTDataType x)
{
	ListNode* node = (ListNode*)malloc(sizeof(ListNode));
	if (node == NULL)
	{
		perror("malloc error!\n");
		return;
	}
	node->data = x;
	node->next = node->prev = NULL;
	return node;
}

//尾插
void LTPushBack(ListNode* phead, LTDataType x)
{
	assert(phead);//注意哨兵位不能为空
	//申请空间
	ListNode* node = ListBuyNode(x);
	//先处理node的前驱指针和后继指针
	node->prev = phead->prev;
	node->next = phead;
	//再处理之前的尾节点和phead
	phead->prev->next = node;
	phead->prev = node;
}

//打印
void LTPrint(ListNode* phead)
{
	//哨兵位不能改变
	ListNode* cur = phead->next;
	while (cur != phead)//当cur再次指向phead的时候,循环结束
	{
		printf("%d->", cur->data);
		cur = cur->next;
	}
	printf("\n");
}

//头插
void LTPushFront(ListNode* phead, LTDataType x)
{
	assert(phead);//注意哨兵位不能为空
	//申请空间
	ListNode* node = ListBuyNode(x);
	//node插入头节点之后才算头插
	//先处理node的前驱指针和后继指针
	node->prev = phead;
	node->next = phead->next;
	//再处理phead和phead->next
	phead->next->prev = node;
	phead->next = node;
}

2.3.3 test.c

#include"List.h"
void ListTest()
{
	ListNode* plist = LTInit();
	LTPushBack(plist, 1);
	LTPushBack(plist, 2);
	LTPushBack(plist, 3);
	LTPushBack(plist, 4);
	LTPrint(plist);//1 2 3 4 
	LTPushFront(plist, 5);
	LTPrint(plist);//5 1 2 3 4 
}
int main()
{
	ListTest();
	return 0;
}

2.3.4 代码测试运行

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


2.4 尾删+头删🐊

2.4.0 思路分析

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


2.4.1 List.h

//尾删
void LTPopBack(ListNode* phead);

//头删
void LTPopFront(ListNode* phead);

2.4.2 List.c

//尾删
void LTPopBack(ListNode* phead)
{
	//不能为空链表,只有一个哨兵位不能尾删
	assert(phead&&(phead->prev!=phead||phead->next!=phead));
	ListNode* del = phead->prev;//phead->prev就是尾节点
	//先处理del
	del->prev->next = phead;
	//再处理phead
	phead->prev = del->prev;
	free(del);
	del = NULL;
}

//头删
void LTPopFront(ListNode* phead)
{
	//不能为空链表,只有一个哨兵位不能头删
	assert(phead && (phead->prev != phead || phead->next != phead));
	ListNode* del = phead->next;
	del->next->prev = phead;
	phead->next = del->next;
	free(del);
	del = NULL;
}

2.4.3 test.c

#include"List.h"
void ListTest()
{
	ListNode* plist = LTInit();
	LTPushBack(plist, 1);
	LTPushBack(plist, 2);
	LTPushBack(plist, 3);
	LTPushBack(plist, 4);
	LTPrint(plist);//1 2 3 4 
	LTPushFront(plist, 5);
	LTPrint(plist);//5 1 2 3 4 
	LTPopBack(plist);
	LTPrint(plist);//5 1 2 3
	LTPopFront(plist);
	LTPrint(plist);//1 2 3
}
int main()
{
	ListTest();
	return 0;
}

2.4.4 代码测试运行

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


2.5 查找数据+pos节点后插入+删除pos节点🦩

2.5.0 思路分析

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


2.5.1 List.h

//查找数据
ListNode* LTFind(ListNode* phead, LTDataType x);

//pos节点之后插入
void LTPushAfter(ListNode* pos, LTDataType x);

//删除pos节点
void LTErase(ListNode* pos);

2.5.2 List.c

//查找数据
ListNode* LTFind(ListNode* phead, LTDataType x)
{
	assert(phead);
	ListNode* cur = phead->next;
	while (cur!= phead)
	{
		if (cur->data == x)
		{
			return cur;
		}
		cur = cur->next;
	}
	return NULL;
}

//pos节点之后插入
void LTPushAfter(ListNode* pos, LTDataType x)
{
	assert(pos);
	ListNode* node = ListBuyNode(x);
	//node
	node->next = pos->next;
	node->prev = pos;
	//pos
	pos->next = node;
	node->next->prev = node;
}

//删除pos节点
void LTErase(ListNode* pos)
{
	assert(pos);
	pos->prev->next = pos->next;
	pos->next->prev = pos->prev;
	free(pos);
	pos = NULL;
}

2.5.3 test.c

#include"List.h"
void ListTest()
{
	ListNode* plist = LTInit();
	LTPushBack(plist, 1);
	LTPushBack(plist, 2);
	LTPushBack(plist, 3);
	LTPushBack(plist, 4);
	LTPrint(plist);//1 2 3 4 
	LTPushFront(plist, 5);
	LTPrint(plist);//5 1 2 3 4 
	LTPopBack(plist);
	LTPrint(plist);//5 1 2 3
	LTPopFront(plist);
	LTPrint(plist);//1 2 3
	ListNode* find = LTFind(plist, 1);
	/*LTPushAfter(find, 4);*/	
	//LTPrint(plist);//1 4 2 3
	LTErase(find);
	LTPrint(plist);//2 3

}
int main()
{
	ListTest();
	return 0;
}

2.5.4 代码测试运行

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


2.6 销毁☄️

2.6.0思路分析

一开始的初始化,我们直接调用了接口,返回头节点进行初始化。我们没有考虑一级指针还是二级指针的问题。

那么,最后的销毁又该怎么办?是一级指针?还是二级指针?下面我们一一来尝试

1. 一级指针

2.6.1 List.h
//销毁
void LTDestroy(ListNode* phead);

2.6.2 List.c
//销毁
void LTDestroy(ListNode* phead)
{
	assert(phead);
	ListNode* cur = phead->next;
	while(cur!=phead)
	{
		ListNode* next = cur->next;
		free(cur);
		cur = next;
	}
	//注意哨兵位还没有释放
	free(phead);
	phead = NULL;
}

2.6.3 test.c
#include"List.h"
void ListTest()
{
	ListNode* plist = LTInit();
	LTPushBack(plist, 1);
	LTPushBack(plist, 2);
	LTPushBack(plist, 3);
	LTPushBack(plist, 4);
	LTPrint(plist);//1 2 3 4 
	//LTPushFront(plist, 5);
	//LTPrint(plist);//5 1 2 3 4 
	//LTPopBack(plist);
	//LTPrint(plist);//5 1 2 3
	//LTPopFront(plist);
	//LTPrint(plist);//1 2 3
	//ListNode* find = LTFind(plist, 1);
	/*LTPushAfter(find, 4);*/	
	LTPrint(plist);//1 4 2 3
	//LTErase(find);
	//LTPrint(plist);//2 3
	LTDestroy(plist);

}
int main()
{
	ListTest();
	return 0;
}

2.6.4 代码测试运行

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


一级指针:
phead的改变不影响plist,phead释放之后,plist指向已经释放掉的空间——>把plist置为空

那么置为空之前,还要不要将plist指向的空间再free一次?

我们尝试一下

那么再思考一下:一级指针是会导致phead的改变不影响plist,那么plist是什么没有改变?是指plist保存的值没有被改变还是plist的这块空间的地址没有被释放?


学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


这里报错指的是plist指向无效地址

注意⚠️
如果plist的地址没有被释放,那么直接free(plist)是不会报错的

所以在一级指针的情况下:plist的地址已经被释放了,没有被置为空的可以理解是plist的地址名称

2.6.5 一级指针的改进---test.c

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


2. 二级指针

2.6.1 List.h
//销毁
//void LTDestroy(ListNode* phead);
void LTDestroy(ListNode** phead);

2.6.2 List.c
//销毁
void LTDestroy(ListNode** phead)
{
	assert(phead && *phead);
	ListNode* cur = (*phead)->next;
	while (cur != *phead)
	{
		ListNode* next = cur->next;
		free(cur);
		cur = next;
	}
	free(*phead);
	*phead = NULL;
}

2.6.3 test.c
#include"List.h"
void ListTest()
{
	ListNode* plist = LTInit();
	LTPushBack(plist, 1);
	LTPushBack(plist, 2);
	LTPushBack(plist, 3);
	LTPushBack(plist, 4);
	LTPrint(plist);//1 2 3 4 
	//LTPushFront(plist, 5);
	//LTPrint(plist);//5 1 2 3 4 
	//LTPopBack(plist);
	//LTPrint(plist);//5 1 2 3
	//LTPopFront(plist);
	//LTPrint(plist);//1 2 3
	//ListNode* find = LTFind(plist, 1);
	///*LTPushAfter(find, 4);*/	
	LTPrint(plist);//1 4 2 3
	//LTErase(find);
	//LTPrint(plist);//2 3
	//LTDestroy(plist);
	//plist = NULL;
	LTDestroy(&plist);
}
int main()
{
	ListTest();
	return 0;
}

2.6.4 代码测试运行

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言


虽然,二级指针不用手动将plist置为空
但是,更推荐一级指针,因为其他接口基本上都是一级指针——>保持接口的一致性


2.7 完整代码💦

2.7.1 List.h

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<string.h>
#include<stdbool.h>

typedef int LTDataType;
//创建双向链表的结构体
typedef struct ListNode {
	LTDataType data;
	struct ListNode* prev;
	struct ListNode* next;
}ListNode;

//初始化
ListNode* LTInit();//不用传入参数,直接调用接口返回一个头节点

//在双向链表中不会改变哨兵位,所以这里都可以传一级指针
//尾插
void LTPushBack(ListNode* phead, LTDataType x);

//打印
void LTPrint(ListNode* phead);

//头插
void LTPushFront(ListNode* phead, LTDataType x);

//尾删
void LTPopBack(ListNode* phead);

//头删
void LTPopFront(ListNode* phead);

//查找数据
ListNode* LTFind(ListNode* phead, LTDataType x);

//pos节点之后插入
void LTPushAfter(ListNode* pos, LTDataType x);

//删除pos节点
void LTErase(ListNode* pos);

//销毁
void LTDestroy(ListNode* phead);

2.7.2 List.c

#include"List.h"
//初始化
ListNode* LTInit()//不用传入参数,直接调用接口返回一个头节点
{
	//为头节点申请空间
	ListNode* phead = (ListNode*)malloc(sizeof(ListNode));
	//判断开辟是否成功
	if (phead == NULL)
	{
		perror("malloc error!\n");
		return;
	}
	//开辟成功--->初始化头节点
	phead->data = -1;//头节点不存储有效数据,可以任意赋值
	//只有哨兵位的时候,要实现双向链表,不能指向NULL,否则无法双向循环,所以我们指向自己
	phead->prev = phead->next = phead;
	return phead;
}

//在双向链表中不会改变哨兵位,所以这里都可以传一级指针
// 只改变数据,不改变地址

//开辟空间
ListNode* ListBuyNode(LTDataType x)
{
	ListNode* node = (ListNode*)malloc(sizeof(ListNode));
	if (node == NULL)
	{
		perror("malloc error!\n");
		return;
	}
	node->data = x;
	node->next = node->prev = NULL;
	return node;
}

//尾插
void LTPushBack(ListNode* phead, LTDataType x)
{
	assert(phead);//注意哨兵位不能为空
	//申请空间
	ListNode* node = ListBuyNode(x);
	//先处理node的前驱指针和后继指针
	node->prev = phead->prev;
	node->next = phead;
	//再处理之前的尾节点和phead
	phead->prev->next = node;
	phead->prev = node;
}

//打印
void LTPrint(ListNode* phead)
{
	//哨兵位不能改变
	ListNode* cur = phead->next;
	while (cur != phead)//当cur再次指向phead的时候,循环结束
	{
		printf("%d->", cur->data);
		cur = cur->next;
	}
	printf("\n");
}

//头插
void LTPushFront(ListNode* phead, LTDataType x)
{
	assert(phead);//注意哨兵位不能为空
	//申请空间
	ListNode* node = ListBuyNode(x);
	//node插入头节点之后才算头插
	//先处理node的前驱指针和后继指针
	node->prev = phead;
	node->next = phead->next;
	//再处理phead和phead->next
	phead->next->prev = node;
	phead->next = node;
}

//尾删
void LTPopBack(ListNode* phead)
{
	//不能为空链表,只有一个哨兵位不能尾删
	assert(phead&&(phead->prev!=phead||phead->next!=phead));
	ListNode* del = phead->prev;//phead->prev就是尾节点
	//先处理del
	del->prev->next = phead;
	//再处理phead
	phead->prev = del->prev;
	free(del);
	del = NULL;
}

//头删
void LTPopFront(ListNode* phead)
{
	//不能为空链表,只有一个哨兵位不能头删
	assert(phead && (phead->prev != phead || phead->next != phead));
	ListNode* del = phead->next;
	del->next->prev = phead;
	phead->next = del->next;
	free(del);
	del = NULL;
}

//查找数据
ListNode* LTFind(ListNode* phead, LTDataType x)
{
	assert(phead);
	ListNode* cur = phead->next;
	while (cur != phead)
	{
		if (cur->data == x)
		{
			return cur;
		}
		cur = cur->next;
	}
	return NULL;
}

//pos节点之后插入
void LTPushAfter(ListNode* pos, LTDataType x)
{
	assert(pos);
	ListNode* node = ListBuyNode(x);
	//node
	node->next = pos->next;
	node->prev = pos;
	//pos
	pos->next = node;
	node->next->prev = node;
}

//删除pos节点
void LTErase(ListNode* pos)
{
	assert(pos);
	pos->prev->next = pos->next;
	pos->next->prev = pos->prev;
	free(pos);
	pos = NULL;
}

//销毁
void LTDestroy(ListNode* phead)
{
	assert(phead);
	ListNode* cur = phead->next;
	while(cur!=phead)
	{
		ListNode* next = cur->next;
		free(cur);
		cur = next;
	}
	//注意哨兵位还没有释放
	free(phead);
	phead = NULL;
}

2.7.3 test.c

#include"List.h"
void ListTest()
{
	ListNode* plist = LTInit();
	LTPushBack(plist, 1);
	LTPushBack(plist, 2);
	LTPushBack(plist, 3);
	LTPushBack(plist, 4);
	LTPrint(plist);//1 2 3 4 
	LTPushFront(plist, 5);
	LTPrint(plist);//5 1 2 3 4 
	LTPopBack(plist);
	LTPrint(plist);//5 1 2 3
	LTPopFront(plist);
	LTPrint(plist);//1 2 3
	ListNode* find = LTFind(plist, 1);
	/*LTPushAfter(find, 4);*/	
	//LTPrint(plist);//1 4 2 3
	LTErase(find);
	LTPrint(plist);//2 3
	LTDestroy(plist);
	plist = NULL;
}
int main()
{
	ListTest();
	return 0;
}

3. 顺序表和双向链表的分析🍻

不同点 顺序表 链表(单链表)
存储空间上 物理上一定连续 逻辑上连续,但物理上不一定连续
随机访问 支持O(1) 不支持:O(N)
任意位置插入或者删除元素 看你需要搬移元素,效率低O(N) 只需要改变指针指向
插入 动态顺序表,空间不够的时候需要扩容 没有容量的概念
应用场景 元素高效存储+频繁访问 任意位置插入和删除频繁

本次的分享到这里就结束了!!!

PS:小江目前只是个新手小白。欢迎大家在评论区讨论哦!有问题也可以讨论的!

如果对你有帮助的话,记得点赞👍+收藏⭐️+关注➕

学习笔记---更进一步的双向链表专题~~,笔记,链表,双向链表,数据结构,算法,新手,c语言文章来源地址https://www.toymoban.com/news/detail-731428.html

到了这里,关于学习笔记---更进一步的双向链表专题~~的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • 青岛大学_王卓老师【数据结构与算法】Week04_04_双向链表的插入_学习笔记

    青岛大学_王卓老师【数据结构与算法】Week04_04_双向链表的插入_学习笔记

    本文是个人学习笔记,素材来自青岛大学王卓老师的教学视频。 一方面用于学习记录与分享,另一方面是想让更多的人看到这么好的《数据结构与算法》的学习视频。 如有侵权,请留言作删文处理。 课程视频链接: 数据结构与算法基础–第04周04–2.5.4双向链表2–双向链表

    2024年02月12日
    浏览(14)
  • 进一步探讨二分

    接下来简单探讨几种查询 严格大于等于x的情况,只需要 去掉等号号 即可 严格小于x的情况,将 小于符号改为大于符号 即可 严格小于等于x的情况,也只需要 去掉等号 即可 写题过程中还有具体的探讨,可以从这几种方法中迁移应用

    2024年02月16日
    浏览(7)
  • Unity的GPUSkinning进一步介绍

    Unity的GPUSkinning进一步介绍

      大家好,我是阿赵。   在几年前,我曾经写过一篇介绍GPUSkinning的文章,这么多年之后,还是看到不停有朋友在翻看这篇旧文章。今天上去GitHub看了一下,GPUSkinning这个开源的插件已经很久没有更新过了,还是停留在2017年的0.2.3版本。GPUSkinning的魅力在于可以在消耗比较

    2024年02月05日
    浏览(8)
  • 【Vue路由(router)进一步详解】

    【Vue路由(router)进一步详解】

    本篇文章主要针对已经掌握Vue路由(router)基础以及路由嵌套的用户群体,如果你是Vue路由初学者的话,不仿先去看看 【Vue路由(router)的基本使用】这篇文章 接上一篇文章理解Vue路由中常用的知识点 在实际开发过程中,我们可能不单单要实现简单的页面跳转动作,可能在

    2023年04月08日
    浏览(7)
  • Druid未授权漏洞进一步的利用

    Druid未授权漏洞进一步的利用

    Druid是阿里巴巴数据库出品的为监控而生的数据库连接池。并且Druid提供的监控功能包括监控SQL的执行时间、监控Web URI的请求、Session监控等。Druid本身是不存在什么漏洞的,但当开发者配置不当时就可能造成未授权访问。本文除了介绍Druid未授权漏洞之外,还要讲的是一种该漏

    2024年02月11日
    浏览(9)
  • web自动化测试入门篇03——selenium使用教程_(2)在上述学习基础上,自行选择一个合适的网站,进一步在实践中去运用selenium webd(1)

    web自动化测试入门篇03——selenium使用教程_(2)在上述学习基础上,自行选择一个合适的网站,进一步在实践中去运用selenium webd(1)

    先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7 深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前! 因此收集整理了一份《2024年最新大数据全套学习资料》,

    2024年04月27日
    浏览(16)
  • 数据结构--并查集的进一步优化

    数据结构--并查集的进一步优化

    压缩路径 − − F i n d 操作,先找到根节点,再将查找路径上所有结点都挂到根结点下 color{red}压缩路径 -- Find操作,先找到根节点,再将查找路径上所有结点都挂到根结点下 压缩路径 − − F in d 操作,先找到根节点,再将查找路径上所有结点都挂到根结点下 每次Find操作,

    2024年02月15日
    浏览(13)
  • 进一步了解WPF UI 实现XAML语法

    进一步了解WPF UI 实现XAML语法

    Extensible Application Markup Language (XAML) 是一种用于声明性应用程序编程的标记语言。 Windows Presentation Foundation (WPF) 实现 XAML 处理器实现并提供 XAML 语言支持。 WPF 类型的实现为 XAML 表示提供了必要的类型支持,从而确保了顺畅的集成和高效的运行。 在 XAML 标记中创建 WPF 应用程序

    2024年02月02日
    浏览(11)
  • [架构之路-203] - 对系统需求类型的进一步澄清

    [架构之路-203] - 对系统需求类型的进一步澄清

    目录 业务/商业需求: 用户/客户需求: 功能性需求: 非功能性需求: 系统需求: 约束条件: 软件需求说明书: 软件质量: 是自顶向下的需求,往往来自于中高层管理人员(或监管、政策要求),基于业务运营管理的直接诉求和要求。需要使用商业/工作语言描述业务/商业

    2024年02月07日
    浏览(13)
  • C++面向对象程序设计 - 类和对象进一步讨论

    C++面向对象程序设计 - 类和对象进一步讨论

            在C++中,关于面向对象程序设计已经讲了很大篇幅,也例举很多案例,此篇将通过一些习题来进一步了解对象、静态成员、指针、引用、友元、类模板等等相关知识。         示例代码:         如上代码,运行后结果如下图:          现在将上述代码中,第

    2024年04月16日
    浏览(11)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包