IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 学习一下字符串(超基础,希望能够会其中的函数运用) -> 正文阅读

[数据结构与算法]学习一下字符串(超基础,希望能够会其中的函数运用)

一、字符与整数之间的关系

字符其实是以ASCII的形式在计算机中展现的。
每个常用字符都对应一个-128 ~ 127的数字,二者之间可以相互转化:
(但是ASCII为负数的情况一般用不到)

#include <iostream>

using namespace std;

int main()
{
    char c = 'a';
    cout << (int)c << endl;//强制类型转换

    int a = 66;
    cout << (char)a << endl;

    return 0;
}

常用ASCII值:'A'- 'Z'是65 ~ 90,'a' - 'z'是97 - 122,0 - 9是 48 - 57。
字符可以参与运算,运算时会将其当做整数:

#include <iostream>

using namespace std;

int main()
{
    int a = 'B' - 'A';
    int b = 'A' * 'B';
    char c = 'A' + 2;

    cout << a << endl;
    cout << b << endl;
    cout << c << endl;

    return 0;
}

二、字符串数组

字符串就是字符数组加上结束符'\0'
可以使用字符串来初始化字符数组,但此时要注意,每个字符串结尾会暗含一个'\0'字符,因此字符数组的长度至少要比字符串的长度1!

#include <iostream>

using namespace std;

int main()
{
    char a1[] = {'C', '+', '+'};            // 列表初始化,没有空字符
    char a2[] = {'C', '+', '+', '\0'};      // 列表初始化,含有显示的空字符
    char a3[] = "C++";                      // 自动添加表示字符串结尾的空字符
    char a4[6] = "Daniel";                  // 错误:没有空间可以存放空字符

    return 0;
}

2.1 字符数组的输入输出:

#include <iostream>

using namespace std;

int main()
{
    char str[100];

    cin >> str;             // 输入字符串时,遇到空格或者回车就会停止
    cout << str << endl;    // 输出字符串时,遇到空格或者回车不会停止,遇到'\0'时停止
    printf("%s\n", str);	//这里一定不要用“&”,因为,所有数组的名字就是一个指针
    puts(str);           
    return 0;
}

读入一行字符串,包括空格:

#include <iostream>

using namespace std;

int main()
{
    char str[100];
//也相当于把标准输入当作文件来读
    fgets(str, 100, stdin);  // gets函数在新版C++中被移除了,因为不安全。
  						   // 可以用fgets代替,但注意fgets不会删除行末的回车字符
    string s;
    getline(cin,s);
    char a[100];
    cin.getline(a,100);//读入一行
    cout << str << endl;

    return 0;
}

2.2 字符数组的常用操作

下面几个函数需要引入头文件:

#include <string.h>

(1) strlen(str),求字符串的长度
(2) strcmp(a, b),比较两个字符串的大小,a < b返回-1a == b返回0a > b返回1。这里的比较方式是字典序!
(3) strcpy(a, b),将字符串b复制给从a开始的字符数组。(将后面一个复制到前面的字符串上面)

#include <iostream>
#include <string.h>

using namespace std;

int main()
{
    char a[100] = "hello world!", b[100];

    cout << strlen(a) << endl;

    strcpy(b, a);

    cout << strcmp(a, b) << endl;

    return 0;
}

2.3 遍历字符数组中的字符:

#include <iostream>
#include <string.h>

using namespace std;

int main()
{
    char a[100] = "hello world!";

    // 注意:下述for循环每次均会执行strlen(a),运行效率较低,最好将strlen(a)用一个变量存下来
    for (int i = 0; i < strlen(a); i ++ )
        cout << a[i] << endl;

    return 0;
}

题目:只出现一次的字符

题目链接
练习:给定一个只包含小写字母的字符串,请你找到第一个仅出现一次的字符。如果没有,输出no。
在这里插入图片描述
考点:
1、怎么读入这个字符串。
2、如何遍历这个字符串当中的每一个字符。
3、如何统计每个字符串的出现字符的次数。
写法一:27ms

#include<bits/stdc++.h>

using namespace std;
int a[30];
int main()
{
    string s;
    cin>>s;
    for(int i=0,len=s.size();i<len;i++)
    a[s[i]-'a']++;
    bool f=1;
    for(int i=0,len=s.size();i<len;i++)
        if(a[s[i]-'a']==1)
    {
        f=0;
       printf("%c",s[i]);
        break;
    } 
    if(f)printf("no");
    
    return 0;
}

写法二:127ms

#include<iostream>
#include<cstring>
#include<map>

using namespace std;

int main()
{
    string s;
    cin>>s;
    map<char,int>m;
    for(int i=0;i<s.length();i++)
    m[s[i]]++;
    for(int i=0;i<s.length();i++)
    if(m[s[i]]==1){cout<<s[i];return 0;}
  
    cout<<"no";

    return 0;
}

可知,上面的两个写法都是用一个思想,但是用的是两个容器储存的,一个是map,一个是数组,可知,数组运行的速度比map(STL)会快很多!

题目:替换字符

题目链接
练习:把一个字符串中特定的字符全部用给定的字符替换,得到一个新的字符串。
在这里插入图片描述

写法一:

#include<bits/stdc++.h>

using namespace std;


int main()
{

    string s,k;
    cin>>s>>k;

    while(s.find(k)!=-1)
    {
        s=s.replace(s.find(k),1,"#");
    }
    cout<<s<<endl;


    return 0;
}

写法二:

#include<bits/stdc++.h>

using namespace std;


int main()
{
    char str[31];
    scanf("%s",str);
    char c;
    scanf("\n%c",&c);//这样子可以过滤掉回车,获得真实的值
    for(int i=0;str[i];i++)//由于\0的ascii是0
        if(str[i]==c)
        str[i]='#';
    puts(str);
    return 0;
}

关于replace()函数

参考了这篇博客
用法一:用str替换指定字符串从起始位置pos开始长度为len的字符

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str = "he is@ a@ good boy";
	str=str.replace(str.find("a"),2,"#");  //从第一个a位置开始的两个字符替换成#
	cout<<str<<endl; 
	return 0;
}

运行结果:
在这里插入图片描述

用法二: 用str替换 迭代器起始位置 和 结束位置 的字符

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str = "he is@ a@ good boy";
	 str=str.replace(str.begin(),str.begin()+5,"#"); //用#替换从begin位置开始的5个字符
	 cout<<str<<endl;
	 return 0; 
}

在这里插入图片描述
用法三: 用substr的指定子串(给定起始位置和长度)替换从指定位置上的字符串

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str = "he is@ a@ good boy";
	string substr = "12345";
	str=str.replace(0,5,substr,substr.find("1"),4); //用substr的指定字符串替换str指定字符串
	cout << str << endl;
	return 0; 
}

在这里插入图片描述

用法四:string转char*时编译器可能会报出警告,不建议这样做 用str替换从指定位置0开始长度为5的字符串

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str = "he is@ a@ good boy";
	char * str1 = "12345";
	str=str.replace(0,5,str1);   //用str替换从指定位置开始长度为5的字符串
	cout<<str<<endl;
	return 0; 
}

在这里插入图片描述
用法五:string转char*时编译器可能会报出警告,不建议这样做
用str替换从指定迭代器位置的字符串

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str = "he is@ a@ good boy";
	char * str1 = "12345";
	str=str.replace(str.begin(),str.begin()+6,str1);   //用str替换从指定迭代器位置的字符串
	cout<<str<<endl;
	return 0; 
}

在这里插入图片描述
用法六:string转char*时编译器可能会报出警告,不建议这样做
用s的前n个字符替换从开始位置pos长度为len的字符串

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str = "he is@ a@ good boy";
	char * str1 = "12345";
	str=str.replace(0,6,str1,4);   //用str1的前4个字符串替换从位置0~6的字符串
	cout<<str<<endl;
	return 0; 
}

在这里插入图片描述
用法七:string转char*时编译器可能会报出警告,不建议这样做
用s的前n个字符替换指定迭代器位置(从i1到i2)的字符串

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str = "he is@ a@ good boy";
	char * str1 = "12345";
	str = str.replace(str.begin(),str.begin()+6,str1,4);   //用str1的前4个字符串替换从位置0~6的字符串
	cout<<str<<endl;
	return 0; 
}

在这里插入图片描述
用法八: 用重复n次的c字符替换从指定位置pos长度为len的内容

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str = "he is@ a@ good boy";
	char  str1 = '#';
	str = str.replace(0,6,3,str1);   //用重复3次的str1字符替换的替换从位置0~6的字符串
	cout<<str<<endl;
	return 0; 
}

在这里插入图片描述
用法九: 用重复n次的c字符替换从指定迭代器位置(从i1开始到结束)的内容

#include<iostream>
#include<string>
using namespace std;
int main()
{
	string str = "he is@ a@ good boy";
	char  str1 = '#';
	str = str.replace(str.begin(),str.begin()+6,3,str1);   //用重复3次的str1字符替换的替换从指定迭代器位置的内容
	cout<<str<<endl;
	return 0;
}

在这里插入图片描述

3. 标准库类型string

可变长的字符序列,比字符数组更加好用。需要引入头文件:

#include <string>

3.1 定义和初始化

#include <iostream>
#include <string>

using namespace std;

int main()
{
    string s1;              // 默认初始化,s1是一个空字符串
    string s2 = s1;         // s2是s1的副本,注意s2只是与s1的值相同,并不指向同一段地址
    string s3 = "hiya";     // s3是该字符串字面值的副本
    string s4(10, 'c');     // s4的内容是 "cccccccccc"

    return 0;
}

3.2 string上的操作

(1) string的读写:


#include <iostream>
#include <string>

using namespace std;

int main()
{
    string s1, s2;

    cin >> s1 >> s2;
    cout << s1 << s2 << endl;

    return 0;
}

注意:不能用printf直接输出string,需要写成:printf(“%s”, s.c_str());

(2) 使用getline读取一整行

#include <iostream>
#include <string>

using namespace std;

int main()
{
    string s;

    getline(cin, s);

    cout << s << endl;

    return 0;
}

(3) string的empty和size操作(注意size是无符号整数,因此 s.size() <= -1一定成立):

#include <iostream>
#include <string>

using namespace std;

int main()
{
    string s1, s2 = "abc";

    cout << s1.empty() << endl;
    cout << s2.empty() << endl;

    cout << s2.size() << endl;//s2.length()和s2.size()是等价的

    return 0;
}

(4) string的比较:

支持 >, <, >=, <=, ==, !=等所有比较操作,按字典序进行比较。

(5) 为string对象赋值:

string s1(10, 'c'), s2;     // s1的内容是 cccccccccc;s2是一个空字符串
s1 = s2;                    // 赋值:用s2的副本替换s1的副本
                            // 此时s1和s2都是空字符串

(6) 两个string对象相加:

string s1 = "hello,  "", s2 = "world\n";
string s3 = s1 + s2;                    // s3的内容是 hello, world\n
s1 += s2;                               // s1 = s1 + s2

(7) 字面值和string对象相加:

做加法运算时,字面值和字符都会被转化成string对象,因此直接相加就是将这些字面值串联起来:

string s1 = "hello", s2 = "world";      // 在s1和s2中都没有标点符号
string s3 = s1 + ", " + s2 + '\n';

当把string对象和字符字面值及字符串字面值混在一条语句中使用时,必须确保每个加法运算符的两侧的运算对象至少有一个是string:

string s4 = s1 + ", ";  // 正确:把一个string对象和有一个字面值相加
string s5 = "hello" + ", "; // 错误:两个运算对象都不是string

string s6 = s1 + ", " + "world";  // 正确,每个加法运算都有一个运算符是string
string s7 = "hello" + ", " + s2;  // 错误:不能把字面值直接相加,运算是从左到右进行的

3.3 处理string对象中的字符

可以将string对象当成字符数组来处理:

#include <iostream>
#include <string>

using namespace std;

int main()
{
    string s = "hello world";

    for (int i = 0; i < s.size(); i ++ )
        cout << s[i] << endl;

    return 0;
}

或者使用基于范围的for语句:

#include <iostream>
#include <string>

using namespace std;

int main()
{
    string s = "hello world";

    for (char c: s) cout << c << endl;

    for (char& c: s) c = 'a';//这里只有加上了“&”才可以改变这个值

    cout << s << endl;

    return 0;
}

由于string里面的每一个字符都是char类型,所以我们就可以用auto,但是auto一个字符串的话,可能就会有问题。

题目:信息加密

题目链接
练习:密码翻译,输入一个只包含小写字母的字符串,将其中的每个字母替换成它的后继字母,如果原字母是’z’,则替换成’a’。
在这里插入图片描述
写法一:

#include<bits/stdc++.h>

using namespace std;

int main()
{
    string s;
    getline(cin,s);
    for(int i=0;i<s.size();i++)
    if(s[i]<'Z'&&s[i]>='A')
    s[i]=s[i]+1;
    else if(s[i]<'z'&&s[i]>='a')
    s[i]=s[i]+1;
    else if(s[i]=='Z')s[i]='A';
    else if(s[i]=='z')s[i]='a';
    cout<<s<<endl;
    return 0;
}

写法二:

#include<bits/stdc++.h>

using namespace std;

int main()
{
    string s;
    getline(cin,s);
   for(auto &c:s)
   if(c>='a'&&c<='z')c=(c-'a'+1)%26+'a';
   else if(c>='A'&&c<='Z')c=(c-'A'+1)%26+'A';
   cout<<s<<endl;
    return 0;
}

练习:输入两个字符串,验证其中一个串是否为另一个串的子串。

题目:字符串中的数字个数

题目链接
在这里插入图片描述
写着这道题其实是为了记录一下fgets()这个函数的用法。

#include<bits/stdc++.h>

using namespace std;

int main()
{
    char str[101];
    fgets(str,101,stdin);//getline()、cin.getline()等函数会把回车过滤掉,但是fgets()会把回车读进来
    int cnt=0;
    for(int i=0;str[i];i++)
        if(str[i]>='0'&&str[i]<='9')
        cnt++;
    printf("%d\n",cnt);
    return 0;
}

题目:循环相克令(找规律,狗熊版石头剪刀布)

链接

在这里插入图片描述
狗熊版的石头剪刀布
在这里插入图片描述

#include <cstdio>
#include <iostream>

using namespace std;

int main()
{
    int n;
    cin >> n;

    while (n -- )
    {
        string a, b;
        cin >> a >> b;

        int x, y;
        if (a == "Hunter") x = 0;
        else if (a == "Bear") x = 1;
        else x = 2;

        if (b == "Hunter") y = 0;
        else if (b == "Bear") y = 1;
        else y = 2;

        if (x == y) puts("Tie");
        else if (x == (y + 1) % 3) puts("Player1");
        else puts("Player2");
    }

    return 0;
}

若要删除字符串的最后一个字符,就要用s.pop_back()
若要将字符串中大写字符转化为小写字符,我们可以用函数tolower()

题目:字符串插入(substr()的应用)

题目链接
在这里插入图片描述

#include <iostream>

using namespace std;

int main()
{
    string a, b;

    while (cin >> a >> b)
    {
        int p = 0;
        for (int i = 1; i < a.size(); i ++ )
            if (a[i] > a[p])
                p = i;

        cout << a.substr(0, p + 1) + b + a.substr(p + 1) << endl;
    }

    return 0;
}

将字符串转化为全小写后,进行比较

#include<bits/stdc++.h>

using namespace std;

int main()
{
  string a,b;
  getine(cin,a);
  getline(cin,b);
  for(auto &c:a)c=tolower(c);
  for(auto &c:b)c=tolower(c);
  if(a==b)cout<<'='<<endl;
  else if(a>b)cout<<'>'<<endl;
  else cout<<'<'<<endl;
    return 0;
}

题目:去掉多余的空格

题目链接
在这里插入图片描述

方法一:

#include <iostream>

using namespace std;

int main()
{
    string s;
    while (cin >> s) cout << s << ' ' ;

    return 0;
}

方法二:第一类双指针算法

#include <iostream>

using namespace std;

int main()
{
    string s;
    getline(cin, s);

    string r;
    for (int i = 0; i < s.size(); i ++ )
        if (s[i] != ' ') r += s[i];
        else
        {
            r += ' ';
            int j = i;
            while (j < s.size() && s[j] == ' ') j ++ ;
            i = j - 1;
        }

    cout << r << endl;

    return 0;
}

写法三:局部性判断方法

#include <iostream>

using namespace std;

int main()
{
    string s;
    getline(cin, s);

    string r;
    for (int i = 0; i < s.size(); i ++ )
        if (s[i] != ' ') r += s[i];
        else
        {
            if (!i || s[i - 1] != ' ') r += ' ';//每次我们只看他是不是连续空格中的最后一个
        }

    cout << r << endl;

    return 0;
}

题目:单词替换

任意门
在这里插入图片描述

#include <iostream>
#include <sstream>

using namespace std;

int main()
{
    string s, a, b;

    getline(cin, s);
    cin >> a >> b;

    stringstream ssin(s);
    string str;
    while (ssin >> str)
        if (str == a) cout << b << ' ';
        else cout << str << ' ';

    return 0;
}

题目:字符串中最长的连续出现的字符

题目链接
在这里插入图片描述
双指针算法!

#include <iostream>

using namespace std;

int main()
{
    int n;
    cin >> n;

    while (n -- )
    {
        string str;
        cin >> str;

        int cnt = 0;
        char c;

        for (int i = 0; i < str.size(); i ++ )
        {
            int j = i;
            while (j < str.size() && str[j] == str[i]) j ++ ;
            if (j - i > cnt) cnt = j - i, c = str[i];
            i = j - 1;
        }

        cout << c << ' ' << cnt << endl;
    }

    return 0;
}

题目:最长单词

题目链接
在这里插入图片描述

#include <iostream>

using namespace std;

int main()
{
    string res, str;

    while (cin >> str)
    {
        if (str.back() == '.') str.pop_back();//去除掉字符串的最后一个数,字符串的尾巴这两个函数要会用
        if (str.size() > res.size()) res = str;
    }

    cout << res << endl;

    return 0;
}

题目:字符串移位包含问题

题目链接
在这里插入图片描述

#include <iostream>
#include <algorithm>

using namespace std;

int main()
{
    string a, b;

    cin >> a >> b;
    if (a.size() < b.size()) swap(a, b);

    for (int i = 0; i < a.size(); i ++ )
    {
        a = a.substr(1) + a[0];//移位方法

        for (int j = 0; j + b.size() <= a.size(); j ++ )
        {//看是是否有匹配的
            int k = 0;
            for (; k < b.size(); k ++ )
                if (a[j + k] != b[k])
                    break;
            if (k == b.size())
            {
                puts("true");
                return 0;
            }
        }
    }

    puts("false");

    return 0;
}

题目:字符串乘方(这道题目的思维超级赞)

任意门

在这里插入图片描述

#include <iostream>

using namespace std;

int main()
{
    string str;

    while (cin >> str, str != ".")
    {
        int len = str.size();

        for (int n = len; n; n -- )
            if (len % n == 0)
            {
                int m = len / n;
                string s = str.substr(0, m);
                string r;
                for (int i = 0; i < n; i ++ ) r += s;

                if (r == str)
                {
                    cout << n << endl;
                    break;
                }
            }
    }

    return 0;
}

题目:最长公共字符串后缀

任意门
在这里插入图片描述

#include <iostream>

using namespace std;

const int N = 200;

int n;
string str[N];

int main()
{
    while (cin >> n, n)
    {
        int len = 1000;
        for (int i = 0; i < n; i ++ )
        {
            cin >> str[i];
            if (len > str[i].size()) len = str[i].size();
        }

        while (len)//枚举相匹配的字符串的长度
        {
            bool success = true;
            for (int i = 1; i < n; i ++ )
            {
                bool is_same = true;
                for (int j = 1; j <= len; j ++ )
                    if (str[0][str[0].size() - j] != str[i][str[i].size() - j])
                    {
                        is_same = false;
                        break;
                    }
                if (!is_same)
                {
                    success = false;
                    break;
                }
            }

            if (success) break;
            len -- ;
        }

        cout << str[0].substr(str[0].size() - len) << endl;
    }

    return 0;
}


  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-04-22 19:01:27  更:2022-04-22 19:01:30 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/26 7:21:55-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码