一、字符串指针问题
思路:
①:如果要输出nice,则对应输出的字符串的下标为14,28,3,6;最简便的一个输出方式则是:
printf("%c%c%c%c",str[14],str[28],str[3],str[6]);
输出结果如下:
②:从选项中找出不同的几种输出方式:(x为某个整数)
p + x:首先p的定义是一个指针,是一个指向字符串str的指针,所以p实际上是一个字符串指针,p指向的是字符串首元素的首地址,对p进行加减操作只是对一个地址进行加减操作,没有对这个地址取值的操作,故含有这个的选项均不可以;
输出结果显示:
&p + x:在上一条中说过,p实际上是一个地址,没有对他进行取值的操作,但是在这一条中,是对指针p取地址然后再进行加减操作,实际上还是和上一条的性质一样,输出的都是一个地址的值;
输出结果显示:
*&p + x:首先想先看*&p,根据运算符的优先结合律,p先与&(取地址符)结合,先取指针p的地址,在取指针p的地址里面的值,相当于就是p的值,实际上还是对地址进行加减操作;其输出结果跟p+x一样;
输出结果显示:
*(p+x):在第一条中说过,p+x实际上代表的是一个地址,如果想要输出正确的答案,就要对这个地址进行取值操作,而 *(p+x)正好符合这一操作;
输出结果显示:
p[x]:我们都知道数组名a(假如定义一个数组 int a[10];)代表的是数组的首地址。a[i]表示的是数组第i个元素,而且指针p也是指向这个首地址,则p[i]也可以表示数组的第i个元素
输出结果显示:
*&p[x]:在上一条中说到p[x]表示的是这个数组中的第i个元素的值,则根据运算符的优先级的原则,先对p[x]进行取地址操作,再对其进行取值的操作,所运行的结果和p[x]运行结果相同;
输出结果显示:
*(str+x):str代表的是字符串的首地址,对首地址进行加减操作得到的是下面元素的首地址,再对其进行取值操作,得到的即是想要的结果;
输出结果显示:
二、define重定义
关于define重定义的用法:
简单的宏定义:#define <宏名> 字符串 例:#define PI 3.1415926
带参数的宏定义:#define <宏名>(<参数表>) <宏体> 例:#define A(x) x*x
在使用带参数的宏定义时,极易产生误解,比如在以下的程序中:
#include <stdio.h>
#define area(x) x*x
void main()
{
int y = area(2+2);
printf("%d",y);
}
按到普通人的思维来说,传进去的参数是2+2,所得到的结果应该是4*4 = 16;但是该程序的运行结果是8,任然是没有按照纯粹的简单替换的原则,让我们来看一下他到底是怎样运行的:
在该程序中,2+2是area宏中的参数,应该由它(2+2)来替换宏定义中的x,即替换成了2+2*2+2 = 8,那如果我们想要的结果是16,解决办法就是将2+2括起来,即把宏定义体中的x括起来:#define area(x) (x)*(x);对于area(2+2),则他就会被替换为(2+2)*(2+2) = 16;
程序运行结果如下:
回归正题:
对于第9题来说:#define SQR(x) x/x;
注意事项:该宏定义是带参数的宏定义,在使用该宏定义的时候,就需要注意在宏体中,参数有没有带括号;
很明显,该题是没有括号的,所以在以下代码中:
num = SQR(k-m)*SQR(k-m);
对该宏定义运算展开式应该为:
num = k-m/k-m*k-m/k-m
运算结果:num = 2 - 1/2 - 1*2 - 1/2 - 1 = 2-0-2-0-1 = -1;
如果对该宏定义加括号:#define SQR(x) (x)/(x)
则运行结果为:num = (2-1)/(2-1)*(2-1)/(2-1) = 1
运行结果如下: