C语言链表

head.h

c 复制代码
typedef struct Node_s{
	int data; //数据域
	struct Node_s *pNext; //指针域
} Node_t, *pNode_t;

void headInsert(pNode_t *ppHead, pNode_t *ppTail, int data);
void print(pNode_t pHead);
void tailInsert(pNode_t *ppHead, pNode_t *ppTail, int data);
void sortInsert(pNode_t *ppHead, pNode_t *ppTail, int data);
void modify(pNode_t pHead, int findNum, int changeNum);
void listDelete(pNode_t *ppHead, pNode_t *ppTail, int deleteNum);

main.c

c 复制代码
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "head.h"
int main() {
	pNode_t pHead = NULL;
	pNode_t pTail = NULL;
	int data;
	while (scanf("%d", &data) != EOF) {
		//headInsert(&pHead, &pTail, data);
		//tailInsert(&pHead, &pTail, data);
		sortInsert(&pHead, &pTail, data);
		print(pHead);
	}
	/*int findNum, changeNum;
	while (scanf("%d%d", &findNum, &changeNum) != EOF) {
		modify(pHead, findNum, changeNum);
		print(pHead);
	}*/
	int deleteNum;
	while (scanf("%d", &deleteNum) != EOF) {
		listDelete(&pHead, &pTail, deleteNum);
		print(pHead);
	}
}
//头插法
void headInsert(pNode_t *ppHead, pNode_t *ppTail, int data) {
	//在堆上申请空间,存储链表结点
	pNode_t pNew = (pNode_t)malloc(sizeof(Node_t));
	memset(pNew, 0, sizeof(Node_t));//数据域为0,指针域为NULL
	pNew->data = data;
	if (*ppHead == NULL) {
		*ppHead = pNew;
		*ppTail = pNew;
	}
	else {
		pNew->pNext = *ppHead;
		*ppHead = pNew;
	}
}
//打印链表
void print(pNode_t pHead) {
	pNode_t pCur = pHead;
	while (pCur) {
		printf("%3d", pCur->data);
		pCur = pCur->pNext;
	}
	printf("\n");
}
//尾插法
void tailInsert(pNode_t *ppHead, pNode_t *ppTail, int data) {
	pNode_t pNew = (pNode_t)calloc(1, sizeof(Node_t));
	//使用calloc可以不用执行memset了
	pNew->data = data;
	if (*ppHead == NULL) {
		*ppHead = pNew;
		*ppTail = pNew;
	}
	else {
		(*ppTail)->pNext = pNew;
		*ppTail = pNew;
	}
}
//有序插入
void sortInsert(pNode_t *ppHead, pNode_t *ppTail, int data) {
	pNode_t pNew = (pNode_t)calloc(1, sizeof(Node_t));
	pNew->data = data;
	if (*ppHead == NULL) {
		//插入唯一的结点
		*ppHead = pNew;
		*ppTail = pNew;
	}
	else if (data < (*ppHead)->data) {
		//执行头插法
		pNew->pNext = *ppHead;
		*ppHead = pNew;
	}
	else {
		//存在某个结点,其指针域将要改变指向
		pNode_t pPre = *ppHead;//要修改指针域的结点
		pNode_t pCur = pPre->pNext;//是pPre的后继,pCur所指结点的数据域决定了是否要插入
		//经验:写出每次循环迭代的语句,根据这个语句是否有问题,判断循环的条件是否有问题
        while (pCur) {
			if (pCur->data > data) {
				//插入到中间的位置
				pPre->pNext = pNew;
				pNew->pNext = pCur;
				break;
			}
			pPre = pCur;
			pCur = pCur->pNext;
		}
		//如果退出循环时,pCur为NULL,说明没有找到中间位置,要执行尾插法
		if (pCur == NULL) {
			(*ppTail)->pNext = pNew;
			*ppTail = pNew;
		}
	}
}
//查找和修改
void modify(pNode_t pHead, int findNum, int changeNum) {
	pNode_t pCur = pHead;
	while (pCur) {
		if (pCur->data == findNum) {
			pCur->data = changeNum;
			break;
		}
		pCur = pCur->pNext;
	}
	if (pCur == NULL) {
		fprintf(stderr, "Cannot find the number!\n");
	}
}
//删除
void listDelete(pNode_t *ppHead, pNode_t *ppTail, int deleteNum) {
	pNode_t pCur = *ppHead;//pCur指向待删除的结点,以供后续free
	if (pCur == NULL) {
		fprintf(stderr, "List is empty!\n");
		return;//return 终止本次函数调用
	}
	else if(pCur->data == deleteNum){
		//删除的是第一个结点
		*ppHead = (*ppHead)->pNext;//pCur依然指向原来的pHead pHead后移了
		if (*ppHead == NULL) {
			//删除之后没有结点,意味着删除的是唯一的结点
			*ppTail = NULL;
		}
	}
	else {
		pNode_t pPre = *ppHead;//双指针法,pPre指向要修改指针域的结点
		pCur = pPre->pNext;
		while (pCur) {
			if (pCur->data == deleteNum) {
				//链表结点删除,意味着其前驱结点的指针域改变了指向
				pPre->pNext = pCur->pNext;
				break;
			}
			pPre = pCur;
			pCur = pCur->pNext;
		}
		if (pCur == *ppTail) {
			*ppTail = pPre;
		}
		if (pCur == NULL) {
			fprintf(stderr, "DeleteNum is not found!\n");
			return;
		}
	}
	free(pCur);
	pCur = NULL;
}
相关推荐
蟹至之5 分钟前
字符函数 和 字符串函数 的使用与模拟
c语言·字符串·指针·const关键词
翔云API8 分钟前
人证合一接口:智能化身份认证的最佳选择
大数据·开发语言·node.js·ocr·php
jimmy.hua8 分钟前
C++刷怪笼(5)内存管理
开发语言·数据结构·c++
xiaobai12 311 分钟前
二叉树的遍历【C++】
开发语言·c++·算法
DieSnowK18 分钟前
[项目][WebServer][Makefile & Shell]详细讲解
开发语言·c++·http·makefile·shell·项目·webserver
Freak嵌入式19 分钟前
全网最适合入门的面向对象编程教程:50 Python函数方法与接口-接口和抽象基类
java·开发语言·数据结构·python·接口·抽象基类
冷凝女子21 分钟前
【QT】基于HTTP协议的网络应用程序
开发语言·qt·http
知识分享小能手24 分钟前
mysql学习教程,从入门到精通,SQL 删除数据(DELETE 语句)(19)
大数据·开发语言·数据库·sql·学习·mysql·数据开发
鸽芷咕33 分钟前
【Python报错已解决】libpng warning: iccp: known incorrect sRGB profile
开发语言·python·机器学习·bug
白总Server39 分钟前
MongoDB解说
开发语言·数据库·后端·mongodb·golang·rust·php