2014秋C++第16周 项目2参考 用指针玩字符串

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,课程资源在云学堂“贺老师课堂”同步展示,使用的帐号请到课程主页中查看。 

【项目2-用指针玩字符串】
  指针是神奇的,指向整型的指针int *p1,可以操作整型数组int a[];指向字符型的指针char *p2,可以操作字符数组(字符串)char str[];更灵活的是,在函数的传递中,指针、数组名在一定程度上可以互换。请编制函数,对字符串的进行各种操作。

功能

用数组名作形参

用指针作形参

1

字符串str1和str2连接,连接后的结果存放到str1中

char *astrcat(char str1[], const char str2[])

char *pstrcat(char *str1, const char *str2)

2

去除字符串str中的特定字符c(如空格),结果仍保存到原字符串中

char *adelchar(char str[], const char c)

char *pdelchar(char *str, const char c)

3

求字符串str的长度并返回

int astrlen(char str[])

int pstrlen(char *str)

4

统计句子str中单词的个数

不再重重复,做右边的(下同)

int pwordnum(char *str)

5

去除句子中第一个单词前的空格,返回去除了空格的字符串

 

char *ptrim(char *str)

6

去除句子中所有多余的空格,返回去除了空格的字符串

 

char *palltrim(char *str)

7

比较两个字符串,返回值同strcmp()

 

int pstrcmp(const char *str1, const char *str2)

  提示1:不少函数中的参数用了const 限定符,如char *astrcat(char str1[], const char str2[])中的str2,其意义在于指定str2数组(或称str2指针指向的值)可以视为常变量,是不可改变的,从而有效的“保护”了数据,免得出现意外的修改;
  提示2:在完成任务时,自编main函数进行测试,测试中,给出的的实际参数可以是以’\0’结束的字符数组,也可以是指向字符的指针。
  提示3:在实际项目中,通过包括string.h头文件,可以调用C语言中操作字符串的函数完成相关操作,而不必专门定义。另外C++中加入的string类是更强大的支撑。不过,此处的练习,更利于同学们从微观理解,这也是专业人员要体会到的。
  

1、
功能:字符串str1和str2连接,连接后的结果存放到str1中
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;
}

char *pstrcat(char *str1, const char *str2)

#include <iostream>
using namespace std;
char *pstrcat(char *str1, const char *str2);
int main(){
    char s1[50]="Hello world. ";
    char s2[50]="Good morning. ";
    char s3[50]="vegetable bird! ";
    pstrcat(s1,s2);
    cout<<"连接后:"<<s1<<endl;
    cout<<"连接后:"<<pstrcat(s2,s3)<<endl;  //返回值为char*型,可以直接显示
    return 0;
}
//在实现中,str1用指针法访问,而str2用下标法访问
//在实际工程中,这种风格并不好,要尽可以用同一种方式,一般而言,指针法效率更高
//此处str2用下标法访问,除了作为示例的考虑外,还有一个难言之隐
//如果声明char *q,并用q=str2、q++等操作str2,这是非法的。因为str2有const的限制,从而赋值不兼容
char *pstrcat(char *str1, const char *str2)
{
    char *p;
    int i;
    for(p=str1;*p!='\0';p++); //找到str1的结束
    for(i=0;str2[i]!='\0';i++,p++)
    {
        *p=str2[i];
    }
    *p='\0';//切记!!
    return str1;
}
//2. 去除字符串str中的特定字符c(如空格),结果仍保存到原字符串中
#include <iostream>
#include <string>
using namespace std;
char* adelchar(char str[], const char c);
char* pdelchar(char *str, const char c);
int main()
{
	char s[50]="Hello world. ";
	adelchar(s,'o');
	cout<<"\""<<s<<"\"去除 o 后,字符串为:"<<s<<endl;   //这个输出有瑕疵,见评论,感谢greenbrowser12
	strcpy(s, "my name is tom.");
	adelchar(s,'m');
	cout<<"\""<<s<<"\" 去除 m 后 ,字符串为:"<<s<<endl;
	return 0;
}

char* adelchar(char str[], const char c)
{
	int i=0,j;
	for(j=0;str[j]!='\0';j++)
	{
		if(str[j]!=c)
			str[i++]=str[j];
	}
	str[i]='\0';
	return str;
}

char* pdelchar(char *str, const char c)
{
	char *p=str,*q=str;
	for(;*q!='\0';q++)
	{
		if(*q!=c)
			*p++=*q;
	}
	*p='\0';
	return str;
}
//3. 求字符串str的长度并返回
#include <iostream>
using namespace std;
int astrlen(char str[]);
int pstrlen(char *str);
int main()
{
	char s[50]="Hello world. ";
	cout<<"\""<<s<<"\"的长度为:"<<astrlen(s)<<endl;
	cout<<"\""<<s<<"\"的长度为:"<<pstrlen(s)<<endl;
	return 0;
}

int astrlen(char str[])
{
	int l;
	for(l=0;str[l]!='\0';l++);
	return l;
}

int pstrlen(char *str)
{
	int l=0;
	char *p;
	for(p=str;*p!='\0';p++) l++;
	return l;
}
//4. 统计句子str中单词的个数
#include <iostream>
#include <cstdio>
using namespace std;
int pwordnum(char *str);

int main()
{
    char s[81];
    cout<<"请输入一个句子:";
    gets(s);
    cout<<"\""<<s<<"\"中的单词数为:"<<pwordnum(s)<<endl;
    return 0;
}

int pwordnum(char *str)
{
    int i,num=0,word=0;  //word为0,代表现在并不
    for(i=0; (*(str+i)!='\0'); i++)
    {
        if (*(str+i)==' ')
            word=0;
        else if (word==0)
        {
            word=1;
            num++;
        }
    }
    return num;
}

//5. 去除句子中第一个单词前的空格
#include <iostream>
using namespace std;
void ptrim(char *str);
int main(void)
{
	char s[81]="     Some spaces before the first word.    ";
	cout<<"请输入一个句子:";
	cout<<"原句子:\""<<s<<"\""<<endl;
	ptrim(s);
	cout<<"去除前导空格后为:\"";
	cout<<s<<"\""<<endl;
	return 0;
}

void ptrim(char *str)
{
 	char *p=str,*q=str;
	while(*q==' ')
		q++;
	while(*q!='\0')
		*p++=*q++;
	*p='\0';
}
//6. 去除句子中所有多余的空格
#include <iostream>
using namespace std;

void palltrim(char str[]);
int main(void)
{
    char s[81]="     Some    spaces   are    unnecessary.    ";
    cout<<"请输入一个句子:";
    cout<<"原句子:\""<<s<<"\""<<endl;
    palltrim(s);
    cout<<"去除多余空格后为:\"";
    cout<<s<<"\""<<endl;
    return 0;
}

void palltrim(char *str)
{
    char *p=str,*q=str;
    bool notSpace=false;
    while(*q==' ')
        q++;
    while(*q!='\0')
    {
        if (*q!=' ')
        {
            notSpace=true;
            *p++=*q++; // 不是空格,复制
        }
        else if (notSpace)   //是空格,但目前是第一个(因为之前notSpace=true;的条件是遇非空格)
        {
            notSpace=false;   //第一个空格仍然要复制
            *p++=*q++;
        }
        else   //如遇第二个或更后的空格,正是由于遇到第一个空格并复制后,notSpace=false;的原因,会走到这儿
        {
            q++;   //不复制
        }
    }
    *p='\0';
}
//7. 比较两个字符串,返回值同strcmp()
#include <iostream>
#include <iomanip>
using namespace std;

int pstrcmp(char *str1, char *str2);
int main(void)
{
    char *s[81]= {"Tudou","Shanyaodan","Yangyu","Malingshu"};
    int i=0,j=0;
    cout<<"土豆大PK:"<<endl;
    cout<<setw(12)<<" ";
    for(; i<4; i++)
    {
        cout<<setw(12)<<s[i];
    }
    cout<<endl;
    for(i=0; i<4; i++)
    {
        cout<<setw(12)<<s[i];
        for(j=0; j<4; j++)
        {
            cout<<setw(12)<<pstrcmp(s[i],s[j]);
        }
        cout<<endl;
    }
    return 0;
}

//str>str2,返回1
//str==str2,返回0
//str<str2,返回-1
int pstrcmp(char *str1, char *str2)
{
    char *p=str1,*q=str2;
    while(*p==*q&&*p!='\0'&&*q!='\0')
    {
        p++;
        q++;
    }
    if (*p>*q)
        return 1;
    else if(*p<*q)
        return -1;
    else
        return 0;
}
=================== 迂者 贺利坚 CSDN博客专栏=================
|== IT学子成长指导专栏 专栏文章的分类目录(不定期更新) ==|
|== C++ 课堂在线专栏  贺利坚课程教学链接(分课程年级) ==|
|== 我写的书——《逆袭大学——传给IT学子的正能量》    ==|
===== 为IT菜鸟起飞铺跑道,和学生一起享受快乐和激情的大学 =====



时间: 2024-09-20 19:46:46

2014秋C++第16周 项目2参考 用指针玩字符串的相关文章

2014秋C++第16周 项目1参考 数组的排序

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,课程资源在云学堂"贺老师课堂"同步展示,使用的帐号请到课程主页中查看.  [项目1-数组的排序]按给定部分程序的要求,用指向数组的指针变量作为形式参数编写函数,完成排序.重点体会:(1)排序算法,可以是冒泡,也可以是选择:(2)用指向数组的指针变量作为形式参数,用数组名(即数组的地址)作为实际参数,函数中对于形参的改变实质上也就是对实参对应内存单元的改变:(3)形

2014秋C++第16周 项目3参考 用函数指针调用函数

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,课程资源在云学堂"贺老师课堂"同步展示,使用的帐号请到课程主页中查看.  [项目3-用函数指针调用函数] 将下面的程序补充完整(包括定义函数),使其能够完成图示的功能.请使用已有程序的风格. void eat(); void sleep(); void hitdoudou(); void run(void (*f)()); int main() { int iCh

2014秋C++第16周 项目4参考 为动态数组扩容

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,课程资源在云学堂"贺老师课堂"同步展示,使用的帐号请到课程主页中查看.  [项目4-为动态数组扩容] 下面的程序,利用动态数组保存学生的成绩.当再有一批学生成绩需要保存时,要为之扩容(和吃自助一样,用多少,取多少,这好),请补充完整下面的程序,实现如图所示的功能. int main( ) { int num,i,addNum; //num是小组人数 cout<

2014秋C++第16周 项目 指针和数组、函数

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,课程资源在云学堂"贺老师课堂"同步展示,使用的帐号请到课程主页中查看.  [项目1-数组的排序]按给定部分程序的要求,用指向数组的指针变量作为形式参数编写函数,完成排序.重点体会:(1)排序算法,可以是冒泡,也可以是选择:(2)用指向数组的指针变量作为形式参数,用数组名(即数组的地址)作为实际参数,函数中对于形参的改变实质上也就是对实参对应内存单元的改变:(3)形

2014秋C++第14周项目5参考-体会棋盘游戏

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,课程资源在云学堂"贺老师课堂"同步展示,使用的帐号请到课程主页中查看.  [项目5-体会棋盘游戏中的数据存储]定义一个8行8列的二维数组a[8][8].(1)为二维数组中的数据赋50以内的随机数(程序模板中setdata()函数已经完成,利用产生随机数的系统函数实现),可能的取值如图所示:(2)设计函数out()按行输出二维数组中的数据:(3)设计函数outDia

2014秋C++第8周项目5参考-定期存款利息计算器

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,实践要求见http://blog.csdn.net/sxhelijian/article/details/39493833. 课程资源在云学堂"贺老师课堂"同步展示,使用的帐号请到课程主页中查看.   [项目5:定期存款利息计算器] 输入存款金额并选择存款种类,计算出利息(不计利息税)和本息合计.要求使用switch语句,根据选择的存款种类,确定利率和存期后计算.

2014秋C++第5周项目3参考-简单分支问题体验

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,实践要求见http://blog.csdn.net/sxhelijian/article/details/39493833. 课程资源在云学堂"贺老师课堂"同步展示,使用的帐号请到课程主页中查看. [项目3-简单分支问题体验]至少选择(1)-(3)中的一个完成体验,小组共同完成(4) 阅读课件中"给两个数,求两数中的大者"例子(见课件),结合对

2014秋C++第8周项目2参考-加班费

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,实践要求见http://blog.csdn.net/sxhelijian/article/details/39493833. 课程资源在云学堂"贺老师课堂"同步展示,使用的帐号请到课程主页中查看.   [项目2:加班费,要不要?] 小贺刚上班,按工作时间小时制领取周工资,工资标准是,每小时rate元RMB.每周工作时间40小时,如果要加班,超出部分按正常工资的1.

2014秋C++第13周项目1参考-数组大折腾

课程主页在http://blog.csdn.net/sxhelijian/article/details/39152703,课程资源在云学堂"贺老师课堂"同步展示,使用的帐号请到课程主页中查看.  [项目1 - 数组大折腾](1)创建一个有20个元素的整型数组,通过初始化,为数组中的前10个元素赋初值,然后通过键盘输入后10个元素的值,从前往后(从第0个到第19个)输出数组中元素的值,每5个元素换一行. 参考解答: #include <iostream> using nam