C语言数据结构-单向链表
头文件:link.h
#ifndef __LINK_H__
#define __LINK_H__
#include <stdio.h>
#include <stdlib.h>
typedef int DataType;
/*节点数据类型*/
typedef struct node
{
DataType data; //数据域
struct node *pNext; //指针域
}LinkNode;
/* 标签数据类型 */
typedef struct list
{
LinkNode *pHead; //链表头节点指针
int cLen; //当前链表节点个数
}LinkList;
#endif
源文件:link.c
#include "link.h"
创建一个新节点:
LinkList *createLinkList()
{
LinkList *pList = NULL;
pList = malloc(sizeof(LinkList));
if (NULL == pList)
{
perror("fail to malloc");
return NULL;
}
pList->pHead = NULL;
pList->cLen = 0;
return pList;
}
头插:
int insertHeadLinkList(LinkList *pList, DataType data)
{
LinkNode * pInsertNode = malloc(sizeof(LinkNode));
if (NULL == pInsertNode)
{
perror("fail to malloc");
return -1;
}
pInsertNode->data = data;
pInsertNode->pNext = pList->pHead;
pList->pHead = pInsertNode;
pList->cLen++;
return 0;
}
遍历:
void showLinkList(LinkList *pList)
{
LinkNode *pTmpNode = pList->pHead;
while (pTmpNode != NULL)
{
printf("%d ", pTmpNode->data);
pTmpNode = pTmpNode->pNext;
}
printf("\n");
}
判空:
int isEmptyLinkList(LinkList *pList)
{
return pList->cLen == 0;
}
尾插:
int insertTailLinkList(LinkList *pList, DataType data)
{
LinkNode *pInsertNode = malloc(sizeof(LinkNode));
if (NULL == pInsertNode)
{
perror("fail to malloc");
return -1;
}
pInsertNode->data = data;
pInsertNode->pNext = NULL;
if (isEmptyLinkList(pList))
{
pList->pHead = pInsertNode;
}
else
{
LinkNode *pTmpNode = pList->pHead;
while (pTmpNode->pNext != NULL)
{
pTmpNode = pTmpNode->pNext;
}
pTmpNode->pNext = pInsertNode;
}
pList->cLen++;
return 0;
}
头删:
int deleteHeadLinkList(LinkList *pList)
{
if (isEmptyLinkList(pList))
{
return 0;
}
LinkNode *pFreeNode = pList->pHead;
pList->pHead = pFreeNode->pNext;
free(pFreeNode);
pList->cLen--;
return 0;
}
尾删:
int deleteTailLinkList(LinkList *pList)
{
if (isEmptyLinkList(pList))
{
return 0;
}
else if (1 == pList->cLen)
{
deleteHeadLinkList(pList);
}
else
{
LinkNode *pTmpNode = pList->pHead;
while (pTmpNode->pNext->pNext != NULL)
{
pTmpNode = pTmpNode->pNext;
}
free(pTmpNode->pNext);
pTmpNode->pNext = NULL;
pList->cLen--;
}
return 0;
}
查找单链表的某个节点:
LinkNode *findLinkList(LinkList *pList, DataType findData)
{
LinkNode *pTmpNode = pList->pHead;
while (pTmpNode != NULL)
{
if (pTmpNode->data == findData)
{
return pTmpNode;
}
pTmpNode = pTmpNode->pNext;
}
return NULL;
}
单链表倒置:
int reverseLinkList(LinkList *pList, DataType oldData, DataType newData)
{
LinkNode *pTmpNode = NULL;
while ((pTmpNode = findLinkList(pList, oldData)) != NULL)
{
pTmpNode->data = newData;
}
return 0;
}
销毁链表:
void destroyLinkList(LinkList **ppList)
{
while ((*ppList)->pHead != NULL)
{
deleteHeadLinkList(*ppList);
}
free(*ppList);
*ppList = NULL;
}
查找中间节点:
LinkNode *findMidLinkNode(LinkList *pList)
{
LinkNode *pFast = pList->pHead;
LinkNode *pSlow = pFast;
while (pFast != NULL)
{
pFast = pFast->pNext;
if (NULL == pFast)
{
break;
}
pFast = pFast->pNext;
pSlow = pSlow->pNext;
}
return pSlow;
}
查找最后一个节点:
LinkNode *findLastKNode(LinkList *pList, int K)
{
LinkNode *pFast = pList->pHead;
LinkNode *pSlow = pFast;
int i = 0;
for (; i < K; i++)
{
pFast = pFast->pNext;
}
while (pFast != NULL)
{
pFast = pFast->pNext;
pSlow = pSlow->pNext;
}
return pSlow;
}
删除某个节点:
int deletePointNode(LinkList *pList, DataType deleteData)
{
LinkNode *pPreNode = pList->pHead;
LinkNode *pFreeNode = pList->pHead;
while (pFreeNode != NULL)
{
if (pFreeNode->data == deleteData)
{
if (pPreNode == pFreeNode)
{
pList->pHead = pFreeNode->pNext;
free(pFreeNode);
pPreNode = pList->pHead;
pFreeNode = pList->pHead;
}
else
{
pPreNode->pNext = pFreeNode->pNext;
free(pFreeNode);
pFreeNode = pPreNode->pNext;
}
pList->cLen--;
}
else
{
pPreNode = pFreeNode;
pFreeNode = pFreeNode->pNext;
}
}
return 1;
}
单链表反转:
void invertLinkList(LinkList *pList)
{
LinkNode *pTmpNode = pList->pHead;
LinkNode *pInsertNode = NULL;
pList->pHead = NULL;
while (pTmpNode != NULL)
{
pInsertNode = pTmpNode;
pTmpNode = pTmpNode->pNext;
pInsertNode->pNext = pList->pHead;
pList->pHead = pInsertNode;
}
return ;
}
链表排序:
void sortLinkList(LinkList *pList)
{
//链表为空或只有一个节点不需要排序
if (isEmptyLinkList(pList) || 1 == pList->cLen)
{
return ;
}
//保存第二个节点并且从第一个节点后断开
LinkNode *pInsertNode = NULL;
LinkNode *pTmpNode = pList->pHead->pNext;
pList->pHead->pNext = NULL;
while (pTmpNode != NULL)
{
//找到要插入的节点
pInsertNode = pTmpNode;
pTmpNode = pTmpNode->pNext;
//判断是否需要头插
if (pInsertNode->data < pList->pHead->data)
{
//头插
pInsertNode->pNext = pList->pHead;
pList->pHead = pInsertNode;
}
else
{
//寻找插入位置
LinkNode *p = pList->pHead;
while (p->pNext != NULL && p->pNext->data < pInsertNode->data)
{
p = p->pNext;
}
//节点插入
pInsertNode->pNext = p->pNext;
p->pNext = pInsertNode;
}
}
}
判断链表是否有环:
int isLoopLinkList(LinkList *pList)
{
LinkNode *pFast = pList->pHead;
LinkNode *pSlow = pList->pHead;
while (pFast != NULL)
{
pFast = pFast->pNext;
if (NULL == pFast)
{
return 0;
}
if (pSlow == pFast)
{
return 1;
}
pFast = pFast->pNext;
pSlow = pSlow->pNext;
if (pFast == pSlow)
{
return 1;
}
}
}
int main(int argc, const char *argv[])
{
LinkList *pList = NULL;
LinkNode *pTmpNode = NULL;
pList = createLinkList();
// insertHeadLinkList(pList, 1);
insertHeadLinkList(pList, 2);
insertHeadLinkList(pList, 3);
insertHeadLinkList(pList, 4);
insertTailLinkList(pList, 5);
insertTailLinkList(pList, 3);
insertTailLinkList(pList, 3);
insertTailLinkList(pList, 3);
insertTailLinkList(pList, 3);
insertTailLinkList(pList, 3);
insertTailLinkList(pList, 3);
showLinkList(pList);
#if 0
deleteHeadLinkList(pList);
deleteTailLinkList(pList);
showLinkList(pList);
pTmpNode = findLinkList(pList, 3);
if (pTmpNode != NULL)
{
printf("find %d\n", pTmpNode->data);
}
else
{
printf("not find\n");
}
reverseLinkList(pList, 3, 10);
showLinkList(pList);
#endif
pTmpNode = findMidLinkNode(pList);
printf("mid node = %d\n", pTmpNode->data);
pTmpNode = findLastKNode(pList, 3);
printf("last K node = %d\n", pTmpNode->data);
#if 0
deletePointNode(pList, 3);
showLinkList(pList);
invertLinkList(pList);
showLinkList(pList);
sortLinkList(pList);
showLinkList(pList);
// destroyLinkList(&pList);
#endif
//构造环形链表
pTmpNode = pList->pHead;
while (pTmpNode->pNext != NULL)
{
pTmpNode = pTmpNode->pNext;
}
pTmpNode->pNext = pList->pHead->pNext->pNext->pNext->pNext;
if (isLoopLinkList(pList))
{
printf("is loop link.\n");
}
else
{
printf("is not loop link.\n");
}
return 0;
}
相关文章:

C语言数据结构-单向链表
头文件:link.h #ifndef __LINK_H__ #define __LINK_H__ #include <stdio.h> #include <stdlib.h> typedef int DataType; /*节点数据类型*/ typedef struct node { DataType data; //数据域 struct node *pNext; //指…...

小样本分类新突破:QPT技术详解
问题导向式提示调优(QPT) 这篇论文主要讲了一个针对小样本(数据量少)文本分类问题的新方法,叫问题导向式提示调优(QPT)。 核心思路是让预训练语言模型(比如BERT的升级版RoBERTa)在少量标注数据下,通过设计特定的“提问式模板”和“标签词扩展技术”来提升分类效果。…...

Excel常用公式全解析(1):从基础计算到高级应用
Excel常用公式全解析:从基础计算到高级应用 目录 Excel常用公式全解析:从基础计算到高级应用[toc](目录)一、基础计算类:数据运算的基石1. 求和公式(SUM)2. 平均值公式(AVERAGE)3. 最值与计数公…...
C++ STL 容器:List 深度解析与实践指南
一、List 容器概述 1.1底层结构与特性 数据结构:双向循环链表(带哨兵位头结点),每个节点包含前驱指针、后继指针和数据域。核心优势: 高效插入 / 删除:任意位置操作时间复杂度为 O (1),无需移…...
每天掌握一个Linux命令 - ab(Apache Benchmark)
Linux 命令工具 ab 使用指南 一、工具概述 ab(Apache Benchmark) 是 Apache 官方提供的开源压力测试工具,用于衡量 Web 服务器的性能。它通过模拟多并发请求,测试服务器在高负载下的响应速度、吞吐量和稳定性,常用于…...

与 PyCharm 官方沟通解决开发环境问题记录(进展:官方已推出2个新的修复版本)
主题:有关 PyCharm 中终端和环境激活问题的反馈:PY-81233 前言 目前进展: 官方已有2个修复版本推出测试。 更新方法: 使用JetBrains Toolbox App,如下图所示,从“其他版本”进入查看更新。…...
Python的分布式网络爬虫系统实现
1. 系统架构概述 一个典型的分布式网络爬虫系统通常包含以下几个核心组件: 1.主节点(Master Node): 任务调度:负责将抓取任务分配给各个工作节点。URL 管理:维护待抓取的 URL 队列和已抓取的 URL 集合&a…...
Vue快速上手(业务、技术、报错)
Vue 技术业务报错 技术 业务 Vueelement-ui,实现表格渲染缩略图,鼠标悬浮缩略图放大,点击缩略图播放视频(一) 报错 vue修改配置文件.env.development不生效 vue前端downloadFile报错:Error parsing HT…...

taro + vue3 实现小程序sse长连接实时对话
前言 taro.request是可以实现sse长连接的,但是呢其中有俩大坑,找了许多资料也没解决,后续解决办法也与后端商量改用WebSocket来实现。 代码实现 SSEManager.js: import { getAccessToken } from "../xx/xx"; import { TextDecode…...

使用MATLAB求解微分方程:从基础到实践
使用MATLAB求解微分方程:从基础到实践 微分方程是描述自然界和工程领域中许多现象的重要数学工具。MATLAB提供了强大的工具来求解各种类型的微分方程。本文将介绍如何使用MATLAB求解常微分方程(ODE)。 1. 基本ODE求解器 MATLAB提供了多种ODE求解器,最…...
基于MATLAB的大规模MIMO信道仿真
1. 系统模型与参数设置 以下是一个单小区大规模MIMO系统的参数配置示例,适用于多发多收和单发单收场景。 % 参数配置 params.N_cell 1; % 小区数量(单小区仿真) params.cell_radius 500; % 小区半径(米)…...

如何在 Windows 和 Mac 上擦拭和清洁希捷外置硬盘
希捷外置硬盘广泛用于存储目的,但有时您可能出于多种目的需要擦除或清洁希捷外置硬盘,例如转售、重复使用、捐赠等。为了释放硬盘上的存储空间或确保没有人可以从硬盘中恢复您的信息,擦除硬盘是必要的步骤。无论您使用的是 Windows 还是 Mac&…...
Vue 3.0 中状态管理Vuex 与 Pinia 的区别
在 Vue.js 应用开发中,状态管理是构建复杂应用的关键环节。随着 Vue 3 的普及和 Composition API 的引入,开发者面临着状态管理库的选择问题:是继续使用经典的 Vuex,还是转向新兴的 Pinia?本文将从设计理念、API 设计、…...

第三届黄河流域网安技能挑战赛复现
Web 奶龙牌图片处理器2.0 这题,之前只了解过 .user.ini 文件,并为遇到实操题 但赛前差点就做到下面这题了,不多说,复现之前先看看下面这题 靶场: 攻防世界 没错,又做上文件上传题了,别看…...

python 生成复杂表格,自动分页等功能
python 生成复杂表格,自动分页等功能 解决将Python中的树形目录数据转换为Word表格,并生成带有合并单元格的检测报告的问题。首先,要解决“tree目录数据”和“Word表格互换”,指将树…...
2025年高防IP与游戏盾深度对比:如何选择最佳防护方案?
2025年,随着DDoS攻击规模的指数级增长和混合攻击的常态化,高防IP与游戏盾成为企业网络安全的核心选择。然而,两者在功能定位、技术实现及适用场景上存在显著差异。本文结合最新行业实践与技术趋势,全面解析两者的优劣,…...
在 Vue + Vite 项目中,直接使用相对路径或绝对路径引用本地图片资源时,图片无法正确显示。
Vue 项目中静态资源引用问题 1.问题描述 在 Vue Vite 项目中,直接使用相对路径或绝对路径引用本地图片资源时,图片无法正确显示。 错误示例 javascript // 错误方式1:使用相对路径 const products [ { name: iPhone 14 Pro, image: .…...
判断手机屏幕上的横向滑动(左滑和右滑)
在JavaScript中,你可以通过监听触摸事件(touch events)来判断用户在手机屏幕上的横向滑动方向。以下是实现方法: 基本实现方案 let touchStartX 0; let touchEndX 0;function handleTouchStart(event) {touchStartX event.ch…...
用户有一个Django模型没有设置主键,现在需要设置主键。
用户有一个Django模型没有设置主键,现在需要设置主键。 from django.db import modelsclass CategoryAssistentModel(models.Model):second_level_category models.CharField(max_length100, nullTrue, blankTrue)third_level_category models.CharField(max_len…...

【文献阅读】EndoChat: Grounded Multimodal Large Language Model for Endoscopic Surgery
[2501.11347] EndoChat: Grounded Multimodal Large Language Model for Endoscopic Surgery 2025年1月 数据可用性 Surg-396K 数据集可在 GitHub - gkw0010/EndoChat 公开获取。 代码可用性 EndoChat 的代码可在 GitHub - gkw0010/EndoChat 下载。 摘要 近年来ÿ…...

React JSX语法介绍(JS XML)(一种JS语法扩展,允许在JS代码中编写类似HTML的标记语言)Babel编译
在线调试网站:https://zh-hans.react.dev/learn 文章目录 JSX:现代前端开发的声明式语法概述JSX的本质与工作原理什么是JSXJSX转换流程 JSX语法特性表达式嵌入(JSX允许在大括号内嵌入任何有效的JavaScript表达式)属性传递…...

【R语言编程绘图-箱线图】
基本箱线图绘制 使用ggplot2绘制箱线图的核心函数是geom_boxplot()。以下是一个基础示例,展示如何用iris数据集绘制不同物种(Species)的萼片长度(Sepal.Length)分布: library(ggplot2) ggplot(iris, aes(…...
【elasticsearch 7 或8 的安装及配置SSL 操作指引】
1.标题获取安装文件 cd /opt/tools wget https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-8.11.4-linux-x86_64.tar.gz tar -zxvf elasticsearch-8.11.4-linux-x86_64.tar.gz mv /opt/tools/elasticsearch-8.11.4 /opt/elasticsearch #配置vm.max_map_co…...
GitHub 趋势日报 (2025年05月23日)
本日报由 TrendForge 系统生成 https://trendforge.devlive.org/ 🌐 本日报中的项目描述已自动翻译为中文 📈 今日整体趋势 Top 10 排名项目名称项目描述今日获星总星数语言1All-Hands-AI/OpenHands🙌开放式:少代码,做…...

MongoDB索引:原理、实践与优化指南
为什么索引对数据库如此重要? 在现代应用开发中,数据库性能往往是决定用户体验的关键因素。想象一下,当你在电商平台搜索商品时,如果每次搜索都需要等待5-10秒才能看到结果,这种体验是多么令人沮丧。MongoDB作为最流行…...

SQL实战之索引优化(单表、双表、三表、索引失效)
文章目录 单表优化双表优化三表优化结论索引失效 单表优化 总体原则:建立索引并合理使用,避免索引失效 案例说明:查询category_ id 为1且comments大于1的情况下,views最多的article_ id: 传统方案: explain select id, author_ id…...

[7-1] ADC模数转换器 江协科技学习笔记(14个知识点)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 DMA(Direct Memory Access,直接内存访问)是一种硬件特性,它允许某些硬件子系统直接访问系统的内存,而无需CPU的介入。这样,CPU就可以处理其他任务,从而提高系…...
SSM整合:Spring+SpringMVC+MyBatis完美融合实战指南
前言 在Java企业级开发领域,SSM(SpringSpringMVCMyBatis)框架组合一直占据着重要地位。这三个轻量级框架各司其职又相互配合,为开发者提供了高效、灵活的开发体验。本文将深入探讨SSM框架的整合过程,揭示整合背后的原…...
Spring Boot分页查询进阶:整合Spring Data REST实现高效数据导航
目录: 引言分页查询基础回顾 2.1 Spring Data JPA分页接口 2.2 Pageable与Page的使用 2.3 常见分页参数设计Spring Data REST简介 3.1 HATEOAS与超媒体驱动API 3.2 Spring Data REST核心功能 3.3 自动暴露Repository接口整合Spring Boot与Spring Data REST 4.1 项目…...

阿里云 Serverless 助力海牙湾构建弹性、高效、智能的 AI 数字化平台
作者:赵世振、十眠、修省 “通过阿里云 Serverless 架构,我们成功解决了弹性能力不足、资源浪费与运维低效的痛点。SAE 的全托管特性大幅降低技术复杂度。未来,我们将进一步探索 Serverless 与 AI 的结合,为客户提供更智能的数字…...