指针是神奇的,指向整型的指针 int *p1,可以操作整型数组 int a[];
指向字符型的指针 char *p2,可以操作字符数组(字符串)charstr[];
更灵活的是,在函数的传递中,指针、数组名在一定程度上可以互换。请编制函数,对字符串的进行各种操作。
提示 1:不少函数中的参数用了 const 限定符,如 char *astrcat(char str1[], const char str2[])中的 str2,其意义在于指定 str2 数组(或称 str2 指针指向的值)可以视为常变量,是不可改变的,从而有效的“保护”了数据,免得出现意外的修改;
提示2:在完成任务时,自编 main 函数进行测试,测试中,给出的的实际参数可以是以’\0’结束的字符数组,也可以是指向字符的指针。
示例:字符串连接(任务1)的完成
(1)用数组名作形参方案,其核心是实现char *astrcat(char str1[], const char str2[])
//字符串连接函数实现和测试示例
#include <iostream>
using namespace std;
char *astrcat(char str1[], const char str2[]);
int main(){
char s1[50]="Hello world. ";
char s2[50]="Good morning. ";
char s3[50]="vegetable bird! ";
astrcat(s1,s2);
cout<<"连接后:"<<s1<<endl;
cout<<"连接后:"<<astrcat(s2,s3)<<endl; //返回值为char*型,可以直接显示
return 0;
}
//作为示例,本函数采用了形参为数组,在实现中,直接用下标法进行访问
//实际上,在实现中,完全可以用指针法访问
char *astrcat(char str1[], const char str2[])
{
int i,j;
//请理解:以下所有str1[i]可以替换为*(str1+i),str2[j]可以……
for(i=0; str1[i]!='\0'; i++); //找到str1的结束
for(j=0; str2[j]!='\0'; i++,j++) {
str1[i]=str2[j];
}
str1[i]='\0';//切记!!
return str1;
}
运行结果:
(2)用指针作形参,其核心是实现char *pstrcat(char *str1, const char *str2)函数。pstrcat函数的实现如下所示,main函数作为测试函数,请自行设计。
<pre name="code" class="cpp">#include <iostream>
using namespace std;
char *pstrcat(char str1[], const char str2[]);//astrcat变为pstrcat
int main(){
char s1[50]="Hello world. ";
char s2[50]="Good morning. ";
char s3[50]="vegetable bird! ";
pstrcat(s1,s2);
cout<<"连接后:"<<s1<<endl;//把s1换为pstrcat为相同效果
cout<<"连接后:"<<pstrcat(s2,s3)<<endl; //返回值为char*型,可以直接显示
return 0;
}
//在下面的实现中,str1用指针法访问,而str2用下标法访问
//在实际工程中,这种风格并不好,要尽可以用同一种方式,一般而言,指针法效率更高
//此处str2用下标法访问,除了作为示例的考虑外,还有一个难言之隐
//如果声明char *q,并用q=str2、q++等操作,这是非法的,因为str2有const的限制,从而赋值不兼容
//如何修改,请思考
char *pstrcat(char *str1, const char *str2)
{
char *p;
//char *q=str2; //如果这样会出现错误,如何改程序,可有多种改法
int i;
for(p=str1; *p!='\0'; p++); //找到str1的结束
for(i=0; *(str2+i)!='\0'; i++,p++) //如果q的定义通过,可以用*(q+i)代替*(str2+i)
{
*p=*(str2+i);
}
*p='\0';//切记!!
return str1;
}
运行结果: