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 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> 删除字符串(贵大机试) -> 正文阅读

[C++知识库]删除字符串(贵大机试)

给你一个字符串S,要求你将字符串中出现的所有"gzu"(不区分大小写)子串删除,输出删除之后的S。

就是说出现“Gzu”、“GZU”、“GZu”、"gzU"都可以删除。

输入描述:

输入一行字符串S,长度不超过100。

输出描述:

输出进行删除操作之后的S。

输入样例:

GzzGzukkgzUuu

输出样例:

Gzzkkuu

**分析:**思路很简单,就是先把字符串全部转换成小写再进行模式匹配,删除对应的字串即可,但是实现方式不同。

本人本来想利用STL库中的find和erase实现快速的查找和删除,奈何遇到了大坑,代码如下:

#include<bits/stdc++.h>
using namespace std;
int main(){
	string s,flag;
	cin>>s;
	for(int i=0 ; i<s.length() ; i++){
		flag[i]=s[i];
		if(s[i]>='A'&&s[i]<='Z'){ 
			s[i] = tolower(s[i]);//先全部转换成小写字母 
		}
	}
	int site = s.find("gzu");
	while(site>-1&&site+3<=s.length()){
		s = s.erase(site,3);
		flag = flag.erase(site,3);//同时复制的字符串对应的也要删除 
		site = s.find("gzu");
	} 
	for(int k=0 ; k<s.length() ; k++){
		cout<<flag[k]; 
	}
	return 0;	
}

代码不长,但是却会遇到以下报错:

terminate called after throwing an instance of ‘std::out_of_range’
what(): basic_string::erase: __pos (which is 7) > this->size() (which is 0)

操作数位7但是s.size()=0,导致越界,这是什么原因造成的呢?前后两次赋值str的首地址极有可能是变化的。因为string里面的内存是动态分配的,想要赋值新的值,必须把上一次分配的空间收回,然后再分配更大的空间,所以首地址很有可能是变化的。可能被分配的地址还是上一个字符串的地址,所以导致越界。所以本人最终放弃了用这个方法。

最后还是使用了最原始的方法,巧妙之处是本人并没有把字串真正地从字符串中删除而是选择了牺牲空间换时间开辟一个标记数组来标记需要删除的位置,在cout时避开即可。
代码如下:

#include <bits/stdc++.h>
using namespace std;

char s[105]="gzu";
char t[105];
char t_lower[105];//转小写
int f[105];//辅助数组
int main (){
    scanf("%s",&t);
    int ls=strlen(s);
    int lt=strlen(t);
    //先转小写
    for (int i=0;i<lt;i++){
        if (t[i]>='A'&&t[i]<='Z')
            t_lower[i]=tolower(t[i]);//小写转换函数
        else t_lower[i]=t[i];
    }
    //使用标记数组将要删除的字串标记出来(f[j]=1),而不是真正的删除
    for (int i=0;i<lt;i++){
         int flag=0;
        for (int j=0;j<ls;j++){
            if(s[j]!=t_lower[i+j]) flag=1;
        }
        if (flag==0){
           for (int j=i;j<i+ls;j++) f[j]=1;//连续标记三个位置

        }
    }
    for (int i=0;i<lt;i++){
            if (f[i]==0) printf("%c", t[i]);
    }
    printf("\n");
return 0;
}
  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-02-05 21:32:58  更:2022-02-05 21:33:37 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 2:22:14-

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