04.数组

一维数组

数组名

// 如下声明
int a;
int b[10];
  • a称之为标量,即单一的一个值

  • b称之为数组,因为他是一些值的集合

  • 数组名和下标一起使用,用于标识集合中某个特定的值,例如b[0]就是b集合中的第一个值,b[4]就是第五个值,每个特定值都是一个标量

  • b[4]的类型是整型,在c语言中,数组名的值是一个指针常量,也就是数组第一个元素的内存地址.它的类型取决于数组元素的类型.

  • 数组名具有一些和指针完全不同的特征

    • 数组具有确定数量的元素,而指针只是一个标量值.编译器用数组名来记住这些元素,只有当数组名在表达式中使用时,编译器才会为它产生一个指针常亮

    • 注意数组的名称是一个指针 常量,而不是指针 变量,意味着不能修改常量的值

int a[10];
int b[10];
int *c;
c = &a[0];

表达式&a[0]是指向数组的第一个元素的指针,但那正是数组名的本身,所以
c=&a[0] 与 c=a是一致的

a = c ,c被声明为一个指针常量,这条语句看上去是把c值赋值给a,但是这个赋值是非法的,a的值是个常量,不能被修改

下标引用

*(b+3)
b的值是一个指向整型类型的指针,所以3这个值根据整型值长度进行调整.加法运算的结果是另一个指向整型的指针(数组第一个元素向后移3个整数长度的位置),*间接访问操作符访问这个新位置,或者取得那里的值(右值),或者把一个新值存储于该处(左值)

除了优先级不同,它和使用下标访问完全相同,

int array[10];
int *ap = array + 2;

// 指针加法运算时会对2进行调整,运算结果所产生的指针ap指向array[2]

  • ap array+2 与&a[2]对等

  • *ap array[2] 与 *(array+2) 对等,对数组第三个元素的内存地址执行间接访问

  • ap[0] c语言下标引用与间接访问表达式是一样的,这样,对等的表达式是*(ap+(0)) 除去0和括号,其结果与前一个表达式相等.因此,和array[2]相同

  • ap+6 ap指向array[2],加法运算产生的指针所指向的元素是array[2]向后移动6个整数位置的元素,与之相同的为array[2]+6

  • *(ap+6) 括号迫使加法运算先执行,所以得到的值是array[8]

  • ap[6] 相当于*(ap+(6)) 去除括号为array[8]

  • &ap ap的指针,是合法的

  • ap[-1] 使用偏移量-1可以得到前一个元素,相当于(ap+(-1)),也就是array[1]

  • ap[9] 看上去是正常的,但是array的长度为10,这个表达式是越界访问的,是非法的

指针与下标

对于指针和下标,两种形式都可以使用,小标容易理解,可读性强,尤其在多维数组中,但指针的效率更高

// 使用下标循环
int array[10],a;
for (a=0;a<10;a++)
    array[a] = 0;
// 使用指针循环

int array[10], *ap;
for (ap=array;ap<array+10;ap++)
    *ap = 0;

数组和指针的关系

int a[5];

int *b;

在声明一个数组时,编译器将根据所指定的元素数量分配内存空间,然后在创建数组名,它的值是一个常量,指向这段空间的起始位置。

在声明一个指针时,编译器只为指针本身保留内存空间,不为其他值分配空间,而且指针变量并不指向任何现有的内存空间。如果指针时是一个自动变量,它甚至根本不会被初始化。

作为函数参数的数组

当一个数组传递给函数时,数组名是指向该数组的常量,因此传递给数组的是一份指针的拷贝。函数如果执行了下标引用,实际上就是对这个指针执行间接访问操作,并且通过间接访问,函数可以修改和调用数组元素

由于传递给函数的是一份指针的拷贝,因此函数可以自由操作它的指针形参,而不必担心会修改对应实参的指针

当然,如果传递了一个指向数组的指针,则会修改对应的实参

声明数组参数

int strlen(char *string);

int strlen(char string[]);

首先,这两个声明是相等的,调用函数时实际传递的是一个指针.

数组初始化

int vector[5] = {1,2,3,4}

如果初始化值的数目与元素数目不匹配,编译器会自动补全

自动计算长度

int vector[] = {1,2,3,4,5,6}

可以不必指定元素数组,数组会自动计算长度

计算元素数目

int vector[] = {1,2,3,4,5,6};

void sum(int vector[]){
    int len = sizeof(vector) / sizeof(vector[0]);
    //....
}

多维数组

数组名

一维数组的数组名是指向这个数组首元素的指针常量,二维数组也差不多,唯一区别是多维数组的第一维的元素又是一个数组

int matrix[3][10]

matrix这个名字实际上指向的是二维数组的第一个元素,也就是指向一个包含10个整型元素的指针

matrix + 1 也是指向包含10个整型元素的指针

*(*(matrix + 1) + 5)

*(matrix + 1) 实际上等价于 matrix[1]的,

而后这个包含10整型的数组又移动了5个元素 也就是, *(matrix+1)+5

最后对这个表达式求值 *(*(matrix+1)+5) 等价于 martix[1][5]

指向数组的指针

int vector[10], *vp = vector;

int matrix[3][10],*mp = matrix;

第一个声明是合法的,把vp声明为一个指向整型的指针,vector和vp是相同的类型:指向整形的指针,*vp是访问这个指针保存的值,和vector[0]一个意思

第二个声明是非法的,matrix是一个多维数组,matrix并不是一个指向整型的指针,而是指向整型数组的指针,所以应该这样声明:

int (*mp)[10]

下标引用的优先级高于间接访问,由于括号的存在,首先执行的还是间接访问,所以mp是个指针,接下来执行下标引用,所以mp是指向某种类型的数组的指针。

因此 int (*mp)[10] mp指向matrix的第一行,也就是matrix[0]

如果访问数组元素而不是在数组中移动,则需要

int *p = &matrix[0][0]; 或者 int *p = matrix[0];

作为函数参数传递


int martix[3][10]

void func(int (*mp)[10])
void func(int mp[][10])

指针数组

#include <stdio.h>

void traversal(char **,int);

int main(void) {
        
    char *keyword[] = {
        "do","while","for","return","const","register"
    };
    int len = sizeof(keyword) / sizeof(keyword[0]);
    traversal(keyword,len);
    return 0;
}

void traversal(char **keyword,int size) {
    char **i;
    char *j;
    for (i = keyword; i<keyword+size;i++){
        printf("[");
        for (j=*i;*j!='\0';j++){
            printf ("%c,",*j);
        }
        printf("]->%s\n",*i);
    }
}

上面的代码中,keyword是一个字符串指针数组,通过长度来进行遍历 keyword是指向保存字符数组的字符型指针,keyword是访问字符数组,(*keyword) 是访问字符数组的第一个元素

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容

  • 指针是C语言中广泛使用的一种数据类型。 运用指针编程是C语言最主要的风格之一。利用指针变量可以表示各种数据结构; ...
    朱森阅读 3,422评论 3 44
  • 一、框架 1、Mac系统及常用工具、进制;C数据类型、常量变量、运算符、表达式、格式化输入输出 2、关系运算符、逻...
    师景福阅读 675评论 0 1
  • JavaScript 的Array可以包含任意数据类型,并通过索引来访问每个元素。 表示方式 获取数组的元素个数 ...
    已重置2020阅读 215评论 0 1
  • 我相信自己能够始终温暖,是因为年少遇见的人都这般善良。 今天是二零一七年的第九个月,刚刚开始的九月。 西...
    小小丸子小同学阅读 350评论 0 1
  • 每天写日志,很多人都挺佩服我的。每天也 会受到很多的建议与批评。 我一同学陌陌说:“有时候感觉你就像在记流 水账。...
    者行孙阅读 208评论 0 1