字符串查找与类型转换(C/C++)

字符串查找

strchr,strrchr与strstr

功能:对字符串中的单个字符进行查找

//strchr 函数原型的一般格式
char *strchr(const char *str, int c);

它表示在字符串str中查找字符,返回字符c第一次在字符串str中出现的位置,如果未找到字符c,则返回NULL。也就是说, strchr 函数在字符串str中从前到后(或者称为从左到右)查找字符c,找到字符c第一次出现的位置就返回,返回值指向这个位置,如果找不到字符c就返回NULL。

//strrchr 函数原型的一般格式
char *strrchr(const char *str, int c);

与 strchr 函数一样,它同样表示在字符串str中查找字符c,返回字符c第一次在字符串str中出现的位置,如果未找到字符c,则返回NULL。但两者唯一不同的是, strrchr 函数在字符串s中是从后到前(或者称为从右向左)查找字符c,找到字符c第一次出现的位置就返回,返回值指向这个位置。

代码示例

//strchr和strrchr函数使用演示
int main(void)
{
    char str[] = "I welcome any ideas from readers, of course.";
    char *lc = strchr(str, 'o');
    printf("strchr: %s\n", lc);
    char *rc = strrchr(str, 'o');
    printf("strrchr: %s\n", rc);
    return 0;
}

对于上面的示例代码, strchr 函数是按照从前到后的顺序进行查找,所以得到的结果为“ome any ideas from readers,of course.”; 而 strrchr 函数则相反,它按照从后到前的顺序进行查找,所以得到的结果为“ourse.”。

//示例代码运行结果
strchr: ome any ideas from readers, of course.
strrchr: ourse.

注意】函数的“c”参数是int类型,而不是char类型。这里用的是字符的ASCII 码(因为每个字符都对应着一个ASCII码),这样在传值的时候既可以传char类型的值,又可以传int类型的值(0~127)。

string类中的find系列函数

功能:在母串中查找子串

find(str)

返回值是子串在母串中的位置(下标记录,如果没有找到,那么会返回一个特别的标记npos(返回值可以看成是一个int型的数)。

示例:

#include<cstring>
#include<cstdio>
#include<iostream>
using namespace std;
int main()
{
    ////find函数返回类型 size_type
    string s("1a2b3c4d5e6f7jkg8h9i1a2b3c4d5e6f7g8ha9i");
    string flag;
    string::size_type position;
    //find 函数 返回jk 在s 中的下标位置
    position = s.find("jk");
    if (position != s.npos)  //如果没找到,返回一个特别的标志c++中用npos表示,我这里npos取值是4294967295,
    {
        printf("position is : %d\n" ,position);
    }
    else
    {
        printf("Not found the flag\n");
    }
}

find_first_of(str)和find_last_of(str)

返回子串出现在母串中的首次出现的位置,和最后一次出现的位置。

示例:

flag = "c";
position = s.find_first_of(flag);
printf("s.find_first_of(flag) is :%d\n",position); //5
position = s.find_last_of(flag);
printf("s.find_last_of(flag) is :%d\n",position); //25

find(str, pos)

查找某一给定位置后的子串的位置。

示例:

//从字符串s 下标5开始,查找字符串b ,返回b 在s 中的下标
position=s.find("b",5);
cout<<"s.find(b,5) is : "<<position<<endl; //23
//查找s 中flag 出现的所有位置。
flag="a";
position=0;
int i=1;
while((position=s.find(flag,position))!=string::npos) {
    cout<<"position  "<<i<<" : "<<position<<endl;
    position++;
    i++;
}

rfind()

反向查找子串在母串中出现的位置。通常我们可以这样来使用,当正向查找与反向查找得到的位置不相同说明子串不唯一

//反向查找,flag 在s 中最后出现的位置
flag="3";
position=s.rfind (flag);
printf("s.rfind (flag) :%d\n",position);

【find系列函数应用例题】

1.给出一个字符串,串中会出现有人名,找到一个只有一个人名的字符串。

#include <bits/stdc++.h>
using namespace std;
vector<string> s;
int main()
{
    s.push_back("Danil");
    s.push_back("Olya");
    s.push_back("Slava");
    s.push_back("Ann");
    s.push_back("Nikita");///建立动态数组
    string a;
    cin>>a;
    int res = 0;
    for(int i = 0; i < 5; i++)
    {
        if(a.find(s[i]) != a.npos)
        {
            res++;
            if(a.rfind(s[i]) != a.find(s[i]))///一个字符中出现多个一样的名字
            {
                res++;
            }
        }
    }
    if(res == 1)
    {
        cout<<"YES"<<endl;
    }
    else
    {
        cout<<"NO"<<endl;
    }
    return 0;
}
View Code

2.你有n个字符串。 每个字符串由小写英文字母组成。 重新排序给定的字符串,使得对于每个字符串,在它之前的所有字符串都是它的子串。

https://www.cnblogs.com/wkfvawl/p/9229758.html

#include<string>
#include<cstdio>
#include<algorithm>
#include<iostream>
using namespace std;
bool cmp(string a, string b)
{
    if (a.length() == b.length())
        return a < b;
    return a.length() < b.length();
}
int main()
{
    int n;
    string s[111];
    scanf("%d", &n);
    for (int i = 0; i < n; i++)
    {
        cin >> s[i];
    }
    sort(s, s + n, cmp);
    int flag = 1;
    for (int i = 1; i < n; i++)
    {
        if (s[i].find(s[i-1]) == string::npos)
        {
            flag = 0;
            break;
        }
    }
    if (flag)
    {
        cout << "YES" << endl;
        for (int i = 0; i < n; i++)
        {
            cout << s[i] << endl;
        }
    }
    else
    {
        cout << "NO" << endl;
    }
    return 0;
}
View Code

3.查询区间内子串在母串中的个数。

https://www.cnblogs.com/wkfvawl/p/9452869.html

#include<cstdio>
#include<cstring>
#include<string>
#include<iostream>
#include<algorithm>
using namespace std;
int main()
{
    int n,m,q,i,j,l,r,len;
    int counts;
    int vis[10010];
    string s1,s2;
    cin>>n>>m>>q;
    cin>>s1>>s2;
    len=s2.size();
    memset(vis,0,sizeof(vis));
    string::size_type pos=0;
    while((pos=s1.find(s2,pos))!=string::npos)
    {
        vis[pos+1]=pos+1;
        pos++;
    }
    for(i=1;i<=q;i++)
    {
        counts=0;
        scanf("%d%d",&l,&r);
        for(j=l;j<=r;j++)
        {
            if(vis[j]!=0&&vis[j]+len-1<=r)
            {
               counts++;
            }
        }
        printf("%d\n",counts);
    }
    return 0;
}
View Code

类型转换

string类 <=> char数组

char[] => string

char ch [] = “ABCDEFG”;
string str(ch); //也可string str = ch;
//或者
char ch [] = “ABCDEFG”;
string str;
str = ch; //在原有基础上添加可以用str += ch;

string => char[]

char buf[10];
string str(“ABCDEFG”);
length = str.copy(buf, 9);
buf[length] = ‘\0’;
//或者
char buf[10];
string str(“ABCDEFG”);
strcpy(buf, str.c_str()); //strncpy(buf, str.c_str(), 10);

string类 <=> int整型

int => string

(1)to_string函数

//c++11标准增加了全局函数std::to_string:
string to_string (int val);
string to_string (long val);
string to_string (long long val);
string to_string (unsigned val);
string to_string (unsigned long val);
string to_string (unsigned long long val);
string to_string (float val);
string to_string (double val);
string to_string (long double val);

示例:

// to_string example
#include <iostream>   // std::cout
#include <string>     // std::string, std::to_string

int main ()
{
  std::string pi = "pi is " + std::to_string(3.1415926);
  std::string perfect = std::to_string(1+2+4+7+14) + " is a perfect number";
  std::cout << pi << '\n';
  std::cout << perfect << '\n';
  return 0;
}
Output
pi is 3.141593
28 is a perfect number
to_string example
//实现to_string函数
#include<iostream>
#include<string>
using namespace std;
#define max 100
string to_String(int n)
{
    int m = n;
    char s[max];
    char ss[max];
    int i=0,j=0;
    if (n < 0)// 处理负数
    {
        m = 0 - m;
        j = 1;
        ss[0] = '-';
    }
    while (m>0)
    {
        s[i++] = m % 10 + '0';
        m /= 10;
    }
    s[i] = '\0';
    i = i - 1;
    while (i >= 0)
    {
        ss[j++] = s[i--];
    }
    ss[j] = '\0';
    return ss;
}

int main()
{
    cout << "请输入整数:";
    int m;
    cin >> m;
    string s = to_String(m) + "abc";
    cout << s << endl;
    system("pause");
    return 0;
}
to_string函数实现

(2)字符串流stringstream

标准库定义了三种类型字符串流:istringstream、ostringstream以及stringstream,看名字就知道这几种类型和iostream中的几个非常类似,分别可以读、写以及读和写string类型,它们也确实是从iostream类型派生而来的。要使用它们需要包含sstream头文件。

除了从iostream继承来的操作,

  1. sstream类型定义了一个有string形参的构造函数,即:stringstream stream(s);  ,创建了存储s副本的stringstream对象,s为string类型对象。
  2. 定义了名为str的成员,用来读取或设置stringstream对象所操纵的string值: 
    • string s = stream.str();  返回stream中存储的string类型对象s;
    • stream.str(s);  将string类型的s复制给stream,返回void;

示例:

int aa = 30;
stringstream ss;
ss<<aa; 
string s1 = ss.str();
cout<<s1<<endl; // 30

string => int

(1)采用标准库中atoi函数,对于其他类型也都有相应的标准库函数,比如浮点型atof(),long型atol()等等。

示例:

std::string str = "123";
int n = atoi(str.c_str());
cout<<n; //123
//附:atoi()函数的源码实现
//这个函数的主要功能是将一个字符串转化为一个数字,可能第一眼看上去,你会觉得这是一个很简单的函数,
//甚至是一个不需要多少行代码就可以实现的函数。其实这是一个看着很简单,但是实践起来还有些需要注意的地方。
//总的来说,有以下5种情况:
//1--指针为NULL
//2--空字符处理
//3--正号与负号的处理
//4--溢出处理
//5--如果遇到异常字符怎么处理

#include<iostream>
enum ret { kvalid=0,kinvalid };    // 是否有非法输入的标记
int status = kvalid;
long long Strtointcode(const char* digit, bool minus)
{
    long long num = 0;
    while (*digit != '\0')
    {
        if (*digit >= '0'&&*digit <= '9')
        {
            int flag = minus ? -1 : 1;
            num = num * 10 + flag*(*digit - '0');
            if ((!minus&&num > 0x7FFFFFFF) ||( minus&&num < (signed int)0x80000000))
            {
                num = 0;
                break;
            }
            digit++;
        }
        else
        {
            num = 0;
            break;
        }
    }
    if (*digit == '\0')
        status = kvalid;
    return num;
}
int Strtoint(const char* str)
{
    status = kinvalid;
    long long num = 0;
    if (str != NULL&&*str != '\0')
    {
        bool minus = false;
        if (*str == '+')
            str++;
        else if (*str == '-')
        {
            str++;
            minus = true;
        }
        if (*str != '\0')
            num = Strtointcode(str, minus);
            
    }
    return (int)num;
}

int main()
{
    char arr[20];
    int ret = 0;
    printf("请输入您要转化的字符串:\n");
    scanf("%s", arr);
    ret = Strtoint(arr);
    if (kvalid == status)
    {
        printf("%d\n", ret);
    }
    else
    {
        printf("输入非法\n");
        printf("%d\n", ret);
    }
    system("pause");
    return 0;
}

(2)采用sstream头文件中定义的字符串流对象来实现转换

istringstream is("12"); //构造输入字符串流,流的内容初始化为“12”的字符串   
int i;   
is >> i; //从is流中读入一个int整数存入i中

(整理自网络)

参考资料:

https://www.cnblogs.com/wkfvawl/p/9429128.html

https://www.cnblogs.com/smile233/p/8379802.html

posted @ 2020-11-07 20:13  箐茗  阅读(187)  评论(0编辑  收藏  举报