数据结构:字符串的基本操作

字符串(string)是由0个或多个字符组成的有限序列。一般使用顺序存储结构,末尾以'\0'表示结束,但不计入字符串的长度。


示例程序:(改编自《大话数据结构》)

 C++ Code 
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
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
#include<iostream>
using  namespace std;

#define MAXSIZE  20
typedef  char String[MAXSIZE +  1];  //以'\0'结尾
/* 生成一个串*/

bool StrAssign(String Dest,  char *ptr)
{
    cout <<  "Assign Str ..." << endl;
     int i;
     for (i =  0; ptr[i] !=  '\0' && i < MAXSIZE; i++)
        Dest[i] = ptr[i];
    Dest[i] =  '\0';
     return  true;
}
/* 拷贝一个字符串 */
bool StrCopy(String Dest, String Src)
{
    cout <<  "Copy Str ..." << endl;
     int i;
     for (i =  0; Src[i] !=  '\0' && i < MAXSIZE; i++)
        Dest[i] = Src[i];
    Dest[i] =  '\0';
     return  true;
}

int StrLength(String Src)
{
     int i =  0;
     while (Src[i] !=  '\0')
        i++;
     return i;
}

bool StrEmpty(String Src)
{
     if (StrLength(Src) ==  0)
         return  true;
     else
         return  false;

}
/* 若Str1>Str2,则返回值>0;若Str1=Str2,则返回值=0;若Str1<Str2,则返回值<0 */
int StrCompare(String Str1, String Str2)
{
     int len1 = StrLength(Str1);
     int len2 = StrLength(Str2);
     for ( int i =  0; i < len1 && i < len2; i++)
         if (Str1[i] != Str2[i])
             return Str1[i] - Str2[i];
     return len1 - len2;
}

bool ClearString(String Src)
{
     for ( int i =  0; Src[i] !=  '\0'; i++)
        Src[i] =  '\0';
     return  true;
}
/* 用Dest返回Str1和Str2联接而成的新串。若未截断,则返回TRUE,否则FALSE */
bool StrConcate(String Dest, String Str1, String Str2)
{
    cout <<  "Concate String ..." << endl;
     if (StrLength(Str1) + StrLength(Str2) <= MAXSIZE)
    {
         /*  未截断 */
         int i, j;
         for (i =  0; Str1[i] !=  '\0'; i++)
            Dest[i] = Str1[i];
        j = i;
         for (i =  0; Str2[i] !=  '\0'; i++, j++)
            Dest[j] = Str2[i];
        Dest[j] =  '\0';
         return  true;
    }
     else  //截断Str2
    {
         int i, j;
         for (i =  0; Str1[i] !=  '\0'; i++)
            Dest[i] = Str1[i];
        j = i;
         for (i =  0; Str2[i] !=  '\0' && j <= MAXSIZE -  1; i++, j++)
            Dest[j] = Str2[i];
        Dest[j] =  '\0';
         return  false;
    }

}
/* 用Sub返回串Src的第pos个字符起长度为len的子串。 */
bool SubString(String Sub, String Src,  int pos,  int len)
{
     /*cout<<"Get SubString ..."<<endl;*/
     if (pos <  1 || pos > StrLength(Src) ||
            len <  0 || len > StrLength(Src) - pos +  1)
         return  false;
     int i;
     for (i =  0; i <= len -  1; i++)
        Sub[i] = Src[i + pos -  1];
    Sub[i] =  '\0';
     return  true;
}
/* 返回子串Sub在主串Src中第pos个字符之后的位置。若不存在,则函数返回值为0。 */
int Index1(String Src, String Sub,  int pos)
{
     int len1 = StrLength(Src);
     int len2 = StrLength(Sub);
     int i = pos;
    String sub;
     if (pos >  0)
    {
         while (i <= len1 - len2 +  1)
        {
            SubString(sub, Src, i, len2);
             /* 如果两串相等 , 则返回i值 */
             if (StrCompare(Sub, sub) ==  0)
                 return i;
             else /* 如果两串不相等,前进位置 */
                i++;
        }
    }
     return  0;
}

int Index2(String Src, String Sub,  int pos)
{
     int i = pos -  1;
     int j =  0;
     int len1 = StrLength(Src);
     int len2 = StrLength(Sub);
     while (i <= len1 -  1 && j <= len2 -  1)
    {
         if (Src[i] == Sub[j]) /* 两字母相等则继续 */
        {
            ++i;
            ++j;
        }
         else
        {
             /* i退回到上次匹配首位的下一位 */
            i = i - j +  1;
            j =  0; /* j退回到子串Sub的首位 */
        }
    }

     if (j == len2)  //子串已经遍历完毕
         return i - len2 +  1;
     else
         return  0;
}
/* 在串Src的第pos个字符之前插入串In。完全插入返回TRUE,部分插入返回FALSE */
bool StrInsert(String Src,  int pos, String In)
{
     int i;
     int len1 = StrLength(Src);
     int len2 = StrLength(In);
     if (pos <  1 || pos > len1 +  1)
         return  false;

     if (len1 + len2 <= MAXSIZE)
    {
         /*  完全插入 */
         for (i = len1; i >= pos -  1; i--)
            Src[i + len2] = Src[i];
         for (i = pos -  1; i < pos + len2 -  1; i++)
            Src[i] = In[i - pos +  1];
         if (pos == len1 +  1//末尾插入,最后添加'\0'
            Src[i] =  '\0';
         return  true;
    }

     else
    {
         /*  部分插入,In截断 */
         for (i = MAXSIZE; i > pos; i--)
            Src[i] = Src[pos + i - MAXSIZE];

         for (i =  0; i < MAXSIZE - pos; i++)
            Src[pos -  1 + i] = In[i];
         return  false;
    }
}
/* 从串Src中删除第pos个字符起长度为len的子串 */
bool StrDelete(String Src,  int pos,  int len)
{
     int i;
     if (pos <  1 || pos > StrLength(Src) - len +  1 || len <  0)
         return  false;
     for (i = pos + len -  1; i <= StrLength(Src); i++)
        Src[i - len] = Src[i];
     return  true;
}
/* 用Re替换主串Src中出现的所有与Sub相等的不重叠的子串 */
bool StrReplace(String Src, String Sub, String Re)
{
     int i =  1; /*  从串Src的第一个字符起查找串Sub */
     if (StrEmpty(Sub))
         return  false;
     do
    {
        i = Index1(Src, Sub, i); /*  结果i为从上一个i之后找到的子串Sub的位置 */

         if (i)
        {
            StrDelete(Src, i, StrLength(Sub)); /*  删除该串Sub */
            StrInsert(Src, i, Re);  /*  在原串Sub的位置插入串Re */
            i += StrLength(Re); /*  在插入的串Re后面继续查找串Sub */
        }

    }
     while (i);

     return  true;
}

void StrPrint(String Src)
{
    cout <<  "Print Str ..." << endl;
     for ( int i =  0; Src[i] !=  '\0'; i++)
        cout << Src[i];
    cout << endl;

}

int main( void)
{
    String Str1;
    StrAssign(Str1,  "ILOVEYOU");
    StrPrint(Str1);

    String Str2;
    StrCopy(Str2, Str1);
    StrPrint(Str2);

     if (!StrEmpty(Str1))
        cout <<  "Str1's Length : " << StrLength(Str1) << endl;

    String Str3;
    StrAssign(Str3,  "ILOVEyou");

     if (StrCompare(Str1, Str3) >  0)
        cout <<  "Str1 > Str3" << endl;
     else  if (StrCompare(Str1, Str3) ==  0)
        cout <<  "Str1 = Str3" << endl;
     else
        cout <<  "Str1 < Str3" << endl;

    String Str4, Str5;
    StrAssign(Str4,  " HLZ");
    StrConcate(Str5, Str1, Str4);
    StrPrint(Str5);

    String Str6;
    cout <<  "Get SubString ..." << endl;
    SubString(Str6, Str5,  18);
    StrPrint(Str6);

    cout <<  "Index of (Str5, Str4) " << Index2(Str5, Str4,  2) << endl;

    StrInsert(Str6,  9" HLZ");
    StrPrint(Str6);
    StrInsert(Str6,  8" HLZ");
    StrPrint(Str6);

    StrDelete(Str5,  24);
    StrPrint(Str5);

    String Str7, Str8;
    StrAssign(Str7,  "ILOVEJDWSOVEDSOVEde");
    StrAssign(Str8,  "OVE");
    StrReplace(Str7, Str8,  "ove");
    StrPrint(Str7);

     return  0;
}
输出为:


对于字符串的链式存储结构来说,一个节点存放多少个字符才合适显得很重要,这会直接影响字符串处理的效率,需要根据实际情况做出选择。但字符串的链式存储结构除了在连接串与串操作时有一定方便之外,总的来说不如顺序存储灵活,性能也不如顺序存储结构好。

  • 9
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值