C语言语法笔记 – 高级用法 指针数组 指针的指针 二维数组指针 结构体指针 链表 | IT宅.com
原文: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语言语法笔记 – 高级用法 指针数组 指针的指针 二维数组指针 结构体指针 链表 | IT宅.com的更多相关文章
- C 二维数组,以及自定义二维数组
C 二维数组,以及自定义二维数组 我们通常情况下是这样定义一个二维数组的: int a[10][15]; 我们分别查看一下a,a[0],*a 都是一样的值吧 我们可以这么理解: a是一个数组的数组 a ...
- Android 在资源文件(res/strings.xml)定义一维数组,间接定义二维数组
经常我们会在资源文件(res/strings.xml)定义字符串,一维数组,那定义二维数组?直接定义二维数组没找到,可以间接定义. 其实很简单,看过用过一次就可以记住了,一维数组估计大家经常用到,但是 ...
- Java 数组(三)二维数组
如果一维数组的各个元素仍然是一个数组,那么它就是一个二维数组.二维数组常用于表示表,表中的信息以行和列的形式组织,第一个下标代表元素所在的行,第二个下标代表所在的列. 一.二维数组的创建 1.先声明, ...
- php 将一个或多个二维数组组合成一个二维数组并根据某个字段排序排序
最近再写项目的时候,碰到一个问题:如何将一个或多个二维数组组合成一个二维数组并根据某个字段排序排序:实在是想不到哪个php库中有哪个函数能实现,只能自己写一个了,将代码写出来后,发现自己的代码繁琐,并 ...
- 06-01 Java 二维数组格式、二维数组内存图解、二维数组操作
二维数组格式1 /* 二维数组:就是元素为一维数组的一个数组. 格式1: 数据类型[][] 数组名 = new 数据类型[m][n]; m:表示这个二维数组有多少个一维数组. n:表示每一个一维数组的 ...
- C语言提高 (3) 第三天 二级指针的三种模型 栈上指针数组、栈上二维数组、堆上开辟空间
1 作业讲解 指针间接操作的三个必要条件 两个变量 其中一个是指针 建立关联:用一个指针指向另一个地址 * 简述sizeof和strlen的区别 strlen求字符串长度,字符数组到’\0’就结束 s ...
- java 基本语法(十) 数组(三) 二维数组
1.如何理解二维数组? 数组属于引用数据类型数组的元素也可以是引用数据类型一个一维数组A的元素如果还是一个一维数组类型的,则,此数组A称为二维数组. 2.二维数组的声明与初始化 正确的方式: int[ ...
- JAVA中如何创建一个二维数组,然后给二维数组赋值!
普通的赋值是:int[][] i = {{1,2,3},{2,3,4},{1,3,4}}; 如果是其他情况可以这样:比如: import java.util.* public class TT(){ ...
- c语言 函数返回二位数组 函数参数为二维数组
通过typedef可以简单实现.也可以直接写. 写了两个简单的矩阵操作的函数简单示例. #include <stdio.h> #include <stdlib.h> const ...
随机推荐
- 使用JMX实现的内存监控(转)
public final class MemoryWarningSystem { private static MemoryWarningSystem m_instance = null; /** * ...
- 一C++PSO(PSO)算法
收集和变化PSO算法,它可用于参考实施: #include <cstring> #include <iostream> #include <cmath> #incl ...
- Maven+Spring
Maven+Spring 关于Maven Maven是一个用于项目构建的工具,通过它便捷的管理项目的生命周期.即项目的jar包依赖,开发,测试,发布打包. 做过.NET的人应该会联想到Nuget,是的 ...
- 彩色图像--色彩空间 CMY(K)场地
得知DIP文章63日 转载请注明文章出处:http://blog.csdn.net/tonyshengtan .出于尊重文章作者的劳动,转载请标明出处!文章代码已托管.欢迎共同开发:https://g ...
- imagick获取图片的大小bug
<? php /* imagick的获取图片的高度和宽度函数有问题,使用GD函数可获得正确结果 gd函数 array getimagesize ( string $filename [, arr ...
- (大数据工程师学习路径)第二步 Vim编辑器----Vim快速入门
vim模式介绍 以下介绍内容来自维基百科Vim 从vi演生出来的Vim具有多种模式,这种独特的设计容易使初学者产生混淆.几乎所有的编辑器都会有插入和执行命令两种模式,并且大多数的编辑器使用了与Vim截 ...
- JAVA连接ACCESS、MYSQL、SQLSEVER、ORACLE数据库
. 概要 1.1 JDBC概念 JDBC(Java Database Connectivity)是Java语言为了支持SQL功能而提供的与数据库连接的用户的接口.JDBC中包含了一组由(Java)语言 ...
- 用户 'IIS APPPOOL\IdealTest' 登录失败解决方案
原文:用户 'IIS APPPOOL\IdealTest' 登录失败解决方案 运行MVC框架后可能会提示“用户 'IIS APPPOOL\IdealTest' 登录失败” 详细堆栈信息如下 说明: 执 ...
- 恢复SQLSERVER被误删除的数据
原文:恢复SQLSERVER被误删除的数据 恢复SQLSERVER被误删除的数据 曾经想实现Log Explorer for SQL Server的功能,利用ldf里面的日志来还原误删除的数据 这里有 ...
- Ohloh研究报告
1.底 由于近期接合ospaf同样是一个开源项目的成熟度分析工具,由于该项目现在Ohloh我们有共同的东西,这么ohloh我们进行了调查. 简单的说,初始ohloh是一个代码搜 ...