原文:C语言语法笔记 – 高级用法 指针数组 指针的指针 二维数组指针 结构体指针 链表 | IT宅.com

C语言语法笔记 – 高级用法 指针数组 指针的指针 二维数组指针 结构体指针 链表 | IT宅.com

C语言语法笔记 – 高级用法 指针数组 指针的指针 二维数组指针 结构体指针 链表
本文由 arthinking 发表于315 天前 ⁄ itzhai.com原创文章 ⁄ C语言评论数 3 ⁄ 被围观 1,775 views+
 
指针数组:
在一个数组中,如果它的元素全部都是指针类型的数据,那么这个数组称为指针数组。

定义:类型名 *数组名[数组长度];

char *suit[3] = {"first","second","third"};
指向指针的指针:

如果一个变量存放的又是另一个指针变量的地址,则称这个指针变量为指向指针数据的指针变量,又称多级指针,简称为指向指针的指针。

定义:类型标识符 * * 指针变量名;

利用指针变量访问另一个变量就是“间接访问”,在一个指针变量中存放一个目标变量的地址,就是“单级间址”。

对于数组suit,由于数组名本身就表示地址,所以可以直接创建二级指针:

char **p;
p = suit;
#include<stdio.h>
void main(){
int a[5] = {1,3,5,7,9};
int *num[5],i;
int **p;
for(i=0;i<5;i++){
num[i] = &a[i];
}
p = num;
for(i=0;i<5;i++){
printf("%d",**p);
p++;
}
printf("\n");
}
指向二维数组的指针:



二维数组的地址:

a=a[0][0]=a[0] a+1=a[1] a[0]+1=a[0][1]

a是行指针,*a是列指针,**a表示a[0][0]的值,*a表示a[0]的地址。 a[1]+2 等价于 *(a+1)+2

在行指针前面加上一个*就转换为了列指针,若a和a+1是行指针,则*a和*(a+1)是列指针。



指向数组元素的指针变量

#include<stdio.h>
void main(){
int a[3][4] = {1,2,3,4,5,6,7,8,9,10,11,12}
int *p;
for(p = a[0]; p<a[0]+12; p++){
if((p-a[0])%4 == 0)
printf("\n");
printf("%4d",*p);
}
}
指向由m个元素构成的一维数组的指针变量

这种指针使得p+1不是指向a[0][1],而是指向a[1],p的增值以一位数组的长度为单位,这种指针称为行指针。

数据类型 (*指针变量名)[N];

int a[4][3], (*p)[3];

返回指针的函数
函数类型 * 函数名([形式参数类型声明表])
{
函数体
}
指向函数的指针

指向函数的指针的一般定义形式:

数据类型 (*指针变量名)(参数类型列表)

调用方式:

(*指针变量名)(实际参数列表)

int (*FunctionPointer)(int a);
FunctionPointer = func; //func为函数名
(*FunctionPointer)(100);
带参数的main函数
void main(int argc, char *argv[]){
函数体
}

argc表示命令行参数个数,argv表示参数数组

指向结构体的指针
struct student *p;
struct student stu;
p = &stu;
//获取子元素的三种方法:
stu.name;
(*p).name;
p->name; //指针的方法
指向结构体数组的指针

指向结构体数组的指针实际上与前面定义的指向二维数组的指针类似,可以理解为二位地址数组的行指针。

动态内存分配:

void *malloc(unsigned int size);

newptr = malloc(sizeof(struct node));

void free(void *p)

链表结构:
#include<stdio.h>
#define NULL 0
#define LEN sizeof(struct student) /*定义节点的长度*/
#define NODE struct student
struct student
{
char no[5];
float score;
struct student *next;
}; struct student *create(void);
void printlist(struct student *head);
NODE * insert(NODE *head, NODE *new, int i);
NODE * dellist(NODE *head,char no[]); void main(){
struct student *a;
struct student test1={"abc",1.0,NULL};
struct student *test2;
a = create();
printf("insert new node\n"); test2 = &test1;
a = insert(a,test2,2);
printlist(a); printf("delete node\n");
a = dellist(a,"2");
printlist(a); getch();
}
/*创建一个具有头结点的单链表,返回单链表的头指针*/
struct student *create(void){
struct student *head = NULL, *new1, *tail;
int count = 0;
for(;;)
{
new1 = (struct student *)malloc(LEN); /*申请一个新结点的空间*/
printf("Input the number of student No.%d(5bytes): ",count + 1);
scanf("%5s",new1->no);
if(strcmp(new1->no, "*") == 0) /*这里不用加取址符号,因为no就表示数组的首地址*/
{
free(new1); /*释放最后申请的结点空间*/
break; /*结束for语句*/
}
printf("Input the score of the student No.%d: ",count + 1);
scanf("%f",&new1->score);
count++;
/*将新结点插入到链表尾,并设置新的尾指针*/
if(count == 1){
head = new1; /*是第一个结点,置头指针*/
} else
tail->next = new1; /*不是第一个结点,将新结点插入到链表尾*/
tail = new1; /*设置新的尾结点*/
}
/*置新结点的指针域为空*/
new1->next = NULL;
return(head);
} /*输出链表*/
void printlist(struct student *head){
struct student *p;
p = head;
if(head == NULL) {
printf("List is empty!!!\n");
} else {
while(p!=NULL){
printf("%5s %4.1f\n", p->no,p->score);
p = p->next;
}
}
} /*插入链表结点*/
NODE * insert(NODE *head, NODE *new, int i){
NODE *pointer;
/*将新结点插入到链表中*/
if(head == NULL){
head = new; new->next = NULL;
} else {
if(i == 0){
new -> next = head;
head = new;
} else {
pointer = head;
/*查找单链表的第i个结点(pointer指向它)*/
for(;pointer != NULL && i > 1; pointer = pointer->next,i--);
if(pointer == NULL)
printf("Out of the range,can't insert new node!\n");
else { /*一般情况下pointer指向第i个结点*/
new -> next = pointer->next;
pointer->next = new;
}
}
}
return(head);
} /*删除链表*/
NODE * dellist(NODE *head,char no[]){
NODE *front; /*front表示要删除结点的前一个结点*/
NODE *cursor; /*cursor表示当前要删除的结点*/
if(head == NULL) { /*空链表*/
printf("\nList is empty\n");
return(head);
}
if(strcmp(head->no,no == 0)){ /*要删除的结点是表头结点*/
front = head;
head = head->next;
free(front);
} else { /*非表头结点*/
front = head;
cursor = head->next;
/*通过循环移动到要删除的结点的位置*/
while(cursor != NULL && strcmp(cursor->no,no) != 0) {
front = cursor;
cursor = cursor ->next;
}
if(cursor != NULL){ /*找到需要删除的结点进行删除操作*/
front->next = cursor->next;
free(front);
} else {
printf("%5s has not been found!",*no);
}
}
return(head);
}
除了文章中有特别说明,均为IT宅原创文章,转载请以链接形式注明出处。

本文链接:http://www.itzhai.com/c-language-syntax-notes-advanced-usage-of-two-dimensional-array-of-pointers-to-a-pointer-list-pointer-array-pointer-structure.html
关键字: C语言, 指针, 链表

C语言语法笔记 – 高级用法 指针数组 指针的指针 二维数组指针 结构体指针 链表 | IT宅.com的更多相关文章

  1. C 二维数组,以及自定义二维数组

    C 二维数组,以及自定义二维数组 我们通常情况下是这样定义一个二维数组的: int a[10][15]; 我们分别查看一下a,a[0],*a 都是一样的值吧 我们可以这么理解: a是一个数组的数组 a ...

  2. Android 在资源文件(res/strings.xml)定义一维数组,间接定义二维数组

    经常我们会在资源文件(res/strings.xml)定义字符串,一维数组,那定义二维数组?直接定义二维数组没找到,可以间接定义. 其实很简单,看过用过一次就可以记住了,一维数组估计大家经常用到,但是 ...

  3. Java 数组(三)二维数组

    如果一维数组的各个元素仍然是一个数组,那么它就是一个二维数组.二维数组常用于表示表,表中的信息以行和列的形式组织,第一个下标代表元素所在的行,第二个下标代表所在的列. 一.二维数组的创建 1.先声明, ...

  4. php 将一个或多个二维数组组合成一个二维数组并根据某个字段排序排序

    最近再写项目的时候,碰到一个问题:如何将一个或多个二维数组组合成一个二维数组并根据某个字段排序排序:实在是想不到哪个php库中有哪个函数能实现,只能自己写一个了,将代码写出来后,发现自己的代码繁琐,并 ...

  5. 06-01 Java 二维数组格式、二维数组内存图解、二维数组操作

    二维数组格式1 /* 二维数组:就是元素为一维数组的一个数组. 格式1: 数据类型[][] 数组名 = new 数据类型[m][n]; m:表示这个二维数组有多少个一维数组. n:表示每一个一维数组的 ...

  6. C语言提高 (3) 第三天 二级指针的三种模型 栈上指针数组、栈上二维数组、堆上开辟空间

    1 作业讲解 指针间接操作的三个必要条件 两个变量 其中一个是指针 建立关联:用一个指针指向另一个地址 * 简述sizeof和strlen的区别 strlen求字符串长度,字符数组到’\0’就结束 s ...

  7. java 基本语法(十) 数组(三) 二维数组

    1.如何理解二维数组? 数组属于引用数据类型数组的元素也可以是引用数据类型一个一维数组A的元素如果还是一个一维数组类型的,则,此数组A称为二维数组. 2.二维数组的声明与初始化 正确的方式: int[ ...

  8. JAVA中如何创建一个二维数组,然后给二维数组赋值!

    普通的赋值是:int[][] i = {{1,2,3},{2,3,4},{1,3,4}}; 如果是其他情况可以这样:比如: import java.util.* public class TT(){ ...

  9. c语言 函数返回二位数组 函数参数为二维数组

    通过typedef可以简单实现.也可以直接写. 写了两个简单的矩阵操作的函数简单示例. #include <stdio.h> #include <stdlib.h> const ...

随机推荐

  1. PHP redis操作类 个人总结

    <pre name="code" class="php"><span style="font-size:18px;"> ...

  2. UFLDL接听教程练习(来自编码器和矢量编程疏)

    最近想在深入学习研究,开始看UFLDL(unsuprisedfeature learning and deep learning)教程了.特将课后习题答案放在这里,作为一个笔记. 笔记: 1:自编码算 ...

  3. ASP.NET 异步编程

    ASP.NET 异步编程 相关博文: 异步编程 In .NET(回味无穷!!!) ASP.NET sync over async(异步中同步,什么鬼?) 本来这篇博文想探讨下异步中的异常操作,但自己在 ...

  4. P/Invoke与逆向P/Invoke

    1.在在 C# 中通过 P/Invoke 调用Win32 DLL这篇文中,详细介绍了P/Invoke的基本知识以及使用. 2.InAttribute和OutAttribute特性与C#中ref和out ...

  5. Design Pattern Command 命令设计模式

    这种设计模式是使用不同类的包裹不同的命令,达到什么样的命令执行什么操作. 有可能进一步利用map您最喜欢的对接命令字. 正在运行的类实际上已经包含了操作的所有需求,例如: class SuperMak ...

  6. HDU3549 Flow Problem 【最大流量】

    Flow Problem Time Limit: 5000/5000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others) Tot ...

  7. bootstrap之Click大事

    上一篇文章中谈到了bootstrap流程,本文开始把目光bootstrap它可以接受指令(从源代码视图的透视.因为appium该项目现在还处于不断更新,因此,一些指令已经实现.也许未来会实现一些.从视 ...

  8. ExtJS4 动态生成grid出口excel(纯粹的接待)

    搜索相当长的时间,寻找一些样本,因为我刚开始学习的原因,大多数人不知道怎么用.. 他曾在源代码.搞到现在终于实现了主下载.. 表的采集格不重复下载一个小BUG,一个使用grid初始化发生的BUG 以下 ...

  9. Android开发模板------自己定义SimpleCursorAdapter的使用

    使用SimpleCursorAdapter所设计的table(数据表)一定要有_id字段名称,否则会出现"找不到_id"的错误 SimpleCursorAdapter直接使用的方法 ...

  10. Matlab.NET混合编程技巧之——直接调用Matlab内置函数(附源码)

    原文:[原创]Matlab.NET混合编程技巧之--直接调用Matlab内置函数(附源码) 在我的上一篇文章[原创]Matlab.NET混编技巧之——找出Matlab内置函数中,已经大概的介绍了mat ...