当前位置: 首页 > article >正文

【算法入门】字符串基础

在这里插入图片描述

目录

  • 一.字符串引言
  • 1.字符串基础
    • 二.洛谷P5734详解
      • 1.字符串相关库函数
          • 💫(1) strcpy函数 💫
          • 💫(2) strcat函数 💫
          • 💫(3)strstr函数 💫
      • 2.题目讲解

一.字符串引言

1.字符串基础

字符串通常以\0作为结束标志,\0的ASCll码值为0,计算字符串长度时会忽略斜杠零。

二.洛谷P5734详解

1.字符串相关库函数

在讲解题目之前我们先介绍几个关于字符串操作常用的几个库函数

💫(1) strcpy函数 💫

strcpy也叫 拷贝函数 ,头文件为 string.h ,顾名思义它可以将一个字符串数组的内容拷贝到另一个数组中。
strcpy函数原型为:

char *strcat(char *dest, const char *src)

下面通过一段代码演示它的效果:

#include <stdio.h>
#include <string.h>

int main()
{
	char ch[20] = "helloworld";
	char arr[10] = "xxxxxx";
	printf("%s", strcpy(ch, arr));
	return 0;
}

因为strcpy函数的返回值是拷贝后数组的地址,所以可以通过链式访问在屏幕上打印出xxxxxx,还需要注意的是拷贝函数目的数组要留够拷贝的空间不然程序会崩溃。

💫(2) strcat函数 💫

strcat函数称为追加拷贝函数 ,头文件为头文件为 string.h
strcat函数原型为:

char *strcat(char *dest, const char *src)

strcat可以找到目标数组的结束标志\0,并用起始数组代替\0。
我们还是用代码来演示他的效果:

#include <stdio.h>
#include <string.h>

int main()
{
	char ch[15] = "likou";
	char arr[10] = "luogu";
	printf("%s", strcat(ch, arr));
	return 0;
}


结果:
在这里插入图片描述
这个函数同样需要在目标数组开辟足够的空间,才能 正常追加拷贝

💫(3)strstr函数 💫

strstr函数是字符串查找函数,文件为头文件为 string.h
函数原型为:

char *strstr( const char *arr1, const char *arr2 );

strstr函数可以在arr1数组中查找是否存在arr2数组中的内容,如果存在返回首次出现的地址,不存在则返回空指针 NULL
下面用代码来演示下它的效果:
在这里插入图片描述

以上将几个要用得字符串函数讲解了,接下来我们看题目。

2.题目讲解

  • 我们先来看一下题目:
    在这里插入图片描述
    分析: 题目考察了四个字符串的操作,只要我们掌握以上几个函数,还有熟练运用 数组下标 和字符串的 结束标志 即可。
    逻辑分析与代码编写:

我们先根据题目写出整体的逻辑:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <iostream>

using namespace std;

char ch[110];
char in[110];

int main()
{
	int count;
	cin >> count;
	cin >> ch;
	while (count--)
	{
		int input;
		scanf("%d", &input);
		if (input == 1)
		{
		}
		else if (input == 2)
		{	
		}
		else if (input == 3)
		{
		}
		else
		{
		}
	}
	return 0;
}

代码实现:

  1. 我们先实现第一个操作:在文档后插入字符串并输出,显然要用到追加拷贝函数strcat,如下:
if (input == 1)
	{
		cin >> in;
		strcat(ch, in);
		cout << ch << endl;
	}
  1. 第二步,截取文档从第 a 个字符起 b 个字符输出,只需要在数组第a+b的位置赋为\0在运用拷贝函数操作即可,代码送上 体会一下:
else if (input == 2)
		{
			int start;
			int end;
			scanf("%d %d", &start, &end);
			char in[110] = { 0 };

			ch[start + end] = '\0';
			strcat(in, &ch[start]);
			strcpy(ch, in);

			cout << ch << endl;
		}

因为在上一步操作中对in数组进行了输入,所以这里在使用in数组之前应该将in数组中的元素释放出来。

  1. 第三步,插入片段,在文档中第 a 个(有歧义,是数组下标)字符前面插入字符串,并输出文档的字符串。代码如下:
	else if (input == 3)
		{
			int cur;
			scanf("%d", &cur);
			cin >> in;
			
			strcat(in, &ch[cur]);
			ch[cur] = '\0';
			strcat(ch, in);
			printf("%s\n", ch);
	
		}

上图代码逻辑雀氏妙,大家仔细体会。

  1. 最后一步:查找子串,查找字符串 str 在文档中最先的位置并输出;如果找不到输出 -1。通过观察示例所知,若存在返回的是下标而不是内存地址(返回下标我们可以使用指针减指针的方式实现),显然这里需要用来strstr函数
else
	{
		cin >> in;
		char* ret=strstr(ch, in);
		printf("%d\n", ret != NULL ? (int)(ret-ch) : -1);
	}

这样用三目操作符 没想到吧?
在这里插入图片描述


下面完整代码奉上:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <iostream>

using namespace std;


char ch[110];
char in[110];


int main()
{
	int count;
	cin >> count;
	cin >> ch;
	while (count--)
	{
		int input;
		scanf("%d", &input);
		if (input == 1)
		{
			cin >> in;
			strcat(ch, in);
			cout << ch << endl;
		}
		else if (input == 2)
		{
			int start;
			int end;
			scanf("%d %d", &start, &end);
			char in[110] = { 0 };

			ch[start + end] = '\0';
			strcat(in, &ch[start]);
			strcpy(ch, in);

			cout << ch << endl;
		}
		else if (input == 3)
		{
			int cur;
			scanf("%d", &cur);
			cin >> in;
			
			strcat(in, &ch[cur]);
			ch[cur] = '\0';
			strcat(ch, in);
			printf("%s\n", ch);
	
		}
		else
		{
			cin >> in;
			char* ret = strstr(ch, in);
			printf("%d\n", ret != NULL ? (int)(ret - ch) : -1);

			/*if (NULL == ret)
				cout << -1 << endl;
			else
				cout << ret - ch << endl;*/
		}
	}
	return 0;
}

在源码中可以发现我注释了几处释放in数组的操作,其实我当时在疑惑多次对in数组输入,为什么不需要在下一次之前将in的元素释放掉。真相是 即使数组中保留着上一次输入的字符串 scanf再次输入时 会在字符串后自动补上\0保证上一次的元素无法影响输出(在不越界的情况下)

以上就是这次的全部内容。

感谢阅读,共同进步

http://www.kler.cn/a/69.html

相关文章:

  • 【14】模型训练自制数据集前的一些数据处理操作
  • VS2015 + OpenCV + OnnxRuntime-Cpp + YOLOv8 部署
  • linux音视频采集技术: v4l2
  • 【MySQL】深度学习数据库开发技术:使用CC++语言访问数据库
  • 机器学习基础-机器学习的常用学习方法
  • 【设计模式-2】23 种设计模式的分类和功能
  • 【面试题】Python软件工程师能力评估试题(一)
  • ChatGPT相关技术必读论文100篇(2.27日起,几乎每天更新)
  • Stable Diffusion加chilloutmixni真人图片生成模型,AI绘图杀疯了
  • java面试八股文之------Java并发夺命23问
  • 演唱会总是抢不到票?教你用Python制作一个自动抢票脚本
  • uniCloud在线升级APP配置教程
  • 三天吃透MySQL八股文(2023最新整理)
  • 【笔试题】百度+美团
  • 【Linux】冯.诺依曼体系结构与操作系统
  • 为什么北欧的顶级程序员数量远超中国?
  • html实现浪漫的爱情日记(附源码)
  • Java8使用Lambda表达式(流式)快速实现List转map 、分组、过滤等操作
  • 业内人士真心话,软件测试是没有前途的,我慌了......
  • 在Docker 上完成对Springboot+Mysql+Redis的前后端分离项目的部署(全流程,全截图)
  • 2023华为面试真题
  • 为什么拔掉计算机网线还能ping通127.0.0.1?
  • 【CTF】CTF竞赛介绍以及刷题网址
  • 【数据结构初阶】一文带你学会归并排序(递归非递归)
  • 学生信息表
  • Flask