C语言运算符优先级表(由上至下, 优先级依次递减)

运算符 结合性
()  []  ->  . 自左向右
 !  ~  ++  --  -  (type)  *  &  sizeof 自右至左
*  /  %  自左向右
+  -  自左向右
<<  >>  自左向右
<  <=  >  >=  自左向右
==  !=  自左向右
自左向右
自左向右
自左向右
&&  自左向右
||  自左向右
?:  自右至左
assignments  自右至左 
自左向右

(来自C陷阱与缺陷)



对于( )  或者  [ ] 的优先级是最高的

于是对于  *p[2]  结合的方式应该是   *(p[2])    对于  (*p)[2]  显然结合方式是括号中的 *p 优先于 [2]

具体

char *p[2]  表示p是一个存储指向char类型数据的指针的数组

char (*p)[2]  则表示*p是一个拥有两个char类型元素的数组,那么p则表示指向这个数组的指针

void test(){
char *p1[];
char (*p2)[];
printf("p1----%d\n", sizeof(p1));
printf("p2----%d\n", sizeof(p2));
return;
}

得到的结果是

可以看到p1表示的是一个拥有两个(char *)类型的数组,一个指针占有4个字节,俩个当然是8个字节

而p2仅仅就只是一个指针,只占有4个字节

常用用途


  • 对于  *p[]

在main函数的参数中有使用到,其中的 char *argv[]的argv就是一个指针数组,用来存储函数调用时传进来可变个数的参数

(详细:利用 gdb 探究main(int argc, char *argv[]){} 中的char *argv[]

int main(int argc, char *argv[])
{
something you want to do;
return ;
}

比如可以这样

char *pp[] = {"hello", "world",};

pp[0]存储字符串"hello"的第一个字符的地址,pp[1] 存储字符串"world"的第一个字符的地址

printf("%s\n", pp[]);

将会打印出 hello


  • 对于  (*p)[]

无疑就是可以作为二维数组的引用

其实二维数组的本质也是一维数组,那么当我们定义一个二维数组的时候为什么不可以使用像引用一维数组的那样的指针来引用二维数组呢?

错误样例

void arrays(){
int nums[][] = {};
int *pn = nums;
}

原因就是二维数组其实就是一个成员是一维数组的一维数组,所以它的类型是应该是数组类型

而这里的pn所指的类型是int,就会出现类型不匹配的问题

正确的方式是

void arrays(){
int nums[][] = {};
int (*p_nums)[] = nums;
}

这样的p_num是指向类型为数组的指针,与nums的类型相匹配


补充

二维数组在内存中的存储问题

示例代码

#include <stdio.h>
#include <stdlib.h> int main()
{
/*定义char类型便于观察地址,因为char只占一个byte*/
char chars[][] = {{, , , }, {, , , }, {, , , },};
printf("-------------------------------------------------------------------\n\n"); for(int i = ; i < ; ++i){
for(int j = ; j < ; ++j){
printf("%5d(%d)", chars[i][j], (int)&chars[i][j]);
}
printf("\n");
} printf("\n-------------------------------------------------------------------\n"); printf("%d\n", sizeof(chars));
printf("%d\n", (int)chars);
printf("%d\n", (int)(chars + ));/*移动了4个元素,行移动*/ printf("-------------------------------------------------------------------\n"); printf("%d\n", sizeof(chars[]));
/*可以看作是一维数组的第一个元素的地址 形如 一维数组中的 &array[0]*/
printf("%d\n", (int)&chars[]);
/*可以看作就是一维数组里的第一个元素是数组,它相当于一个数组名表示数组地址 形如 array*/
printf("%d\n", (int)chars[]); printf("-------------------------------------------------------------------\n"); return ;
}

输出

得到的结果以及代码注释已经可以说明一些问题了,就不再重复


本节完......

C语言中的 *p[2] 与 (*p)[2] 的截然不同的更多相关文章

  1. JAVA语言中的修饰符

    JAVA语言中的修饰符 -----------------------------------------------01--------------------------------------- ...

  2. Java语言中的面向对象特性总结

    Java语言中的面向对象特性 (总结得不错) [课前思考]  1. 什么是对象?什么是类?什么是包?什么是接口?什么是内部类?  2. 面向对象编程的特性有哪三个?它们各自又有哪些特性?  3. 你知 ...

  3. python语言中的编码问题

    在编程的过程当中,常常会遇到莫名其妙的乱码问题.很多人选择出了问题直接在网上找答案,把别人的例子照搬过来,这是快速解决问题的一个好办法.然而,作为一个严谨求实的开发者,如果不从源头上彻底理解乱码产生的 ...

  4. 在C语言中利用PCRE实现正则表达式

    1. PCRE简介 2. 正则表达式定义 3. PCRE正则表达式的定义 4. PCRE的函数简介 5. 使用PCRE在C语言中实现正则表达式的解析 6. PCRE函数在C语言中的使用小例子 1. P ...

  5. C语言中函数声明实现的位置

    在学习C语言的时候我遇到了这么个事情,因为之前先学习的C#,在C#编译器中,函数的声明位置不会影响编译的结果,但是在C语言中却发生了错误 先看一段代码: #include <stdio.h> ...

  6. C语言中的栈和堆

    原文出处<http://blog.csdn.net/xiayufeng520/article/details/45956305#t0> 栈内存由编译器分配和释放,堆内存由程序分配和释放. ...

  7. 在易语言中调用MS SQL SERVER数据库存储过程方法总结

    Microsoft SQL SERVER 数据库存储过程,根据其输入输出数据,笼统的可以分为以下几种情况或其组合:无输入,有一个或多个输入参数,无输出,直接返回(return)一个值,通过output ...

  8. C语言中结构体赋值问题的讨论

    今天帮师姐调一个程序的BUG,师姐的程序中有个结构体直接赋值的语句,在我印象中结构体好像是不能直接赋值的,正如数组不能直接赋值那样,我怀疑这个地方有问题,但最后证明并不是这个问题.那么就总结一下C语言 ...

  9. Coursera-Getting and Cleaning Data-week4-R语言中的正则表达式以及文本处理

    博客总目录:http://www.cnblogs.com/weibaar/p/4507801.html Thursday, January 29, 2015 补上第四周笔记,以及本次课程总结. 第四周 ...

随机推荐

  1. exgcd模板

    逆元模板P1082 #include <cstdio> #include <algorithm> int exgcd(int a, int b, int &x, int ...

  2. SSM搭建Spring单元测试环境

    原文链接:https://jingyan.baidu.com/article/93f9803f5a97a4e0e46f55c8.html SSM搭建Spring单元测试环境

  3. 关于 Java 中的 Null

    什么是Java中的Null? null在Java中是一个非常重要的概念,它最初是为了表示缺少某些东西,例如缺少用户.资源或任何东西而发明出来的.但是这也为Java程序员带来了很多麻烦,比如最常见的空指 ...

  4. Python的常用内置函数介绍

    Python的常用内置函数介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.取绝对值(abs) #!/usr/bin/env python #_*_coding:utf-8_ ...

  5. 【1】ConcurrentModificationException 异常解析和快速失败,安全失败

    目录 一.引起异常的代码 二.foreach原理 三.从ArrayList源码找原因 四.单线程解决方案 五.在多线程环境下的解决方法 一.引起异常的代码 以下三种的遍历集合对象时候,执行集合的rem ...

  6. 何凯文每日一句打卡||DAY9

  7. export DataTable To Excel(C)

          static DataTable GetTable() { DataTable table = new DataTable(); // New data table. table.Colu ...

  8. 第6月第6天 opengles 三角形

    1. http://blog.csdn.net/u010963658/article/details/52691578 2.多张图 https://www.oschina.net/question/2 ...

  9. Hive笔记之Fetch Task

    在使用Hive的时候,有时候只是想取表中某个分区的前几条的记录看下数据格式,比如一个很常用的查询: select * from foo where partition_column=bar limit ...

  10. Linux内核源码分析--内核启动之(1)zImage自解压过程(Linux-3.0 ARMv7) 【转】

    转自:http://blog.chinaunix.net/uid-25909619-id-4938388.html 研究内核源码和内核运行原理的时候,很总要的一点是要了解内核的初始情况,也就是要了解内 ...