C++中的字符串

字符串有三种形式

  • 用双引号括起来的字符串常量,如果"CHINA"、"C++ program"
  • 存放于字符串数组中,以'\0'字符(ASCII吗为0)结尾
  • string对象。string是C++标准模板库里的一个类,专门用于处理字符串(略)。

字符串常量

  • 字符串常量占据内存的字节数等于字符串中字符数目加1,多出来的是结尾字符'\0'
  • 字符串的长度不包含'\0'
  • ""也是合法的字符串常量,称为空串。空串仍然会占据一个字节的存储空间,存放'\0'。
  • 如果字符串常量中包含双引号,则双引号应写为'"'。而''字符在字符串中出现时,须连写两次,变成'\'。例如:
cout << "He said, \"I am a stu\\dent.\"";
// He said, "I am a stu\dent."

用一维char数组存放字符串

  • 包含'\0'字符的一维char数组,就是一个字符串。其中存放的字符串即为'\0'前面的字符所组成。
  • 用char数组存放字符串,数组元素个数应该至少为字符串长度+1
  • char数组的内容,可以在初始化时设定,也可以用C++库函数进行修改,还可以用对数组元素赋值的办法任意改变其中的某个字符。
  • 字符数组同样可以用cout、printf输出,用cin、scanf读入。用cin、scanf将字符串读入字符数组时,会自动在字符数组中字符串的末尾加上'\0'

字符串程序示例

# include <iostream>
# include <cstring> //字符串库函数的声明
using namespace std;
int main()
{
char title[] = "Prison Break"; //title的最后一个元素是'\0'
char hero[100] = "Michael Scofiled";
char prisonName[100];
char response[100];
cout << "What's the name of the prison in " << title << endl;
cin >> prisonName; // input your string
if( strcmp( prisonName, "Fox-River" ) == 0 )
cout << "Yeah! Do you love " << hero << endl;
else{
//function to copy a string
strcpy( response, "It seems you haven't watched it!\n" );
cout << response;
}
title[0] = 't';
title[3] = 0; // equal to title[3] = `\0';
cout << title << endl;
return 0;
}
What's the name of the prison in Prison Break
Fox-River
Yeah! Do you love Michael Scofiled
tri What's the name of the prison in Prison Break
Shark
It seems you haven't watched it!
tri

字符串的输入

  • 用scanf可以将字符串读入字符数组
  • scanf会自动添加结尾的'\0'
  • scanf读入到空格为止
  • 在数组长度不足的情况下,scanf可能导致数组越界
  • cin输入字符串的情况和scanf相同
char line[5];
scanf("%s", line); // Don't use &line
// cin >> line; // 若输入"12345",则数组越界
printf("%s", line);

读入一行到字符数组

  • cin.getline(char buf[], int bufSize);
  • 读入一行(行长度不超过bufSize-1)或bufSize-1个字符到buf,自动添加'\0'。回车换行符不会写入buf,会从输入流中去掉
char line[10];
cin.getline(line, sizeof(line));
// 或 cin.getline(line, 10); 读入最多9个字符到line
cout << line;

读入一行到字符数组

  • gets(char buf[])
  • 读入一行,自动添加'\0'
  • 回车换行符不会写入buf,会从输入流中去掉。如果buf的容量不足,可能导致数组越界!
char s[10];
while( gets(s) ){ // Use Ctrl + C to terminate this program
printf("%s\n", s);
}

字符串库函数

  • 使用字符串库函数需要 #include
  • 字符串库函数都根据'\0'来判断字符串结尾
  • 形参为char []类型,则实参可以是char数组或字符串常量
  • 字符串拷贝strcpy(char dest [], char src []); // copy src to dest
  • 字符串比较大小 int strcmp( char s1 [], char s2 [] ); // if equal then 0 is returned
  • 求字符串长度 int strlen( char s[] );
  • 字符串拼接 strcat(char s1[], char s2[]); // s2拼接到s1后面
  • 字符串转成大写 strupr(char s[]);
  • 字符串转成小写 strlwr( char s[] );

字符串库函数用法示例

#include <iostream>
#include <cstring>
using namespace std;
char* strupr(char* src)
{
while (*src != '\0')
{
if (*src >= 'a' && *src <= 'z')
//在ASCII表里大写字符的值比对应小写字符的值小32.
//*p -= 0x20; // 0x20的十进制就是32
*src -= 32;
src++;
}
return src;
}
void PrintSmall( char s1[], char s2[] ) // 输出词典序小的字符串
{
if( strcmp( s1, s2) <= 0 ) // if s1 is smaller than or equal to s2
cout << s1;
else
cout << s2;
}
int main(){
char s1[30];
char s2[40];
char s3[100];
strcpy(s1, "Hello");
strcpy(s2, s1);
cout << "1) " << s2 << endl; // output: 1) Hello
strcat( s1, " ,world" ); // Now s1 becomes "Hello, world"
cout << "2) " << s1 <<endl; // output: 2) Hello, world
cout << "3) "; PrintSmall("abc", s2); cout << endl; // output: 3) Hello
cout << "4) "; PrintSmall("abc", "aaa"); cout << endl; // output: 4) aaa
int n = strlen( s2 );
cout << "5) " << n << "," << strlen("abc") << endl; // output: 5) 5, 3
strupr(s1);
cout <<"6) " << s1 << endl; // output 6) HELLO, WORLD return 0;
}

strlen的常见糟糕用法

  • strlen函数的执行是需要时间的,且时间和字符串的长度成正比
  • 每次循环,都调用strlen函数,这是效率上的很大浪费
char s[100]="test";
for( int i = 0; i < strlen(s); ++i){
s[i] = s[i]+1;
}

改正版本

  • 应取出s的长度存放在一个变量里面,然后再循环的时候使用该变量
  • 或 使用 s[i]
char s[100]="test";
int len = strlen(s);
for( int i = 0; i < len; ++i){
s[i] = s[i]+1;
}

char s[100]="test";
for( int i = 0; s[i]; ++i){
s[i] = s[i]+1;
}

例题:编写判断字串的函数

编写一个函数:int Strstr(char s1[], char s2[]);

  • 如果s2不是s1的子串,返回-1
  • 如果s2是s1的子串,则返回其在s1中第一次出现的位置
  • 空串是任何串的字串,且出现位置为0
int Strstr(char s1[], char s2[]){
if(s2[0] == 0)
return 0;
for(int i = 0; s1[i]; ++i){ //枚举比较的起点
int k = i, j = 0;
for( ; s2[j]; ++j, ++k ){
if(s1[k] != s2[j])
break;
}
if(s2[j] == 0)
return i;
}
return -1;
}

程序设计与算法(一)C语言程序设计CAP之字符串的更多相关文章

  1. 【算法】C语言趣味程序设计编程百例精解

    C语言趣味程序设计编程百例精解 C/C++语言经典.实用.趣味程序设计编程百例精解(1)  https://wenku.baidu.com/view/b9f683c08bd63186bcebbc3c. ...

  2. 【MOOC课程学习记录】程序设计与算法(一)C语言程序设计

    课程结课了,把做的习题都记录一下,告诉自己多少学了点东西,也能给自己一点鼓励. ps:题目都在cxsjsxmooc.openjudge.cn上能看到,参考答案在差不多结课的时候也会在mooc上放出来. ...

  3. 01 C语言程序设计--01 C语言基础--第1章 C语言概述&第2章 GCC和GDB

    走进嵌入式开发的世界,企业级项目课程让你达到企业嵌入式应用开发要求.名师在线答疑,解决疑难.科学评测体系,系统评估学习.核心项目实........ 30 门课程 241小时12分钟 824 人学习 学 ...

  4. C语言程序设计实习报告

    C语言程序设计实习报告 简介 语言实践心得体会范文在科技高度发展的今天,计算机在人们之中的作用越来越突出.而c语言作为一种计算机的语言,我们学习它,有助于我们更好的了解计算机,与计算机进行交流,因此, ...

  5. 2018年秋季学期《C语言程序设计I》教学过程及学期总结

    一学期下来,问题很多,思考也很多,需要整理.总结,好的经验要形成规律,不好的地方要提示警醒. 教学过程小结: C语言程序设计I-第一周教学 C语言程序设计I-第三周教学 C语言程序设计I-第四周教学 ...

  6. C语言程序设计I—第三周教学

    由于本课程是从教学周的第二周开始上课,所以第二次授课是发生在第三周,为了让PTA.云班课和博客能统一,所以将教学周作为随笔的标题.本周由于处理外聘教师随意退课等事情,总结有些延后了. 第三周教学安排 ...

  7. C语言程序设计50例(经典收藏)

    本篇文章是对C语言程序设计的50个小案例进行了详细的分析介绍,需要的朋友参考下 [程序1]题目:有1.2.3.4个数字,能组成多少个互不相同且无重复数字的三位数?都是多少?1.程序分析:可填在百位.十 ...

  8. Python语言程序设计之二--用turtle库画围棋棋盘和正、余弦函数图形

    这篇笔记依然是在做<Python语言程序设计>第5章循环的习题.其中有两类问题需要记录下来. 第一是如何画围棋棋盘.围棋棋盘共有19纵19横.其中,位于(0,0)的星位叫天元,其余8个星位 ...

  9. Python语言程序设计之三--列表List常见操作和错误总结

    最近在学习列表,在这里卡住了很久,主要是课后习题太多,而且难度也不小.像我看的这本<Python语言程序设计>--梁勇著,列表和多维列表两章课后习题就有93道之多.我的天!但是题目出的非常 ...

  10. 《C语言程序设计(第四版)》阅读心得(一)

    本篇开始写我个人觉得谭浩强老师的<C语言程序设计(第四版)>中之前没有认识到,或者忘了的知识.因为本科学过,所以有些简单的东西就没有放进来了,所以可能并不是太全面. 第一章程序设计与语言 ...

随机推荐

  1. Windows系统中python3.7安装数据可视化模块Matplotlib、numpy的各种方法汇总

    安装环境:Windows10 64位Python3.7 32位 确保已经安装PIP工具命令窗口输入PIP出现以下窗口说明PIP已经成功安装 方法1:(1)在Matplotlib的官网下载电脑对应的版本 ...

  2. pta编程总结1

    7-1 打印沙漏 (20 分) 本题要求你写个程序把给定的符号打印成沙漏的形状.例如给定17个"*",要求按下列格式打印 所谓"沙漏形状",是指每行输出奇数个符 ...

  3. 金蝶KIS&K3助记码SQL数据库批量刷新

    金蝶KIS&K3助记码SQL数据库批量刷新 用的次数不多,就没有写入存储过程或者触发里面了,可以自行实现. 第一步选择对应账套的数据库,执行下面的命令,这个是一个函数. go if exist ...

  4. Java Spring Boot VS .NetCore (七) 配置文件

    Java Spring Boot VS .NetCore (一)来一个简单的 Hello World Java Spring Boot VS .NetCore (二)实现一个过滤器Filter Jav ...

  5. Spring Boot 静态资源路径分析

    最近在接触一个看公司的java后台项目(采用的耶鲁大学开源的一个cas单点登录系统),用的是框架是Spring Boot,用的模板是Thymeleaf,于是我生成一个Spring Boot的项目,并且 ...

  6. lambda表达式——写多线程

    JDK1.8 中Lambda 表达式的出现,基本可以取替原来的匿名类实现多线程的方式.下面列举常用的常用的三种情况. 一.普通开启异步线程   new Thread(() -> System.o ...

  7. linux安装git方法

    用git --version命令检查是否已经安装 在CentOS5的版本,由于yum源中没有git,所以需要预先安装一系列的依赖包.在CentOS6的yum源中已经有git的版本了,可以直接使用yum ...

  8. Windows 7 下使用 pandoc 转换文档格式

    工作中我们经常需要面对各种各样的文档格式,文档格式转换也就在所难免.通常有些文档编辑工具会提供自带的格式转换功能,但可转换格式比较有限.pandoc 正好可以解决这个问题,几乎你能见到的所有文档格式都 ...

  9. vue入门1(搭建项目)

    安装node.js 安装cnpm npm install -g cnpm --registry=http://registry.npm.taobao.org 安装vue-cli脚手架构建工具 npm ...

  10. ORACLE11g 重装系统后根据dbf恢复数据库

    1.安装一个和原系统一致的oracle 环境,主要包括版本.数据名sid.实例名.路径和数据库编码一致 2.修改listener.ora的参数 SID_LIST_LISTENER = (SID_LIS ...