C语言深入探索之单链表与typedef的用法

 更新时间:2022年05月18日 09:41:38   作者:对象new不出来  
typedef为C语言的关键字,作用是为一种数据类型定义一个新名字,单链表是后面要学的双链表以及循环链表的基础,要想继续深入了解数据结构以及C语言,我们就要奠定好这块基石!接下来就和我一起学习吧

前言

昨天博主去本站问答贴子逛了逛,然后发现了好多关于数据结构线性表,具体来说是单链表的问题。有的是没有一点思路,无从下手;有的是看不懂代码,不理解关键字以及被形参的形式气的不行,我总结了一下常见问题来给大家带来干货,到后面还有简单案例来巩固知识,弄透一题胜无脑刷百题,接下来是正文内容。

详解typedef关键字

含义

C语言允许用户使用 typedef 关键字来定义自己习惯的数据类型名称,来替代系统默认的基本类型名称、数组类型名称、指针类型名称与用户自定义的结构型名称、共用型名称、枚举型名称等。一旦用户在程序中定义了自己的数据类型名称,就可以在该程序中用自己的数据类型名称来定义变量的类型、数组的类型、指针变量的类型与函数的类型等。

具体使用

单链表结点示例:

typedef struct node
{
	int data;//数据域
	struct node * next;//指针域
}Lnode, * SLinkList;

这里我们创建了 node 结构体,结构体里面包含了 整型数据data,指针next 指向下一个结点的地址。可以看到下面大括号外有 Lnode和*SLinkList。他的意思就是我们给node 起了一个别名叫Lnode,Lnode具有和node相同的结构,struct node n1与Lnode n1 效果完全相同;同时C语言还允许在结构中包含指向它自己的指针,即 SLinkList L 等价于 struct node* L 或者 Lnode *L;

详解单链表参数形式

指针知识补充

示例:

//1、
int* getValue1(int *&L)
{
	int a = 10;
	L = &a;
	return L;
}
//2、
int* getValue2(int *L)
{
	int a = 10;
	L = &a;
	return L;
}
int main()
{
	int* ptr = (int*)malloc(sizeof(int) * 4);
	getValue1(ptr);
    getValue2(ptr);
	cout << *ptr << endl;
}

我在主函数中为指针ptr动态分配了内存空间,大小为4*4字节,不理解的可以参考我的这篇博文详解数据结构线性表里面的动态分布内存函数malloc;然后将地址传递到上面两个函数里面去,输出getValue1 的*ptr 结果是10,但是getValue2 中*ptr 的结果却是乱码。那么原因很明显,就是参数 int *&L和 int *L的区别了。这个函数的返回值是一个地址,然而在栈区开辟的数据,在函数调用结束后就会被编译器自动释放掉,从而返回的地址不会是&a,因此仅仅使用地址传递是不行的。那么加上&为什么就可以了呢,这是因为我们的目的是改变传入指针所指向的地址,第一种只能改变该地址对应的数值,第二种可以从本质上更改地址,所以能得到 &a 从而*ptr结果为10。

单链表形参详解

示例:

typedef struct node
{
	int data;//数据域
	struct node * next;//指针域
}Lnode, * SLinkList;
SLinkList Init_List(Lnode *&L)
{
	L = (SLinkList)malloc(sizeof(Lnode));
	L->next = NULL;
	return L;
}

同样的,根据我前面讲的知识,我定义了一个SLinkList 指针型的初始化链表函数 Init_List(),传入的是结构体指针变量L,接下来为L分配内存空间,这里sizeof(Lnode)是计算了结构体Lnode所占内存大小并将此内存分配给L,接下来让初始化L,让其指针域为空。实际上L->data =NULL,L 就是单链表中的头结点。这段代码是没有问题的,但是如果把形参中的Lnode *&L,改为Lnode *L,那么编译器必然会提示我们,取消对NULL指针的使用,这就是为什么我们要加上&的原因,不加&返回的不是我们分配的指针变量L的地址,那这样我们的初始化毫无意义,虽然不报错,但是毫无作用。所以提醒你们写数据结构的时候记住这个小细节,很重要的!

单链表实战案例

完整代码实现

#include<iostream>
using namespace std;
#define SIZE 10
typedef struct node
{
	int data;//数据域
	struct node * next;//指针域
 
}Lnode, * SLinkList;
SLinkList Init_List(Lnode *&L)
{
	L = (SLinkList)malloc(sizeof(Lnode));
	L->next = NULL;
	return L;
}
SLinkList Creat_List(Lnode* &L,int n)//头插建表
{
	srand((unsigned int)time(NULL));
	SLinkList p = L;
	for (int i = 0; i < n; i++)
	{
		int e = rand() % 20 + 1;
		SLinkList s = (SLinkList)malloc(sizeof(Lnode));
		s->next = p->next;
		p->next = s;
		s->data = e;
	}
	return L;
}
int count_List(Lnode *& L)
{
	int count = 0;
	SLinkList p = L->next;
	while (p)
	{
		count++;
		p = p->next;
	}
	return count;
}
int find_List(SLinkList L,int v)
{
	SLinkList p = L->next;
	int i = 1;
	while (i < v && p->next)
	{
		p = p->next;
		i++;
	}
	return p->data;
}
void display_List(SLinkList L)
{
	SLinkList p = L->next;
	while (p)
	{
		cout << p->data << " ";
		p = p->next;
	}
}
int main()
{
	srand((unsigned int)time(NULL));
	int n = rand()%5 + 5, count = 0, v = 0;
	SLinkList L;
	L = Init_List(L);
	cout << "随机插入元素完成:"<<endl;
	L = Creat_List(L,n);
	count = count_List(L);
	cout << "单链表长度为:" << count << endl;
	cout << "遍历单链表结果为:" << endl;
	display_List(L);
	cout << endl;
	cout << "要查找元素的位置为:"; cin >> v;
	int value = find_List(L, v);
	cout << "查找结果为:" << value << endl;
}

详解头插建表

把头插建表方法复制过来

SLinkList Creat_List(Lnode* &L,int n)//头插建表
{
	srand((unsigned int)time(NULL));
	SLinkList p = L;
	for (int i = 0; i < n; i++)
	{
		int e = rand() % 20 + 1;
		SLinkList s = (SLinkList)malloc(sizeof(Lnode));
		s->next = p->next;
		p->next = s;
		s->data = e;
	}
	return L;
}

这里不用管srand((unsigned int)time(NULL));这段代码是为了产生不同随机数,和我讲的内容没有什么联系;往下看,创建建构体指针变量p并把p设置为头指针:L是头结点,p=L,p指向链表第一个带值结点。进入循环语句,循环语句n我会利用随机数产生;e是1~20范围内随机的一个数值,结构体指针变量s被动态分配内存空间s->next = p>next; p->next = s;s->data = e;这三行代码是头插法的核心。首先待插入原色s指针指向头结点指向的结点,然后头指针指向s,这样链表的链就连好了,最后给s的数据域赋值为e,执行循环语句,这样新插入的结点都会在上一个插入结点的前面,这就是头插法的全部过程。

int count_List(Lnode *& L); 方法对应第一问;

int find_List(SLinkList L,int v);方法对应第二问;

相信认真看完这篇博文的你可以很好理解上面两种方法的含义以及整个源码表达的意思,最后附上运行效果截图:

运行效果

到此这篇关于C语言深入探索之单链表与typedef的用法的文章就介绍到这了,更多相关C语言单链表与typedef内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C语言16进制与ASCII字符相互转换

    C语言16进制与ASCII字符相互转换

    大家好,本篇文章主要讲的是C语言16进制与ASCII字符相互转换,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下
    2022-01-01
  • C++详解PIMPL指向实现的指针

    C++详解PIMPL指向实现的指针

    PIMPL 是 C++ 中的一个编程技巧,意思为指向实现的指针。具体操作是把类的实现细节放到一个单独的类中,并用一个指针进行访问
    2022-07-07
  • C++ AVL树插入新节点后的四种调整情况梳理介绍

    C++ AVL树插入新节点后的四种调整情况梳理介绍

    AVL树是高度平衡的而二叉树,它的特点是AVL树中任何节点的两个子树的高度最大差别为1,本文主要给大家介绍了C++如何实现AVL树,需要的朋友可以参考下
    2022-08-08
  • C语言实现简易扫雷程序

    C语言实现简易扫雷程序

    这篇文章主要为大家详细介绍了C语言实现简易扫雷程序,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-07-07
  • VC++进度条process Bar的用法实例

    VC++进度条process Bar的用法实例

    这篇文章主要介绍了VC++进度条process Bar的用法,是进行VC++应用程序开发中非常常见的实用技巧,需要的朋友可以参考下
    2014-10-10
  • 基于Qt+OpenCV实现图像灰度化像素

    基于Qt+OpenCV实现图像灰度化像素

    在图像处理领域,OpenCV是一款强大而广泛应用的开源库,能够提供丰富的图像处理和计算机视觉功能,本文将介绍如何利用Qt 编辑器调用OpenCV库对照片进行换底色处理,实现更加独特和吸引人的效果
    2023-11-11
  • Qt通过图片组绘制动态图片

    Qt通过图片组绘制动态图片

    这篇文章主要为大家详细介绍了Qt通过图片组绘制动态图片,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-07-07
  • c++栈内存和堆内存的基本使用小结

    c++栈内存和堆内存的基本使用小结

    本文主要介绍了c++栈内存和堆内存的基本使用小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-07-07
  • 浅析C++浅拷贝与深拷贝的联系和区别

    浅析C++浅拷贝与深拷贝的联系和区别

    在c++中,深拷贝和浅拷贝也算是一个难点,特别是对于初学者来说,往往在不知道两者区别的情况下而错误的使用了浅拷贝,从而导致了野指针之类的问题,但是又因为缺少理解所以很难定位到问题所在
    2022-09-09
  • VS+Qt+Halcon 显示图片,实现鼠标缩放、移动图片

    VS+Qt+Halcon 显示图片,实现鼠标缩放、移动图片

    本篇博文记录一下,用VS+Qt+Halcon实现对图片的读取以及鼠标缩放,移动(鼠标事件调用了halcon自带的算子)的过程。感兴趣的可以了解一下
    2021-08-08

最新评论