一:详细代码
#include <stdio.h>
#include <malloc.h>
typedef struct DoubleLinkedNode {
char data;
struct DoubleLinkedNode* previous;
struct DoubleLinkedNode* next;
//创建结构体的三个部分。
}DLNode, *DLNodePtr;
//DLNode为结构体,用来取空间,DLNodePtr为结构体指针可以灵活运用。
DLNodePtr initLinkList() {
DLNodePtr tempHeader = (DLNodePtr)malloc(sizeof(struct DoubleLinkedNode));
tempHeader->data = '\0';
tempHeader->previous = NULL;
tempHeader->next = NULL;
return tempHeader;
}
//对节点进行初始化。
void printList(DLNodePtr paraHeader) {
DLNodePtr p = paraHeader->next;
while (p != NULL) {
printf("%c", p->data);
p = p->next;
}
printf("\n");
}
//通过链表的头指针输出整个链表中的数据。
void insertElement(DLNodePtr paraHeader, char paraChar, int paraPosition) {
DLNodePtr p, q, r;
p = paraHeader;//创建指针p来对链表进行操作,这样不会改变头指针位置。
for (int i = 0; i < paraPosition; i++) {
p = p->next;
if (p == NULL) {
printf("The position %d is beyond the scope of the list.", paraPosition);
return;
}
}
//通过next遍历到插入位置。
q = (DLNodePtr)malloc(sizeof(struct DoubleLinkedNode));
q->data = paraChar;
//创建新节点。
r = p->next;
q->next = p->next;
q->previous = p;
p->next = q;
if (r != NULL) {
r->previous = q;
}
//对新节点进行操作。
}
void deleteElement(DLNodePtr paraHeader, char paraChar) {
DLNodePtr p, q, r;
p = paraHeader;
while ((p->next != NULL) && (p->next->data != paraChar)) {
p = p->next;
}
if (p->next == NULL) {
printf("The char '%c' does not exist.\n", paraChar);
return;
}
//遍历的方式根据传入参数的不同而不同。
q = p->next;
r = q->next;
p->next = r;
if (r != NULL) {
r->previous = p;
}
//删除的操作和插入有些不一样,需要理解清楚。
free(q);
}
void insertDeleteTest() {
DLNodePtr tempList = initLinkList();
printList(tempList);
insertElement(tempList, 'H', 0);
insertElement(tempList, 'e', 1);
insertElement(tempList, 'l', 2);
insertElement(tempList, 'l', 3);
insertElement(tempList, 'o', 4);
insertElement(tempList, '!', 5);
printList(tempList);
deleteElement(tempList, 'e');
deleteElement(tempList, 'a');
deleteElement(tempList, 'o');
printList(tempList);
insertElement(tempList, 'o', 1);
printList(tempList);
}
//进行操作检验自己的代码是否正确。
void basicAddressTest() {
DLNode tempNode1, tempNode2;
tempNode1.data = 4;
tempNode1.next = NULL;
tempNode2.data = 6;
tempNode2.next = NULL;
printf("The first node: %p, %p, %p\n",
&tempNode1, &tempNode1.data, &tempNode1.next);
printf("The second node: %p, %p, %p\n",
&tempNode2, &tempNode2.data, &tempNode2.next);
tempNode1.next = &tempNode2;
}
//输入链表各部分的地址,更容易理解数据在双向链表中的存储。
void main() {
insertDeleteTest();
basicAddressTest();
}
二:运行结果
三:实现方法
注意对结构体指针的应用,指针存储的是地址,在链表中,头指针储存的是链表第一个节点的首地址,保持头指针一直指向链表第一个节点,这样头指针才有用。
双向链表与单向链表的区别就是多了个指向前项的指针,在插入和删除时注意对这个指针的引用。
当链表末尾的节点指向链表第一个节点时,就成为了循环链表。