前言


PS:本文相关头文件、预编译以及typedef如下,阅读一遍以便于下面的理解:  

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAXSTRLEN 40
#define OK 1
#define ERROR 0
#define TURE 1
#define FALSE 0
#define OVERFLOW -2 typedef int Status;
typedef char SString[MAXSTRLEN+1];

  

ADT


  

  串是一个重要的数据结构,我在这里实现串的ADT如下:

  

  

串的结构概念图


  

  我定义了一个数组,0位存储字符长度,1到MAX位存储字符。这和传统的gets();函数或者 scanf(%s,);函数实现的串是有区别的,所以我封装了一个串赋值函数去实现两种串结构之间的转化,代码如下: 

/* 字符串赋值,当赋值字符串长度超过被赋值字符串时候截断,未超过时先将长度存入T[0],再逐位赋值 */
Status StrAssign(SString T,char *chars)
{
int i;
if(strlen(chars)>MAXSTRLEN)
{
for(i = 1;i <= MAXSTRLEN;i++)
{
T[i] = *(chars + i - 1);
}
T[0] = MAXSTRLEN; //T[0]存入int 型数据,%s无法打印
}
else
{
T[0] = strlen(chars); //T[0]存入的是chars的字符长度
for(i = 1;i <= strlen(chars);i++)
{
T[i] = *(chars + i - 1);
}
}
return OK;
}

  

功能函数算法难点


 

  我认为其中的StrIndex();索引子串函数是相对难理解一些的。其函数功能是:索引子串,返回串T中第一个索引到与串S相同的子串的位置。算法思路图如下:

    

  尤其要注意,当T[i] != S[j]时,需要将i指针置于上次相同时的下一个位置,代码实现为 i = i - j + 2,而不是i++

  该模块函数封装代码如下:  

/*  索引子串,返回串T中第一个索引到与串S相同的子串的位置  */
Status StrIndex(SString T,SString S,int *pos)
{
int i=1,j=1;
while((i <= T[0]) && (j <= S[0]))
{
if(T[i] == S[j])
{
i++;
j++;
}
else
{
i = i - j + 2;
j = 1;
}
}
if(j > S[0])
{
*pos = i - j + 1;
printf("Found,position is %d\n",*pos);
return OK;
}
else
{
printf("NOT find child string yet.\n");
return ERROR;
}
}

  其次,StrInsert();插入函数中也比较难想,因为其需要讨论插入位置以及插入后长度是否越界问题,设计算法时易漏了其中某一个情况。

  该模块函数封装代码如下:

/*  在串T的第pos个位置后插入子串S  */
Status StrInsert(SString T,SString S,int pos)
{
int i;
if(pos > T[0])
{
pos = T[0] +1;
}
if(T[0] + S[0] <= MAXSTRLEN)
{
for(i = 1; i <= pos;i++)
T[T[0] + S[0] - i +1] = T[T[0] + S[0] - i -pos];
for(i = 1;i <= S[0];i++)
T[pos + i] = S[i];
T[0] = T[0] + S[0];
return OK;
}
if(T[0] + S[0] > MAXSTRLEN)
{
for(i = 1;i <= (MAXSTRLEN - pos);i++)
T[i + pos] = S[i]; //超出部分被删除
T[0] = MAXSTRLEN;
return ERROR;
}
}

  

代码


/*************************************************************************
> File Name: String fuctions
> Author: Bw98
> Mail: 786016746@qq.com
> Blog: www.cnblogs.com/Bw98blogs/
> Created Time: SUN 23th Jul. 2017
************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAXSTRLEN 40
#define OK 1
#define ERROR 0
#define TURE 1
#define FALSE 0
#define OVERFLOW -2 typedef int Status;
typedef char SString[MAXSTRLEN+1]; Status StrEmpty(SString S);
Status StrAssign(SString T,char *chars);
Status StrCopy(SString T,SString S);
Status StrLength(SString T);
Status StrPrint(SString T);
Status StrClear(SString S);
Status StrConcat(SString T,SString S1,SString S2);
Status StrIndex(SString T,SString S,int *pos);
Status StrInsert(SString T,SString S,int pos);
Status StrDelete(SString T,int pos,int len); int main()
{
/* 根据功能自行调用 */
return 0;
} /* 字符串赋值,当赋值字符串长度超过被赋值字符串时候截断,未超过时先将长度存入T[0],再逐位赋值 */
Status StrAssign(SString T,char *chars)
{
int i;
if(strlen(chars)>MAXSTRLEN)
{
for(i = 1;i <= MAXSTRLEN;i++)
{
T[i] = *(chars + i - 1);
}
T[0] = MAXSTRLEN; //T[0]存入int 型数据,%s无法打印
}
else
{
T[0] = strlen(chars); //T[0]存入的是chars的字符长度
for(i = 1;i <= strlen(chars);i++)
{
T[i] = *(chars + i - 1);
}
}
return OK;
} /* 字符串拷贝,逐个字符拷贝(仅拷贝长度范围内的) */
Status StrCopy(SString T,SString S)
{
int i;
for(i = 1;i <= S[0];i++)
{
T[i] = S[i];
}
T[0] = S[0];
return OK;
} /* 字符串判空,S[0] == 0时为空 */
Status StrEmpty(SString T)
{
if(T[0] == 0)
{
printf("The string is empty!\n");
return TURE;
}
else
{
printf("The string is NOT empty!\n");
return FALSE;
}
} /* 返回该字符串的长度 */
Status StrLength(SString T)
{
return T[0];
} /* 打印字符串 */
Status StrPrint(SString T)
{
int i;
for(i = 1;i <= T[0];i++)
printf("%c",T[i]);
printf("\n");
return OK;
}
/* 清除字符串,即长度清空 */
Status StrClear(SString S)
{
S[0] = 0;
return OK;
} /* 字符串联接,通过T来存储结果,S2连接而成的新串,若未截断则返回TRUE,截断返回FAlSE,注意字符串数组越界问题 */
Status StrConcat(SString T,SString S1,SString S2)
{
int i;
if(S1[0]+S2[0] <= MAXSTRLEN)
{
for(i = 1;i <= S1[0];i++)
{
T[i] = S1[i];
}
for(i = 1;i <= S2[0];i++)
{
T[S1[0]+i] = S2[i];
}
T[0] = S1[0] + S2[0];
return TURE;
}
else
{
for(i = 1;i <= S1[0];i++)
{
T[i] = S1[i];
}
for(i = 1;i <= (MAXSTRLEN-S1[0]);i++)
{
T[S1[0]+i] = S2[i];
}
T[0] = MAXSTRLEN;
return FALSE;
}
} /* 索引子串,返回串T中第一个索引到与串S相同的子串的位置 */
Status StrIndex(SString T,SString S,int *pos)
{
int i=1,j=1;
while((i <= T[0]) && (j <= S[0]))
{
if(T[i] == S[j])
{
i++;
j++;
}
else
{
i = i - j + 2;
j = 1;
}
}
if(j > S[0])
{
*pos = i - j + 1;
printf("Found,position is %d\n",*pos);
return OK;
}
else
{
printf("NOT find child string yet.\n");
return ERROR;
}
} /* 在串T的第pos个位置后插入子串S */
Status StrInsert(SString T,SString S,int pos)
{
int i;
if(pos > T[0])
{
pos = T[0] +1;
}
if(T[0] + S[0] <= MAXSTRLEN)
{
for(i = 1; i <= pos;i++)
T[T[0] + S[0] - i +1] = T[T[0] + S[0] - i -pos];
for(i = 1;i <= S[0];i++)
T[pos + i] = S[i];
T[0] = T[0] + S[0];
return OK;
}
if(T[0] + S[0] > MAXSTRLEN)
{
for(i = 1;i <= (MAXSTRLEN - pos);i++)
T[i + pos] = S[i]; //超出部分被删除
T[0] = MAXSTRLEN;
return ERROR;
}
} /* 删除串T的第pos个位置起,长为len的子串 */
Status StrDelete(SString T,int pos,int len)
{
int i;
if(pos > T[0])
return ERROR;
if(len > T[0])
len = T[0] - pos + 1;
for(i = len + pos;i <= T[0];i++)
T[i - len] =T[i];
T[0] = T[0] - len;
return OK;
}

  

串String(1):串的实现(定长顺序存储结构)的更多相关文章

  1. javascript实现数据结构:串--定长顺序存储表示以及kmp算法实现

    串(string)(或字符串)是由零个或多个字符组成的有限序列.串中字符的数目称为串的长度.零个字符的串称为空串(null string),它的长度为零. 串中任意个连续的字符组成的子序列称为该串的子 ...

  2. 生成定长随机数-可做3des密钥

    3DES加解密需要密钥支持,要求为8的倍数,一般会使用32位的字母数字随机字符串作为密钥. 下面这个工具类,可用做key值的生成,详见下方代码: package test; import java.u ...

  3. Delphi String 常用字串符处理函数

    Delphi 在面对跨平台开发,程序语言也改进不少,不过有些改进,让原本 Delphi 开发者有些不适应,最显注的就是字串处理函数了,原本 Pascal 语言字串起始由 1 开始,几乎是它的经典了,新 ...

  4. 串string (KMP)

    1.Definition 串string,是零个或多个字符组成的有限序列.一般记作S="a1a2a3...an",其中S是串名,双引号括起来的字符序列是串值:ai(1<= i ...

  5. Day07 - Ruby比一比:Symbol符号与String字串

    前情提要: 第六天我们透过Ruby代码练习public,protected和privatemethod时,发现冒号在前面的参数,:mydraft,:myspace,这些就是符号Symbol.在今天,我 ...

  6. LeetCode Reverse Words in a String 将串中的字翻转

    class Solution { public: void reverseWords(string &s) { string end="",tem="" ...

  7. hdu5384 AC自己主动机模板题,统计模式串在给定串中出现的个数

    http://acm.hdu.edu.cn/showproblem.php?pid=5384 Problem Description Danganronpa is a video game franc ...

  8. HDU - 2222,HDU - 2896,HDU - 3065,ZOJ - 3430 AC自动机求文本串和模式串信息(模板题)

    最近正在学AC自动机,按照惯例需要刷一套kuangbin的AC自动机专题巩固 在网上看过很多模板,感觉kuangbin大神的模板最为简洁,于是就选择了用kuangbin大神的模板. AC自动机其实就是 ...

  9. Java随机生成定长纯数字或数字字母混合数

    (转)Java随机生成定长纯数字或数字字母混合数 运行效果图: 具体实现代码

随机推荐

  1. JSON.parse()和JSON.stringify()和eval('(' + result + ')')

    var str = '{"name":"huangxiaojian","age":"23"}' 结果: JSON.par ...

  2. C++语言中的类型(二)

    --分门别类是简化事物最有效的方式. C++语言的强大能力的体现在对程序员自定义数据类型的支持.C++语言主要的一个设计目标就是让程序员自定义的数据类型像内置类型一样好用. 一.自定义数据类型 数据类 ...

  3. debounce去弹跳

    通过返回闭包,来共用timer定时器,通过定时器的清除和设置来实现每次触发后重新计时. /** * * @param fn {Function} 实际要执行的函数 * @param delay {Nu ...

  4. iOS 超大高清图展示策略 TileLayer 及 levelsOfDetailBias 分析

    本次分析针对当下流行的中国地图图片处理,1亿像素,就是下面这张: 原图尺寸:11935x8554 文件大小:22.1MB 原始加载方式 首先,我们尝试一下直接加载的方式,看看效果会有多恐怖 效果请看下 ...

  5. mysql中 union是什么鬼

    在sql注意时,经常会用到一个不怎么常用的联合查询 http://www.php20.com/forum.php?m ... &extra=page%3D1 清空表.从新执行一下以上链接中的s ...

  6. 4、C#基础 - C# 的 常见概念简述

    在上篇文章中,你跟着我写了一个HelloWorld,本篇中,我们来谈谈一些C#程序中的小概念 1.C# 程序结构 一个 C# 程序主要包括以下部分: 命名空间声明(Namespace declarat ...

  7. linux apt-cache使用方法

    apt-cache是linux下的一个apt软件包管理工具,它可查询apt的二进制软件包缓存文件.APT包管理的大多数信息查询功能都可以由apt-cache命令实现,通过apt-cache命令配合不同 ...

  8. webrtc视频数据render流程

  9. 》》stroll--各种特效下拉菜单

    <!doctype html> <html lang="en"> <head> <meta charset="utf-8&quo ...

  10. 打印杨辉三角 --JS

    var arr = new Array(); for(var i = 0 ;i < 6 ; i++){ if(i == 0){arr.push(1);} else if(i == 1){arr ...