在存储一大波数的时候,我们通常使用数组,但有时候数组显得不够灵活,比如有一串已经从小到大排序好的数[2 3 5 8 9 10 18 26 32]。现在需要往这串数中插入6使其得到的新序列仍符合从小到大排列。如果我们使用数组来实现这一操作,则需要将8和8后面的数字都依次往后挪一位,如果你觉得这几个数不算什么,那么100个、1000个数呢?
这样的操作显然很耽误时间,如果使用链表则会快很多,链表需要用到指针。
此时如果需要在8前面插入一个6,只需要一个间接的指针把5这个数和8这个数连接起来,并不需要再挪数字。
c语言中实现链表还需要一个malloc函数实现,这个函数可以直接给指针分配一个地址。
要定义指针只在变量名前面加上一个*就可以了。
int *p;
double *q;
一个指针可以指向一个变量的地址,比如
int a, *p;
cin >> a;
p = &a;
&这个符号我们很熟悉了,在scanf函数里面老会出现,通俗讲这行代码的意思是把a的地址给了p。
例子:
#include <iostream>
using namespace std;
int main()
{
int a;
int *p;
cin >> a;//例如输入4
p = &a;
cout << p << endl;//输出p的地址,也是a的地址
cout << *p << endl;//输出p的值,加上*,就输出值
cout << a << endl;
*p = 3;
cout << a << endl;//发现a和p一起变成了3
return 0;
}
这道题目的代码:
#include <stdio.h>
#include <stdlib.h>
//这里创建一个结构体用来表示链表的结点类型
//一个是int类型的数值
//一个是连接指针
struct node
{
int data;
struct node *next;
};
int main()
{
typedef struct node *init;
init head, p, q, t;
/*
这两句也可以写成一句
struct node *head, *p, *q, *t;
*/
int i, n, a;
scanf("%d", &n); //输入
head = NULL; //头指针设为空
for (i = 1; i <= n; i++) //循环读入n个数
{
scanf("%d", &a);
//动态申请空间,用来存放一个结点
p = (struct node *)malloc(sizeof(struct node));
p->data = a; //将数据存储到当前结点的data域中
p->next = NULL; //设置当前结点的后继指针指向空,也就是当前结点的下一个结点为空
if (head == NULL)
head = p; //如果这是第一个创建的结点,则将头指针指向这个结点
else
q->next = p; //如果不是第一个创建的结点,则将上一个结点的后继指针指向当前结点
q = p; //指针q也指向当前结点
}
scanf("%d", &a); //读入待插入的数
t = head; //从链表头部开始遍历
while (t != NULL)
{
if (t->next == NULL || t->next->data > a) //连续指向两个结点
//如果当前结点是最后一个结点或者下一个结点的值大于待插入的时候插入
{
p = (struct node *)malloc(sizeof(struct node));
//动态申请空间,用来存放新增结点
p->data = a;
p->next = t->next; //新增结点的后继指针指向当前结点的后继指针所指向的结点
t->next = p; //当前结点的后继指针指向新增结点
break;
//插入完毕跳出循环
}
t = t->next;
}
//输出链表中的所有数
t = head;
while (t != NULL)
{
printf("%d ", t->data);
t = t->next; //继续下一个结点
}
getchar();
getchar();
return 0;
}