C语言——头插法和尾插法动态创建链表

在C语言中,动态创建列表通常指的是使用链表(Linked List)这种数据结构。链表由一系列节点(Node)组成,每个节点包含数据部分和指向下一个节点的指针。根据插入节点的位置不同,链表的插入操作可以分为头插法和尾插法。下面将详细介绍这两种方法。

头插法动态创建链表

头插法是将新节点插入到链表的头部,这样新节点将成为链表的第一个节点。由于每次插入都在头部进行,因此这种方法不需要遍历链表来找到尾部,插入操作的时间复杂度为O(1)。

步骤
  1. 定义节点结构体:包含一个数据域和一个指向下一个节点的指针。
  2. 初始化链表:通常将头指针初始化为NULL,表示一个空链表。
  3. 创建新节点:使用malloccalloc动态分配内存给新节点。
  4. 插入新节点:将新节点的next指针指向当前的头节点,然后更新头指针指向新节点。
  5. 重复:根据需要重复步骤3和4,直到所有节点都被插入。
示例代码
#include <stdio.h>  
#include <stdlib.h>  
  
// 定义节点结构体  
typedef struct Node {  
    int data;  
    struct Node* next;  
} Node;  
  
// 头插法插入节点  
void insertAtHead(Node** head, int data) {  
    Node* newNode = (Node*)malloc(sizeof(Node));  
    if (!newNode) {  
        printf("Memory allocation failed\n");  
        exit(1);  
    }  
    newNode->data = data;  
    newNode->next = *head;  
    *head = newNode;  
}  
  
// 打印链表  
void printList(Node* head) {  
    Node* current = head;  
    while (current != NULL) {  
        printf("%d -> ", current->data);  
        current = current->next;  
    }  
    printf("NULL\n");  
}  
  
// 释放链表内存  
void freeList(Node* head) {  
    Node* temp;  
    while (head != NULL) {  
        temp = head;  
        head = head->next;  
        free(temp);  
    }  
}  
  
int main() {  
    Node* list = NULL; // 初始化空链表  
    insertAtHead(&list, 1);  
    insertAtHead(&list, 2);  
    insertAtHead(&list, 3);  
    printList(list); // 输出: 3 -> 2 -> 1 -> NULL  
    freeList(list); // 释放链表内存  
    return 0;  
}

尾插法动态创建链表

尾插法是将新节点插入到链表的尾部,这样新节点将成为链表的最后一个节点。由于每次插入都需要找到链表的尾部,因此这种方法在最坏情况下的时间复杂度为O(n),其中n是链表的长度。

步骤
  1. 定义节点结构体:与头插法相同。
  2. 初始化链表:将头指针和尾指针都初始化为NULL。
  3. 创建新节点:使用malloccalloc动态分配内存给新节点。
  4. 检查链表是否为空
    • 如果为空(头指针为NULL),新节点既是头节点也是尾节点。
    • 如果不为空,将尾节点的next指针指向新节点,并更新尾指针指向新节点。
  5. 更新头指针(可选):如果链表之前为空,现在需要更新头指针指向新节点(但这一步在插入操作中通常已经隐含在更新尾指针的操作中了)。
  6. 重复:根据需要重复步骤3、4和5,直到所有节点都被插入。
示例代码
#include <stdio.h>  
#include <stdlib.h>  
  
// 定义节点结构体  
typedef struct Node {  
    int data;  
    struct Node* next;  
} Node;  
  
// 尾插法插入节点  
void insertAtTail(Node** head, Node** tail, int data) {  
    Node* newNode = (Node*)malloc(sizeof(Node));  
    if (!newNode) {  
        printf("Memory allocation failed\n");  
        exit(1);  
    }  
    newNode->data = data;  
    newNode->next = NULL;  
      
    if (*head == NULL) { // 链表为空  
        *head = newNode;  
        *tail = newNode;  
    } else {  
        (*tail)->next = newNode;  
        *tail = newNode;  
    }  
}  
  
// 打印链表  
void printList(Node* head) {  
    // 与头插法示例中的打印函数相同  
}  
  
// 释放链表内存  
void freeList(Node* head) {  
    // 与头插法示例中的释放函数相同  
}  
  
int main() {  
    Node* list = NULL; // 初始化空链表  
    Node* tail = NULL; // 初始化尾指针  
    insertAtTail(&list, &tail, 1);  
    insertAtTail(&list, &tail, 2);  
    insertAtTail(&list, &tail, 3);  
    printList(list); // 输出: 1 -> 2 -> 3 -> NULL  
    freeList(list); // 释放链表内存  
    return 0;  
}

注意事项

  • 内存管理:确保在链表不再需要时释放所有节点的内存。
  • 错误处理:检查malloc是否成功,并在失败时适当处理。
  • 链表遍历:可以通过遍历链表来验证插入操作是否正确。
  • 头指针和尾指针:在尾插法中,同时维护头指针和尾指针可以方便地实现插入操作,而不需要每次都遍历链表找到尾部。

简易代码:

#include<stdio.h>
#include<stdlib.h>

struct Test{
        int data;
        struct Test *next;
};

void printLink(struct Test *head)
{
        while(1){
                if(head != NULL){
                        printf("%d  ",head->data);
                        head = head->next;
                }else{
                        putchar('\n');
                        break;
                }
        }


}

struct Test* insertFromBehind(struct Test *head,struct Test *new)
{
        struct Test *p = head;
        if(p == NULL){
                head = new;
                return head;
        }
        while(p->next != NULL){
                p = p->next;
        }
        p->next = new;

        return head;
}

struct Test* creatFromTail(struct Test* head)
{
        struct Test *new;
        while(1){
                new = (struct Test*)malloc(sizeof(struct Test));
                printf("input your number of node\n");
                scanf("%d",&(new->data));
                if(new->data == 0){
                        printf("0 quit\n");
                        free(new);
                        return head;
                }
                head = insertFromBehind(head,new);
        }
}
struct Test* insertFromForward(struct Test *head,struct Test *new)
{
                if(head == NULL){
                        head = new;
                }else{
                        new->next = head;
                        head = new;
                }
                return head;
}


struct Test* creatFromHead(struct Test* head)
{
        struct Test *new;
        while(1){
                new = (struct Test*)malloc(sizeof(struct Test));
                printf("input your number of node\n");
                scanf("%d",&(new->data));
                if(new->data == 0){
                        printf("0 quit\n");
                        free(new);
                        return head;
                }
                head = insertFromForward(head,new);
        }
 }

int main()
{
        struct Test *head = NULL;
        printf("this is link be creative insert head\n");
        head = creatFromHead(head);
        printLink(head);

        struct Test *head1 = NULL;
        printf("this is link be creative insert tail\n");
        head1 = creatFromTail(head1);
        printLink(head1);

        return 0;
}
     
头插法创建链表是一种在链表头部入新节点的方。在C语言中,可以使用以下代码实现头插法创建链表: ```c typedef struct Lnode { char data; struct Lnode* next; } Lnode, *Linklist; Linklist Establish_Linklist_head1(void) { char arr\[5\] = {'a', 'b', 'c', 'd', 'e'}; int i = sizeof(arr) / sizeof(arr\[0\]); Linklist l = (Linklist)malloc(sizeof(Lnode)); l->next = NULL; l->data = arr\[i - 1\]; while (i--) { Lnode* p = (Linklist)malloc(sizeof(Lnode)); p->data = arr\[i - 1\]; p->next = l; l = p; } return l; } ``` 这段代码会创建一个带有头结点头指针的链表,其中头结点的数据域为空,头指针指向第一个节点。通过循环将数组中的元素依次链表的头部,最后返回链表的头指针。 方12都是使用头插法创建链表,它们的最终结果是一样的。方1是在头结点之后入新节点,而方2是在链表的最前面入新节点。方2在产生头结点时可能会出现跨界访问的问题,但是编译器可能不会报错或警告。为了避免非访问,可以改变循环条件,并在循环之后再入一个头结点。方2仅供参考,学习链表时建议使用方1。 希望对你有帮助! #### 引用[.reference_title] - *1* *2* *3* [c语言链表的创建(头插法尾插法)](https://blog.csdn.net/m0_69251699/article/details/129103530)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

hardStudy_h

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值