对链表实现插入排序C语言

思路:插入排序就是从第一个数开始排,第一个数就是有序的,后面第二个数和他比较,如果比他大,这两个数就是有序序列,往下遍历,如果比他小则从有序序列的头开始遍历,找到他的位置,插入。

在链表中为了实现这个排序,需要两个指针,一个指针p指向当前需要排的数,一个指针r作为他的前继,p与r比较,如果有序,p和r都直接往下遍历;如果无序则在前面的有序序列找到p的位置。

r从头结点开始,作为一个有序序列,p从第二个数开始比较

这里引入了一个虚拟结点q插在链表的表头,这样可以通过q的下一个结点的值与p进行比较,找到位置之后p可以直接插在p>next,这样就不需要再引入一个前继指针来找位置了。

之前一直发现链表交换会断开,后面才明白是因为交换结点之后头结点没有更新,所以他一直都指向第一个数。要修改这个问题,只要在每次交换之后检查头结点有没有被交换,被交换就更新头结点。

#include <stdio.h>
#include <malloc.h>
struct cell {//单链表结点结构体定义
 int x;
 struct cell* next;
};
struct cell* build(void) {//新建单链表,并将建好的单链表首结点地址返回
 struct cell* head, * tmp, * p;
 head = tmp = p = NULL;
 int n;
  while(scanf("%d",&n),n!=0)
 {
     p=(struct cell*)malloc(sizeof(struct cell));
     p->x=n;
     p->next=NULL;
     if(head==NULL)//必须有这个判断,否则head还是原来的指针空间,没有指向结构体的空间,会出错,与链表断裂
     {
         head=p;
         tmp=p;
     }else{
      tmp->next=p;
     tmp=p;
     }

 }

 return head;//返回单链表头
}
struct cell * sort(struct cell* head) {//递增排序链表,head是单链表首结点指针
 struct cell* p, * p0, * r, * r0, * q;
 p = p0 = r = r0 = q = NULL;
if(head==NULL)
    return NULL;
 q=(struct cell*)malloc(sizeof(struct cell));//加一个虚拟结点,方便排序
q->x=0;q->next=head;
p=head->next;
r=head;
while(p!=NULL)//p是当前需要排序的结点
{
    if(r->x<p->x)//有序不需要交换,向下遍历
    {r=p;
        p=p->next;

    }else
    {
        p0=q;
        while(p0->next->x<p->x)
            p0=p0->next;//找位置

             r->next=p->next;//交换结点,先断后面的指针,再变前面
            p->next=p0->next;
             p0->next=p;
if(p=q->next)//如果p换到了头的位置
{
    head=p;//更新头结点
    p=r->next;//更新p
}else p=r->next;




    }
}

 return head;
}


void print(struct cell* head) {//打印整个单链表,head是单链表首结点指针
    struct cell*p;

            p=head;

            while(p!=NULL)

            {if(p->next==NULL)

           {
               printf("%d",p->x);

               p=p->next;//老是忘记这个,如果没有这条语句就会无限循环,因为p不会==NULL;

           }

            else

               {printf("%d ",p->x);

                p=p->next;

            }
}
}
void release(struct cell* head) {//释放单链表空间,head是单链表首结点指针
  struct cell*p;
    p=head;
    while(head!=NULL)

    {
        p=head->next;
        free(head);
        head=p;
    }
}
int main(void) {
 struct cell* head;
 head= build();
 if (head != NULL) {
  head = sort(head);
  print(head);
 }else
  printf("NULL");
 release(head);
 return 0;
}

头结点很重要,一定要注意他的变化

  • 11
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
插入排序也是一种简单有效的排序算法,对于链表的排序也可以采用插入排序算法。 C 语言实现代码如下: ```c #include <stdio.h> #include <stdlib.h> // 链表节点结构体 struct ListNode { int val; struct ListNode *next; }; // 插入排序函数 struct ListNode* insertionSortList(struct ListNode* head) { if (head == NULL || head->next == NULL) { return head; } struct ListNode *newHead = (struct ListNode *)malloc(sizeof(struct ListNode)); newHead->val = -1; newHead->next = head; struct ListNode *cur = head->next, *pre = head; while (cur != NULL) { if (cur->val >= pre->val) { pre = cur; cur = cur->next; } else { struct ListNode *p = newHead; while (p->next->val < cur->val) { p = p->next; } pre->next = cur->next; cur->next = p->next; p->next = cur; cur = pre->next; } } return newHead->next; } // 创建链表 struct ListNode* createList(int arr[], int n) { struct ListNode *head = NULL, *cur = NULL; for (int i = 0; i < n; i++) { struct ListNode *node = (struct ListNode *)malloc(sizeof(struct ListNode)); node->val = arr[i]; node->next = NULL; if (head == NULL) { head = node; } else { cur->next = node; } cur = node; } return head; } // 打印链表 void printList(struct ListNode* head) { while (head != NULL) { printf("%d ", head->val); head = head->next; } printf("\n"); } int main() { int arr[] = {4, 2, 1, 3}; int n = sizeof(arr) / sizeof(int); struct ListNode *head = createList(arr, n); printf("Original list: "); printList(head); head = insertionSortList(head); printf("Sorted list: "); printList(head); return 0; } ``` 在这个代码中,我们定义了一个 `ListNode` 结构体表示链表节点,其中包含一个 `val` 字段和一个 `next` 指针指向下一个节点。 我们使用 `insertionSortList` 函数对链表进行插入排序。在函数中,我们首先创建一个新的头节点 `newHead`,将其值设为 `-1`,将其 `next` 指向原链表的头节点 `head`。 接下来,我们使用两个指针 `pre` 和 `cur` 分别指向已排序部分的末尾节点和未排序部分的头节点。如果 `cur` 的值大于等于 `pre` 的值,则将 `pre` 和 `cur` 向后移动一个节点;否则,我们需要将 `cur` 插入到已排序部分中。 为了找到 `cur` 应该插入的位置,我们使用一个指针 `p` 遍历已排序部分。如果 `p->next->val` 小于 `cur->val`,则将 `p` 向后移动一个节点;否则,我们将 `cur` 插入到 `p` 后面。 最后,我们返回 `newHead->next`,即为排序后的链表。 通过 `createList` 函数创建链表,并通过 `printList` 函数打印链表

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值