C/C++链表的基本操作(新建,输出,删除,插入,查找,逆序,排序,释放链表,链表长度计算,查找倒数第k节点的元素)

链表的一些操作(新建,输出,删除,插入,查找,逆序,排序,释放链表,链表长度计算,查找倒数第k节点的元素)

一. 链表的概念

1、链表是物理存储单元上非连续的、非顺序的存储结构,数据元素的逻辑顺序是通过链表的指针地址实现,有一系列结点(地址)组成,结点可动态的生成。

2、结点包括两个部分:
一、存储数据元素的数据域(内存空间)
二、存储指向下一个结点地址的指针域。
3、相对于线性表顺序结构,操作复杂。

二.链表的作用

1、实现数据元素的存储按一定顺序储存,允许在任意位置插入和删除结点。

2、包括单向结点,双向结点,循环接点

3、C/C++/Java都可以实现

三.链表的优缺点

优点:链表实现数据元素储存的顺序储存,是连续的

缺点:因为含有大量的指针域,所以占用空间大,同时因为只有头结点(后面说明)是明确知道地址的,所以查找链表中的元素需要从头开始寻找,非常麻烦。

四.建立一个链表结点结构

struct linked_list
{
    int num;
    linked_list *next;
};

五.对链表的操作

介于本人初学链表,只能对单链表进行一些基础操作

1.新建链表(新建一个长度为n的链表)

linked_list* create(linked_list *head,int n)
{
    linked_list *ptr,*tail=NULL;
    head=NULL;
    for(int i=0;i<n;i++)
    {
        ptr=(linked_list*)malloc(sizeof(linked_list));
        scanf("%d",&ptr->num);
        ptr->next=NULL;
        if(!head)
            head=ptr;//对首节点进行赋值
        else
            tail->next=ptr;//新增链表尾部
        tail=ptr;//尾节点移动 
    }
    return head;
}//新建一个长度为n的链表并返回首节点的地址

2.删除(释放)链表

在单链表中我们在程序的最后加上一个释放内存的方法或者操作,这是一个很好的习惯。(我对这个操作其实不是很懂,有问题请指出,以下是我对代码)

void release(linked_list *head)
{
    linked_list *ptr=head;
    while(head)
    {   
        ptr=head;
        head=head->next;
        free(ptr);
    }
    free(head); 
    if(!head)
        cout<<"链表已删除,请重新输入链表长度\n"; 
}//删除整个链表 

3.插入一个数据到链表中的某个位置

linked_list* insert(linked_list* head, linked_list* ptr, int n, int l)
{
    linked_list* h = head;
    if (n == 1)//插入到第一个位置
    {
        ptr->next = head;
        head = ptr;
    }
    else
    {
        for (int i = 0;i < n - 2;i++)
            h = h->next;
        ptr->next = h->next;
        h->next = ptr;
    }
    return head;
}//插入某个数据到n位置 ,并返回首节点

4.删除链表中的第n个位置的的节点

我用的方法是覆盖法
当需要删除一个节点p时,只需要将p的前一个节点的next赋为p->next,但是由于是单向的,只知道p,无法知道p前一个节点,所以需要转换思路。将p下一个的节点覆盖到p节点即可,这样就等效于将p删除了。(当然呢我这并没有把那个节点给释放掉,不知道会不会出问题)

linked_list* del(linked_list* head, int l, int n)
{
    linked_list* h = head;
    if (l == 1)//删除首节点
        head = head->next;
    else//删除中间节点
    {
        for (int i = 0;i < l - 2;i++)
            h = h->next;
        h->next = h->next->next;
    }
    return head;
}//删除第l个位置的节点,并返回首节点

5.对单向链表第a到第b位置的节点反转

思路是这样的:
要求将一带链表头List head的单向链表逆序。

分析:

1). 若链表为空或只有一个元素,则直接返回;

2). 设置两个前后相邻的指针p,q. 将p所指向的节点作为q指向节点的后继;

3). 重复2),直到q为空

4). 调整链表头和链表尾


Alt
linked_list* reverse(linked_list *head,int a,int b,int n)
{
    linked_list *p=head,*q=head->next,*s=NULL,*h=head;
    if(a>b)
    {
        int temp=a;
        a=b;
        b=temp;
    }
    
    int qq=b-a,t=a-2;
    if(head==NULL||head->next==NULL||a==b)//当链表为空或者链表只有一个节点或者a==b时候,直接返回head 
        return head;

    if(a==1&&b>1)//a==1时候对链表a到b进行逆序 
    {
        b--;
        while(b--)
        {
            s=q->next;
            q->next=p;
            p=q;
            q=s;
        }
        head->next=q;
        head=p;
        return head;
    }
    
    //a!=1的时候对a到b位置进行逆序 
    a--;
    
    while(a--)
    {
        p=p->next;
        q=q->next;
    }//p移动到第a个位置,q移动到a+1个位置
    while(t--)
        h=h->next;//h是p的前一个位置
    
    while(qq--)//对第a个位置到第b个位置的链表进行倒序 
    {
        s=q->next;
        q->next=p;
        p=q;
        q=s;
    }
    h->next->next=q;
    h->next=p;
    return head;
}//对链表a到b位置的数据逆序并返回head的位置; 

6.查找链表中数据为a的第一个节点的位置

int find(linked_list *head,int a,int n)
{
    int sum=1;
    while(head)
    {
        if(head->num==a)
            break;
        sum++;
        head=head->next;
    }
    if(sum>n)
        return -1;
    return sum;
}//查找数据为a的节点的位置 

7.链表排序操作

linked_list* linked_listsort(linked_list *head,int p)//p判断进行升序还是降序操作 
{
    linked_list *slow=head,*fast=NULL;
    int temp;
    if(p==1)//升序排序 
    {
        while(slow)//冒泡思想,就不解释了 
        {
            fast=slow->next;
            while(fast)
            {
                if(fast->num<slow->num)
                {
                    temp=slow->num;
                    slow->num=fast->num;
                    fast->num=temp;
                }
                fast=fast->next;
            }
            slow=slow->next;
        } 
        cout<<"你已进行了升序排序,查看排序结果请输入1\n"; 
    }
    else//降序排序 
    {
        while(slow)
        {
            fast=slow->next;
            while(fast)
            {
                if(fast->num>slow->num)
                {
                    temp=slow->num;
                    slow->num=fast->num;
                    fast->num=temp;
                }
                fast=fast->next;
            }
            slow=slow->next;
        } 
        cout<<"你已进行了降序排序,查看排序结果请输入1\n"; 
    }
    return head;
}

8.计算链表长度

int lengthofLinklist(linked_list *head) 
{
    int len = 0;
    while(head)
    {
        head=head->next;
        len++;
    }
    return len;
}//计算链表长度

9.查找链表倒数第k个元素

//两次遍历查找倒数第k元素
linked_list* findKthTail(linked_list *head, int k) 
{
    if (head==NULL || k<=0) return NULL; 
    int len = 0;
    linked_list *root=head;
    while (head) 
    {
        head=head->next;
        len++;
    }
    if (len<k) return NULL;
    int countdown = len-k;
    while (countdown--)
        root=root->next;
    return root;
}

//快慢指针 查找倒数第k元素
linked_list* findKthTail2(linked_list* head, int k) 
{
    if (head == NULL || k <= 0) return NULL;
    linked_list *slow = head;
    linked_list *fast = head;
    for(int i=0;i<k;i++) 
    {  //快指针先走k步
        if(fast) 
            fast = fast->next;
        else 
            return NULL;//如果k>链表长度 返回NULL; 
    }
    while(fast) //相当于slow移动了 链表长度len - k步; 
    {
        fast = fast->next;
        slow = slow->next;
    }
    return slow;
}

10.代码整合

#include<bits/stdc++.h>
using namespace std;
struct linked_list
{
    int num;
    linked_list* next;
};

linked_list* reverse(linked_list* head, int a, int b, int n)
{
    linked_list* p = head, * q = head->next, * s = NULL, * h = head;
    if (a > b)
    {
        int temp = a;
        a = b;
        b = temp;
    }

    int qq = b - a, t = a - 2;
    if (head == NULL || head->next == NULL || a == b)//当链表为空或者链表只有一个节点或者a==b时候,直接返回head 
        return head;

    if (a == 1 && b > 1)//a==1时候对链表a到b进行逆序 
    {
        b--;
        while (b--)
        {
            s = q->next;
            q->next = p;
            p = q;
            q = s;
        }
        head->next = q;
        head = p;
        return head;
    }

    //a!=1的时候对a到b位置进行逆序 
    a--;

    while (a--)
    {
        p = p->next;
        q = q->next;
    }//p移动到第a个位置,q移动到a+1个位置
    while (t--)
        h = h->next;//h是p的前一个位置

    while (qq--)//对第a个位置到第b个位置的链表进行倒序 
    {
        s = q->next;
        q->next = p;
        p = q;
        q = s;
    }
    h->next->next = q;
    h->next = p;
    return head;
}//对链表a到b位置的数据逆序并返回head的位置; 

//head是首节点,l是待删除位置,n是链表长度
//这个函数的l和n和上个插入函数的不大一样,无伤大雅吧,嘿嘿
linked_list* del(linked_list* head, int l, int n)
{
    linked_list* h = head;
    if (l == 1)//删除首节点
        head = head->next;
    else//删除中间节点
    {
        for (int i = 0;i < l - 2;i++)
            h = h->next;
        h->next = h->next->next;
    }
    return head;
}//删除第l个位置的节点,并返回首节点

//head是首节点,ptr是存放待插入数据的节点,l是链表长度,n是待插入的位置
linked_list* insert(linked_list* head, linked_list* ptr, int n, int l)
{
    linked_list* h = head;
    if (n == 1)//插入到第一个位置
    {
        ptr->next = head;
        head = ptr;
    }
    else
    {
        for (int i = 0;i < n - 2;i++)
            h = h->next;
        ptr->next = h->next;
        h->next = ptr;
    }
    return head;
}//插入某个数据到n位置 ,并返回首节点


void release(linked_list* head)
{
    linked_list* ptr = head;
    while (head)
    {
        ptr = head;
        head = head->next;
        free(ptr);
    }
    free(head);
    if (!head)
        cout << "链表已删除,请重新输入链表长度\n";
}//删除整个链表 

linked_list* create(linked_list* head, int n)
{
    linked_list* ptr, * tail = NULL;
    head = NULL;
    for (int i = 0;i < n;i++)
    {
        ptr = (linked_list*)malloc(sizeof(linked_list));
        cin >> ptr->num;
        ptr->next = NULL;
        if (!head)
            head = ptr;//对首节点进行赋值
        else
            tail->next = ptr;//新增链表尾部
        tail = ptr;//尾节点移动 
    }
    return head;
}//新建一个长度为n的链表并返回首节点的地址

int find(linked_list* head, int a, int n)
{
    int sum = 1;
    while (head)
    {
        if (head->num == a)
            break;
        sum++;
        head = head->next;
    }
    if (sum > n)
        return -1;
    return sum;
}//查找数据为a的节点的位置 

int lengthofLinklist(linked_list* head)
{
    int len = 0;
    while (head)
    {
        head = head->next;
        len++;
    }
    return len;
}//计算链表长度

linked_list* linked_listsort(linked_list* head, int p)//p判断进行升序还是降序操作 
{
    linked_list* slow = head, * fast = NULL;
    int temp;
    if (p == 1)//升序排序 
    {
        while (slow)//冒泡思想,就不解释了 
        {
            fast = slow->next;
            while (fast)
            {
                if (fast->num < slow->num)
                {
                    temp = slow->num;
                    slow->num = fast->num;
                    fast->num = temp;
                }
                fast = fast->next;
            }
            slow = slow->next;
        }
        cout << "你已进行了升序排序,查看排序结果请输入1\n";
    }
    else//降序排序 
    {
        while (slow)
        {
            fast = slow->next;
            while (fast)
            {
                if (fast->num > slow->num)
                {
                    temp = slow->num;
                    slow->num = fast->num;
                    fast->num = temp;
                }
                fast = fast->next;
            }
            slow = slow->next;
        }
        cout << "你已进行了降序排序,查看排序结果请输入1\n";
    }
    return head;
}

//两次遍历查找倒数第k元素
linked_list* findKthTail(linked_list* head, int k)
{
    if (head == NULL || k <= 0) return NULL;
    int len = 0;
    linked_list* root = head;
    while (head)
    {
        head = head->next;
        len++;
    }
    if (len < k) return NULL;
    int countdown = len - k;
    while (countdown--)
        root = root->next;
    return root;
}
//快慢指针 查找倒数第k元素
linked_list* findKthTail2(linked_list* head, int k)
{
    if (head == NULL || k <= 0) return NULL;
    linked_list* slow = head;
    linked_list* fast = head;
    for (int i = 0;i < k;i++)
    {  //快指针先走k步
        if (fast)
            fast = fast->next;
        else
            return NULL;//如果k>链表长度 返回NULL; 
    }
    while (fast) //相当于slow移动了 链表长度len - k步; 
    {
        fast = fast->next;
        slow = slow->next;
    }
    return slow;
}

int main()
{
    int n, q, op, a, b, p, k;
    cout << "请输入链表长度\n";
    while (cin >> n)
    {
        linked_list* head;
        head = (linked_list*)malloc(sizeof(linked_list));
        cout << "请输入" << n << "个数据存入链表\n";
        head = create(head, n);
        cout << "请输入你要进行的操作:\n";
        cout << "1.输出链表\n";
        cout << "2.删除链表第q个位置的数据\n";
        cout << "3.输入一个数据插入到第q个位置\n";
        cout << "4.对链表a到b位置的数据进行逆序\n";
        cout << "5.查找某个元素所在节点的位置\n";
        cout << "6.对链表进行排序\n";
        cout << "7.查找链表导数第k个元素\n";
        cout << "8.删除并释放链表\n";
        while (1)
        {
            linked_list* h = head;
            cin >> op;
            if (op == 1)
            {
                h = head;
                cout << "长度为" << lengthofLinklist(h) << "的链表为:\n";
                while (h)
                {
                    cout << h->num << " ";
                    h = h->next;
                }
                cout << endl << endl << "你还要进行什么操作?" << endl;
            }
            else if (op == 2)
            {
                cout << "请输入你要删除的位置:";
                cin >> q;
                head = del(head, q, n);
                cout << "你已删除第" << q << "位置的数据,要看输出结果请输入1\n";
                n--;//链表长度-1 
            }
            else if (op == 3)
            {
                linked_list* ptr = NULL;
                ptr = (linked_list*)malloc(sizeof(linked_list));
                cout << "请输入待插入的数据:";
                cin >> ptr->num;
                ptr->next = NULL;
                cout << "你要将" << ptr->num << "这个数据插入哪个位置?\n" << "请输入待插入的位置:";
                cin >> q;
                head = insert(head, ptr, q, n);
                cout << "要看输出结果请输入1\n";
                n++;//链表长度+1; 
            }
            else if (op == 4)
            {
                cout << "请输入a和b,对链表第a位置和第b位置的链表反转\n";
                cin >> a >> b;
                head = reverse(head, a, b, n);
                cout << "已对链表第" << a << "到第" << b << "位置的数据进行了反转\n";
                cout << "要看输出结果请输入1\n";
            }
            else if (op == 5)
            {
                cout << "请输入你要查找的元素:";
                cin >> a;
                int node = find(head, a, n);
                if (node < 0)
                    cout << "链表中没有这个元素\n";
                else
                    cout << a << "在第" << node << "个位置\n";
                cout << endl << "你还要进行什么操作?" << endl;
            }
            else if (op == 6)
            {
                cout << "你要进行升序(1)排序还是降序(2)排序?请输入你选择的排序方法:";
                cin >> p;
                head = linked_listsort(head, p);

            }
            else if (op == 7)
            {
                cout << "请输入你想查找链表中倒数第几个节点位置的值:";
                cin >> k;
                if (k > lengthofLinklist(head))
                {
                    cout << "输入的长度不合法,此时链表长度为:" << lengthofLinklist(head);
                    cout << "请重新输入7 进行此操作\n";
                }
                else
                {
                    linked_list* q = findKthTail(head, k);
                    cout << "链表中倒数第" << k << "个位置节点的值为:" << q->num << endl << "你还要进行什么操作?" << endl;
                }
            }
            else if (op == 8)
            {
                release(head);
                break;
            }
            else
                cout << "输入的操作不对哦!\n请重新输入\n";
        }
    }
    return 0;
}

在借鉴了好多大牛写的博客之后,写出了我第一篇博客;
希望有任何问题大家可以指出,我会改进的!!!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容