XY
没有任何借口
posts - 9, comments - 31, trackbacks - 0, articles - 0
C++博客
::
首页
::
新随笔
::
联系
::
聚合
::
管理
删除字符串中的子串
Posted on 2008-06-04 14:09
路缘
阅读(6614)
评论(14)
编辑
收藏
引用
所属分类:
C/C++
题目:
请编写一个函数,删除一个字符串的一部分。函数原型如下:
int del_substr(char *str, char const *substr)
函数首先应判断substr是否出现在str中。如果它并未出现,函数就返回0;如果出现,函数应该把str中位于该子串后面的所有字符复制到该子串的位置,从而删除这个子串,然后函数返回1。如果substr多次出现在str中,函数只删除第1次出现的子串。函数的第2个参数绝不会被修改。
举个例子,假定str指向ABCDEFG。如果substr指向FGH、CDF或XABC,函数应该返回0,str未作任何修改。但如果substr指向CDE,函数就把str修改为指向ABFG,方法是把F、G和结尾的NUL字节复制到C的位置,然后函数返回1。不论出现什么情况,函数的第2个参数都不应该被修改。
要求:a.你不应该使用任何用于操纵字符串的库函数(如strcpy, strcmp, 等)。
b.函数中的任何地方都不应该使用下标引用。
一个值得注意的是,空字符串是每个字符串的一个子串,如果字符串中删除一个空字符串不会产生变化。
算法:
#include
<
stdlib.h
>
#include
<
stdio.h
>
#define
TRUE 1
#define
FALSE 0
/**/
/*
********************************************************
函数声明
*/
int
del_substr(
char
*
str,
char
const
*
substr);
/**/
/*
********************************************************
主函数
*/
int
main()
{
char
str1 []
=
"
aabcdefgh324
"
;
char
const
*
str2
=
"
abc
"
;
printf(
"
before delete: %s\n
"
, str1);
printf(
"
will delete che chars: %s\n
"
, str2);
del_substr(str1, str2);
printf(
"
After delete: %s\n
"
, str1);
system(
"
pause
"
);
return
0
;
}
/**/
/*
**删除字符串str中包含的子串substr,不进行重复删除
*/
int
del_substr(
char
*
str,
char
const
*
substr)
{
char
const
*
subP;
char
*
strP;
char
*
temp;
int
flag
=
TRUE;
strP
=
str;
if
((
!*
str)
||
*
substr
==
'
\0
'
)
/**/
/*
**如果主串为空或子串为空字符串,则返回
*/
return
FALSE;
while
(
*
strP)
{
temp
=
strP;
for
(subP
=
substr;
*
subP; subP
++
)
{
if
(
*
strP
==
*
subP)
/**/
/*
**如果当前的主串和子串字符相等
*/
{
strP++;/**//***则主串指针向前移一个字符*/
flag = FALSE;
/**//*设置为false,防止后续的主串指针,重复移动*/
continue;
}
break
;
}
if
(
*
subP
==
'
\0
'
)
/**/
/*
**如果subP指向末尾的'\0'则表示子串匹配成功
*/
{
while
(
*
temp
++
=
*
strP
++
)
/**/
/*
**进行字符复制
*/
;
/**/
/*
**空语句
*/
break
;
}
if
(flag)
strP
++
;
}
return
TRUE;
}
说明:
最先str1的定义为 char *,然而就会出现while(*temp++ = *strP++)语句报错。报错误写入位置 0x00447240 时发生访问冲突,我的理解是因为字符指针temp指向的是字符串常量,不允许进行修改。所以定义为
char str1 []
很关键。
经上网查资料,解释是:
C++中数据存储区分为五种:
栈、堆(new分配)、自由存储区(malloc分配)、全局/静态变量存储区、常量存储区(不允许修改,除非采用特殊手段)。 字符串这种常量就存在常量区中。全局、静态变量放在静态存储区,而它们是可以被修改的。
Feedback
#
re: 删除字符串中的子串[未登录]
回复
更多评论
2008-06-04 16:11 by
raof01
感觉博主对于传参方式(传值)没有理解。
函数原型有错误:无法返回该字符串。感觉下面会好一点:
char* del_substr(char* str, const char * substr);
说不用字符串操作函数,我就hack一把——自己写操纵字符串的库函数,呵呵……:
ssize_t Strlen(const char* str)
{
ssize_t len = 0;
while (*str++) ++len;
return len;
}
int StrCmp(const char* str1, const char* str2, ssize_t len)
{
for (int i = 0; i < len; ++i)
{
if (*(str1 + i) != *(str2 + i))
return (*(str1 + i) - *(str2 + i));
}
return 0;
}
char* del_substr(char* str, const char * substr)
{
char* temp = NULL;
ssize_t len = Strlen(substr);
while (*substr++ && *str++)
{
if (*substr == *str)
{
if (!StrCmp(substr, str, len))
{
temp = str + len;
while (*temp++)
{
*str++ = *temp;
}
break;
}
}
}
return str;
}
#
re: 删除字符串中的子串[未登录]
回复
更多评论
2008-06-04 16:15 by
raof01
这几行代码里面有写错误,不知道你看出来了没有?呵呵。希望你能改掉这些错误。
#
re: 删除字符串中的子串
回复
更多评论
2008-06-05 01:41 by
passerby
为何不用KMP来比较字符呢
#
re: 删除字符串中的子串
回复
更多评论
2008-06-05 10:13 by
路缘
@
raof01
确实有问题,我现把更改的代码贴在下面,但不知为为何说我的方法无法返回字符串,对传参方式没有理解。del_substr(
char
*
str,
char
const
*
substr)传递的是主串指针的拷贝,但它和原字符串指针指向的是相同的位置,通过改变所指位置的内容来达到改变原字符串的目的。
关于你提供的代码,我修改如下。还是谢谢你的热心回复。让我同时也学了一些东西
char
*
del_substr(
char
*
str,
const
char
*
substr)
{
char
*
temp
=
NULL,
*
cp
=
str;
ssize_t len
=
Strlen(substr);
while
(
*
str)
{
if
(
*
substr
==
*
str)
{
if
(
!
StrCmp(substr, str, len))
{
temp
=
str
+
len;
while
(
*
str
++
=
*
temp
++
)
;
break
;
}
}
str
++
;
}
return
cp;
}
#
re: 删除字符串中的子串
回复
更多评论
2008-06-05 13:11 by
raof01
@路缘
看来没有唬住你。你的理解是没错的,呵呵。
除了del_substr(),别的还有问题吗?
#
re: 删除字符串中的子串
回复
更多评论
2008-06-05 15:52 by
路缘
看来没有唬住你。你的理解是没错的,呵呵。
除了del_substr(),别的还有问题吗?
--------------------------------------------------------
@raof01,问题我是找不出来了,不过我对比了哈C的库函数,你的代码,
while (*str++) ++len;
不如库函数中的
const char *eos = str;
while( *eos++ )
;
return( eos - str - 1 );
写法效率高。
关于字符串的比较,跟库函数的写法的出入,我还得研究哈一些细节东西,看能不能琢磨出差别的用意。
如果代码中还有其他问题,还望@raof01不吝赐教,谢谢了。
#
re: 删除字符串中的子串
回复
更多评论
2008-06-05 15:54 by
路缘
@passerby
谢谢你,KMP算法大学时学过,谢谢你的提醒,你让我知道事情不是做完那么简单,还得精益求精,我会抽时间,再写一个改进的算法来实现。
#
re: 删除字符串中的子串
回复
更多评论
2008-06-05 17:51 by
raof01
@路缘
也没啥问题了——我没有仔细考虑,时间紧,我只拿了5分钟来写这个。
#
re: 删除字符串中的子串
回复
更多评论
2008-07-09 10:20 by
chu
效率似乎太低了点
#
re: 删除字符串中的子串
回复
更多评论
2008-12-03 16:33 by
佰锐科技
StrCmp 应该改为StrNCmp
#
re: 删除字符串中的子串
回复
更多评论
2010-01-08 14:30 by
ff
if(flag)
strP++;
后少了点吧?没有找到的时候死循环了吧?
加else break;可以解决~
#
re: 删除字符串中的子串
回复
更多评论
2010-01-08 15:03 by
ff
char* del_substr(char* str, const char * substr)
{
char* temp = NULL, *cp = str;
ssize_t len = Strlen(substr);
while (*str)
{
if (*substr == *str)
{
if (!StrCmp(substr, str, len))
{
temp = str + len;
while (*str++ = *temp++)
;
break;
}
}
str++;
}
return cp;
} 能删除全部出现的子字符串,但是“如果substr多次出现在str中,函数只删除第1次出现的子串”,这段代码会把重复出现的也删除。。。
#
re: 删除字符串中的子串
回复
更多评论
2010-01-08 15:07 by
ff
看错了~~sorry
#
re: 删除字符串中的子串
回复
更多评论
2010-03-13 00:47 by
sb
ssize_t是什么类型?
刷新评论列表
只有注册用户
登录
后才能发表评论。
【推荐】100%开源!大型工业跨平台软件C++源码提供,建模,组态!
相关文章:
算法:快速排序之python实现
算法:找出n个数中重复最多的10个数
反转字符串
删除字符串中的子串
得到字符串S1中第一个且是字符串S2中的位置指针
网站导航:
博客园
IT新闻
BlogJava
博问
Chat2DB
管理
Powered by:
C++博客
Copyright © 路缘
日历
<
2008年6月
>
日
一
二
三
四
五
六
25
26
27
28
29
30
31
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
1
2
3
4
5
常用链接
我的随笔
我的评论
我参与的随笔
留言簿
(5)
给我留言
查看公开留言
查看私人留言
随笔分类
(9)
C/C++(5)
计算机图形学(1)
数字图像
网络通信
系统体系(2)
移动开发(1)
随笔档案
(9)
2017年9月 (2)
2008年10月 (2)
2008年9月 (1)
2008年7月 (1)
2008年6月 (3)
博客
本人博客园的博客
搜索
积分与排名
积分 - 35958
排名 - 573
最新评论
1. re: 反转字符串
@周翀
刚一看,吓我一跳,以为一直认为的解法有问题。细想一下,值相等也无妨。当毁掉其中一个值后,再做异或,另一个值就会得到保留,做第3次计算,毁掉的那个值也被恢复了,所以没有问题。
--路缘
2. re: 反转字符串
三次异或可以互换两个值,也可以毁掉两个值,当它们相等的时候……
--周翀
3. re: 深入理解计算机系统1_程序是如何运行的
抄袭
--图—图—
4. re: 删除字符串中的子串
ssize_t是什么类型?
--sb
5. re: 删除字符串中的子串
看错了~~sorry
--ff
阅读排行榜
1. 反转字符串(9782)
2. 深入理解计算机系统1_程序是如何运行的(7940)
3. 删除字符串中的子串(6614)
4. 分形的乐趣之_Hilbert曲线(2093)
5. 深入理解计算机系统2_信息存储(2062)
评论排行榜
1. 删除字符串中的子串(14)
2. 深入理解计算机系统1_程序是如何运行的(12)
3. 反转字符串(2)
4. 分形的乐趣之_Hilbert曲线(2)
5. symbian应用程序开发1(1)