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

简介: 原文:C语言语法笔记 – 高级用法 指针数组 指针的指针 二维数组指针 结构体指针 链表 | IT宅.comC语言语法笔记 – 高级用法 指针数组 指针的指针 二维数组指针 结构体指针 链表 | IT宅.comC语言语法笔记 – 高级用法 指针数组 指针的指针 二维数组指针 结构体指针 链表本文由 arthinking 发表于315 天前 ⁄ itzhai.com原创文章 ⁄ C语言 ⁄ 评论数 3 ⁄ 被围观 1,775 views+  指针数组:在一个数组中,如果它的元素全部都是指针类型的数据,那么这个数组称为指针数组。
原文: 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语言, 指针, 链表
目录
相关文章
|
7天前
|
IDE 编译器 开发工具
【C语言】全面系统讲解 `#pragma` 指令:从基本用法到高级应用
在本文中,我们系统地讲解了常见的 `#pragma` 指令,包括其基本用法、编译器支持情况、示例代码以及与传统方法的对比。`#pragma` 指令是一个强大的工具,可以帮助开发者精细控制编译器的行为,优化代码性能,避免错误,并确保跨平台兼容性。然而,使用这些指令时需要特别注意编译器的支持情况,因为并非所有的 `#pragma` 指令都能在所有编译器中得到支持。
76 41
【C语言】全面系统讲解 `#pragma` 指令:从基本用法到高级应用
|
7天前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
61 14
|
11天前
|
存储 编译器 C语言
【C语言】结构体详解 -《探索C语言的 “小宇宙” 》
结构体通过`struct`关键字定义。定义结构体时,需要指定结构体的名称以及结构体内部的成员变量。
57 10
|
11天前
|
存储 算法 C语言
【C语言】深入浅出:C语言链表的全面解析
链表是一种重要的基础数据结构,适用于频繁的插入和删除操作。通过本篇详细讲解了单链表、双向链表和循环链表的概念和实现,以及各类常用操作的示例代码。掌握链表的使用对于理解更复杂的数据结构和算法具有重要意义。
64 6
|
11天前
|
传感器 算法 安全
【C语言】两个数组比较详解
比较两个数组在C语言中有多种实现方法,选择合适的方法取决于具体的应用场景和性能要求。从逐元素比较到使用`memcmp`函数,再到指针优化,每种方法都有其优点和适用范围。在嵌入式系统中,考虑性能和资源限制尤为重要。通过合理选择和优化,可以有效提高程序的运行效率和可靠性。
49 6
|
15天前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
37 5
|
15天前
|
存储 数据建模 程序员
C 语言结构体 —— 数据封装的利器
C语言结构体是一种用户自定义的数据类型,用于将不同类型的数据组合在一起,形成一个整体。它支持数据封装,便于管理和传递复杂数据,是程序设计中的重要工具。
|
14天前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
19天前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
19天前
|
存储 NoSQL 编译器
C 语言中指针数组与数组指针的辨析与应用
在C语言中,指针数组和数组指针是两个容易混淆但用途不同的概念。指针数组是一个数组,其元素是指针类型;而数组指针是指向数组的指针。两者在声明、使用及内存布局上各有特点,正确理解它们有助于更高效地编程。