Opener
当我们希望无返回值的函数可以修改主调函数中的变量值的时候,我们通常会用到指针形参的函数,操控变量的地址。
比如说,交换函数swap(int *a, int*b)
,它的两个参变量都是整型指针。
当在主调函数中进行swap(&x,&y)
操作时,指针形参a和b就指向了x和y的地址。后续在函数中操作指针a和b时,是不会改变x和y的地址的。即便在swap()函数中解引用指针交换地址上的值,x和y的地址依然是不会改变的。
有如下代码,证明
void swap(int *a,int *b){
int t;
t=*a;
*a=*b;
*b=t;
}
int main(){
int x=10,y=20;
printf("x_add:%p\ny_add:%p",&x,&y);
swap(&x,&y);
printf("x_add:%p\ny_add:%p",&x,&y);
printf("%d %d",x,y);
}
运行结果如下:
不难看出,函数调用前后,变量x和y的地址没有发生改变,函数中指针的运用只是交换了x和y地址上存储的值而已。
由此,我们知道,当变量被定义的那一刻开始,它的地址就不会发生改变。
Continue
我们再来看一道题目:
设计并测试一个函数,从一行输入中把一个单词读入一个数组中,并丢弃输入行中的其余字符。该函数应该跳过第1个非空白字符前面的所有空白。将一个单词定义为没有空白、制表符或换行符的字符序列。
出自《C primer Ver.6》编程练习11_3
有代码如下:
#include <stdio.h>
char* saveWords(char*);
int main() {
char input[1000];
//定义一个数组input储存输入流
fgets(input,1000,stdin);
//fgets()函数,将键盘输入写入数组
saveWords(input);
//通过调用函数,将input变为单词数组
printf("%s\n",input);
}
void saveWords(char*ar){
int i,j;
for(i=0;i<1000;i++){
//第一个循环,储存非空字符的下标
if(ar[j]=='\t'||ar[j]=='\n'||ar[j]==' ')
continue;
else
break;
}
for(j=i;j<1000-i;j++)
//第二个循环,丢弃掉单词后面的字符
if(ar[j]=='\t'||ar[j]=='\n'||ar[j]==' ')
ar[j]='\0';
ar=ar+i;
//ar指向为原数组的第i项
}
很显然,这个函数不能达到我们的目的。
在这个程序中,数组input的地址并没有改变,改变的仅仅是函数中指针ar的指向。所以当主调函数中最后一行打印字符串input时,它仍然是我们输入的字符串。
此时,我们可以使用返回值。
我们将函数修改如下:
char* saveWords(char*ar){
int i,j;
for(i=0;i<1000;i++){
//第一个循环,储存非空字符的下标
if(ar[j]=='\t'||ar[j]=='\n'||ar[j]==' ')
continue;
else
break;
}
for(j=i;j<1000-i;j++)
//第二个循环,丢弃掉单词后面的字符
if(ar[j]=='\t'||ar[j]=='\n'||ar[j]==' ')
ar[j]='\0';
return ar+i;
}
这一次我们将函数定义为返回一个指针变量的函数,将指针ar+i传回主调函数。
主调函数中作如下修改:
int main() {
char input[1000];
fgets(input,1000,stdin);
char *word;
word=saveWords(input);
printf("%s\n",word);
}
定义一个指针变量word,接受函数传回的指针指向,再打印该地址上储存的字符串就不会错了。
本程序中,我们没有对数组input进行直接操作,而是用指针操作再返回指针的方法,input的内容自始至终都是不变的。
自此,我们也可以总结出,指针是只读操作,它不能修改它指向的地址。