C语言入门之查找子串问题

 更新时间:2022年11月14日 11:46:45   作者:Revival_S  
这篇文章主要介绍了C语言入门之查找子串问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

C语言查找子串

查找子串

输入两个字符串s和t,在字符串s中查找子串t,输出起始位置,若不存在,则输出-1.要求自定义函数char*  search(char * s,char * t)返回子串t的首地址,若未找到,则返回NULL。试编写相应程序。

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

char* search(char*s,char*t)
{	
	int size_t=strlen(t),size_s=strlen(s),T,i,j;
	char *p=NULL;
	for(i=0;i<=(size_s-size_t);i++)
	{
		p=s+i;
		T=1;
		for(j=0;j<size_t;j++)
		{
			if(*p!=*(t+j))
			{
				T=0;
				break;
			}
			p++;
		}
		if(T==1)
			break;
	}
	if(T==0)
		return NULL;
	else
		return s+i;
}



int main()
{
	char *s,*t,*p,ch;
	int count,i;
	while(1) 
	{
		printf("请输入字符串s:(直接输入回车退出)\n");
		s=(char *)malloc(sizeof(char));       //这里使用动态输入字符串
		count=0;                //内存足够下可以使字符串字符个数不受数组大小限制
		while((ch=getchar())!='\n')
		{
			s[count]=ch;
			count++;
			s=(char *)realloc(s,(count+1)*sizeof(char));
		}
		s[count]='\0';
		if(count==0)                        //程序结束判断
		{
			printf("程序运行结束!\n");
			break;
		}
		while(1)                //防止子串输入字符个数超过s的循环
		{
			t=(char *)malloc((count+1)*sizeof(char));
		    printf("输入子串t:\n");
			i=0;
		    while((ch=getchar())!='\n')   //循环输入字符来计数
			{
				t[i]=ch;
		    	i++;
			}
			t[i]='\0';
		    if(i>count)
			{
				printf("子串t串长>s串长,请重新");
				t=(char *)realloc(t,i*sizeof(char));   //保证内存成功释放,重新分配t内存
				free(t);
			}
			else
				break;
		}
		if((p=search(s,t))!=NULL)
		{
			printf("起始位置:从第%d个字符开始\n",p-s+1);
	        printf("%s\n\n",p);
		}
		else
		    printf("-1\n\n");     //题目要求没有找到则输出-1
		free(s); 
		free(t);
	}
	return 0;
}

用了malloc和realloc来实现动态内存分配从而实现输入字符个数不再受数组大小限制(内存足够的话),并且考虑到子串t可能因为超出s的个数而出错做了个容错判断,总体上加了循环实现循环输入并设置了一个退出程序出口。这里所有用动态分配的指针全部都及时释放了,不会因为循环次数的增多而导致内存占用增大。

 由于realloc动态分配操作较慢,此程序还可以做优化,比如先开辟10字符的空间,每输入10个字符后再多开辟10字符空间,程序运行速度可以比此程序更快。(但对于这么一点可以忽略不计( ̄▽ ̄)~*)

 自定义函数部分使用了从被查找串s第一个字符开始,与t串的字符逐个比较的方法来查找,目前也只能想到了这种查找方法。

结果:

        

C语言查找子串(采用两个循环查找)

题目来源:浙大PTA

本题要求实现一个字符串查找的简单函数。

函数接口定义

char *search( char *s, char *t );

函数search在字符串s中查找子串t,返回子串t在s中的首地址。若未找到,则返回NULL。

裁判测试程序样例

#include <stdio.h>
#define MAXS 30

char *search(char *s, char *t);
void ReadString( char s[] ); /* 裁判提供,细节不表 */

int main()
{
    char s[MAXS], t[MAXS], *pos;

    ReadString(s);
    ReadString(t);
    pos = search(s, t);
    if ( pos != NULL )
        printf("%d\n", pos - s);
    else
        printf("-1\n");

    return 0;
}

/* 你的代码将被嵌在这里 */

输入样例1:
The C Programming Language
ram
输出样例1:
10
输入样例2:
The C Programming Language
bored
输出样例2:
-1

对于这道题的思路是,在一串

解答

🌰不用指针的方法

char *search(char *s, char *t)
{
	int i,j;                               /*定义两个下标变量*/
	int m=0,n1=0,n2=0;                     /*定义m作为开关和计数器,n1和n2为计数器*/
	while(*t)                              /*计算t中元素的个数,用于下面的循环*/
    {
        t++;
        n1++;
    }
    for(i=0;i<n1;i++) t--;                 /*让指针t归位*/
   	while(*s)                              /*计算s中元素的个数,用于下面的循环*/
    {
        s++;
        n2++;
    }
    for(i=0;i<n2;i++) s--;                 /*让指针s归位*/
    for(i=0;i<n2;i++)                      /*在n2的范围内开启循环*/
    {
		if(s[i]==t[0])                     /*当s[i]扫描到的元素等于t[0]*/
		{
			for(j=0;j<n1;j++)              /*往后循环t个元素,看是否全都一致*/
        	{
        	    if(s[i+j]==t[j])           /*扫描到后一个元素,如果相等m加上1*/
        	    	m++;
                else m=0;                  /*如果不等,m等于0*/
            }
            if(m!=n1) m=0;                 /*如果相等的数没有达到t的个数(即n1),m归零*/
            if(m==n1)                      /*如果相等的数达到了t的个数*/
	    	{   
				for(j=0;j<i;j++) s++;      /*让s指向第i个元素*/
		    	return s;                  /*最后返回s的位置*/
	    	}
		}
	}
	if(m==0) return NULL;                  /*没有发现指定字符串*/
}

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • C语言实现简单通讯录

    C语言实现简单通讯录

    这篇文章主要为大家详细介绍了C语言实现简易通讯录,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-05-05
  • C++ 基础教程之虚函数实例代码详解

    C++ 基础教程之虚函数实例代码详解

    虚函数在 c++ 的继承体系中是一个非常重要概念,让我们可以在子类中复写父类的方法。这篇文章主要介绍了C++ 基础教程之虚函数实例代码的相关资料,需要的朋友可以参考下
    2020-02-02
  • C++ 11新特性之大括号初始化详解

    C++ 11新特性之大括号初始化详解

    这篇文章主要介绍了C++ 11新特性之大括号初始化的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用C++具有一定的参考学习价值,需要的朋友们下面跟着小编来一起学习学习吧。
    2017-08-08
  • C++ Date类的具体使用(构建,重载等)

    C++ Date类的具体使用(构建,重载等)

    本文主要介绍了C++ Date类的具体使用(构建,重载等),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • LintCode-排序列表转换为二分查找树分析及实例

    LintCode-排序列表转换为二分查找树分析及实例

    这篇文章主要介绍了LintCode-排序列表转换为二分查找树分析及实例的相关资料,需要的朋友可以参考下
    2017-04-04
  • C语言详细讲解strcpy strcat strcmp函数的模拟实现

    C语言详细讲解strcpy strcat strcmp函数的模拟实现

    这篇文章主要介绍了怎样用C语言模拟实现strcpy与strcat和strcmp函数,strcpy()函数是C语言中的一个复制字符串的库函数,strcat()函数的功能是实现字符串的拼接,strcmp()函数作用是比较字符串str1和str2是否相同
    2022-05-05
  • C++11如何实现无锁队列

    C++11如何实现无锁队列

    这篇文章主要介绍了C++11如何实现无锁队列,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-08-08
  • C#桌面应用开发实现番茄定时器

    C#桌面应用开发实现番茄定时器

    本文主要介绍了C#桌面应用开发实现番茄定时器,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-07-07
  • 字符串拷贝函数memcpy和strncpy以及snprintf 的性能比较

    字符串拷贝函数memcpy和strncpy以及snprintf 的性能比较

    以下是对字符串拷贝函数memcpy和strncpy以及snprintf它们之间的性能进行了比较,需要的朋友可以过来参考下
    2013-07-07
  • C语言实现从指定位置截取文件内容

    C语言实现从指定位置截取文件内容

    这篇文章主要为大家详细介绍了如何利用C语言实现从指定位置截取文件内容,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-11-11

最新评论