欢迎来到我的博客https://www.cnblogs.com/veis/

https://www.cnblogs.com/veis/p/14182037.html

单向链表程序

 

 

 

复制代码
/*****************************************************************************
*                                    程序说明
*            程序名称:简单单向链表
*
*            功能:    实现单向链表的创建、插入、删除、遍历和查找
*
*            作者:    Veis
*
*            注:此代码已在VC6.0及VS2010环境下编译通过,能正常实现所对应的功能
*            程序元素排序与数组下标一样,从0开始。
******************************************************************************/
#include<stdio.h>
#include<string.h>
#include<malloc.h>
#include<stdlib.h>

//宏定义数组大小,便于修改
#define N 10        

struct student
{
    char name[20];                //数据域
    struct student *next;        //指针域
};
//判断查找和删除时对应位置的内容是否为空
bool judge(struct student *p, int position)
{
    int count;
    
    //初始化很重要
    count = 0;

    while( NULL != p )
    {
        p = p->next;
        count++;
    }
    //    printf("count:%d\n",count);
    if( position > (count - 1) )
    {
        return false;
    }else 
    {
        return true;
    }
}
//创建链表
struct student * create()
{
    // head链表的头指针,即链表的首地址
    // current当前处理的元素
    //next下一个状态的指针
    struct student *current, *head, *next;
    char str[N];
    char flag;
    
    //提示用户输入信息
    printf("请输入学生姓名:\n");
    scanf("%s",str);
    
    //动态申请内存
    head = (struct student *)malloc(sizeof(struct student));
    //字符串之间不能直接复制,需要调用字符串处理函数strcpy,包含在库string.h
    strcpy( head->name , str );

    //把链表的头指针赋给当前状态指针
    current = head;
    
    //输入y/n
    printf("是否继续输入?");
    scanf("%s",&flag);

    //排除回车符的干扰
    getchar();            

    while(flag != 'n')
    {
        //提示用户输入信息
        printf("请输入学生姓名:\n");
        scanf("%s",str);

        //动态申请内存
        next = (struct student *)malloc(sizeof(struct student));
        
        //判断是否分配内存成功
        if( NULL == next )
        {
            printf("Error:内存分配失败!");
            //退出程序,作用和 return 相似
            exit(-1);        
        }
        strcpy( next->name , str );

        //把当前状态指针的下一个指向下一个链表
        current->next = next;

        //把当前状态指针指往后移一个
        current = next;
    
        printf("是否继续输入?");
        scanf("%s",&flag);

        //排除回车符的干扰
        getchar();    
    }

    //输入完成后把当前状态指针的下一个状态指向NULL
    current->next = NULL;

    return head;
}

// 遍历链表
void list(struct student *p)
{
    while(1)
    {
        printf("%s \n", p->name);

        if( p->next != NULL )
        {
            p = p->next;
        }else
        {
            break;
        }
    }
}

//插入元素到链表
void insert(struct student *p)
{
    // insert要插入的元素
    // current当前处理的元素
    struct student *insert, *current;
    char str[N];
    int position;

    current = p;

    printf("请输入需要插入的学生姓名:\n");
    scanf("%s",str);
    
    //动态申请内存
    insert = (struct student *)malloc(sizeof(struct student));

    //判断是否分配内存成功
    if( NULL == insert )
    {
        printf("Error:内存分配失败!");
        exit(-1);
    }
    strcpy( insert->name , str );
loop:
    printf("请输入需要插入的位置:\n");
    scanf("%d",&position);

    
    //查询并判断元素插入位置
    if( position > 0 )
    {
        while( position > 1 )
        {
            current = current->next;
            position--;
        }
        //插入的元素指向当前元素的下一个
        insert->next = current->next;

        //当前元素的下一个指向插入的元素
        current->next = insert;
    }
    else if( position == 0 ) 
    {
        p = insert;
        insert->next = current;
    }
    else 
    {
        printf("\nError:输入错误!\n");
        goto loop;
    }
    list(p);
}
//删除链表元素
void delet(struct student *p)
{
    // insert用来保存要删除的元素
    // current当前处理的元素
    struct student *current, *delet;
    int position;

    current = p;
    delet = current;
loop:
    printf("请输入需要删除的位置:\n");
    scanf("%d",&position);
    
    if( judge(p,position) == true)
    {
        if( position > 0 )
        {
            while( position > 1 )
            {
                current = current->next;
                position--;
            }
            //记录删除的节点
            delet = current->next;

            //使当前指针指向下一个状态
            current->next = delet->next;

            //释放要删除的内存
            free(delet);
        
        }else if( position == 0 )
        {
            //使当前指针直接指向下一个节点
            p = current->next;

            //释放内存
            free(current);
        }
        else 
        {
            printf("\nError:输入错误!\n");
            goto loop;
        }
        //调用查看效果
        list(p);
    }else 
    {
        printf("\nError:输入错误!\n");
        goto loop;
    }
}

//回显查找的元素,可用于以后的拓展,故写成一个函数
void display(struct student *p)
{
    printf("%s \n", p->name);
}

//查找元素
void search(struct student *p)
{
    struct student *current;
    int position;
    
    current = p;
loop:
    printf("请输入需要查找的位置:\n");

    scanf("%d",&position);
    if( judge(p,position) == true)
    {
        //根据输入查找对应位置
        if( position > 0 )
        {
            while( position > 1 )
            {
                current = current->next;
                position--;
            }

            p = current->next;

        }else if( position == 0 )
        {
            p = current;
        }
        else 
        {
            printf("Error:输入错误!");
            goto loop;
        }
        //显示查找的内容
        display(p);
    }else
    {
        printf("\nError:输入错误!\n");
        goto loop;
    }
}
void main()
{
    struct student *p;

    //创建链表
    p = create();

    //删除链表
//    delet(p);

    //插入元素
//    insert(p);

    //查找元素
    search(p);
}
复制代码

 

查找功能效果如下图所示:

 

posted @   veis  阅读(1023)  评论(0编辑  收藏  举报
编辑推荐:
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
阅读排行:
· Spring AI + Ollama 实现 deepseek-r1 的API服务和调用
· 《HelloGitHub》第 106 期
· 数据库服务器 SQL Server 版本升级公告
· 深入理解Mybatis分库分表执行原理
· 使用 Dify + LLM 构建精确任务处理应用
点击右上角即可分享
微信分享提示