//_DataStructure_C_Impl:链串
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#define ChunkSize 4
#define stuff '#'
//串的结点类型定义
typedef struct Chunk{
char ch[ChunkSize];
struct Chunk *next;
}Chunk;
//链串的类型定义
typedef struct{
Chunk *head;
Chunk *tail;
int length;
}LinkString;
//初始化字符串S
void InitString(LinkString *S){
S->length=0; //将串的长度置为0
S->head=S->tail=NULL; //将串的头指针和尾指针置为空
}
//生成一个其值等于cstr的串S。 成功返回1,否则返回0
int StrAssign(LinkString *S,char *cstr){
int i,j,k,len;
Chunk *p,*q;
len=strlen(cstr); //len为链串的长度
if(!len)
return 0;
S->length=len;
j=len/ChunkSize; //j为链串的结点数
if(len%ChunkSize)
j++;
for(i=0;i<j;i++){
p=(Chunk *)malloc(sizeof(Chunk)); //动态生成一个结点
if(!p)
return 0;
for(k=0;k<ChunkSize&&*cstr;k++)
*(p->ch+k)=*cstr++; //将字符串ctrs中的字符赋值给链串的数据域
if(i==0) //假设是第一个结点
S->head=q=p; //头指针指向第一个结点
else{
q->next=p;
q=p;
}
if(!*cstr){ //假设是最后一个链结点
S->tail=q; //将尾指针指向最后一个结点
q->next=NULL; //将尾指针的指针域置为空
for(;k<ChunkSize;k++)
*(q->ch+k)=stuff; //将最后一个结点用'#'填充
}
}
return 1;
}
//推断串是否为空。假设S为空串,则返回1,否则返回0
int StrEmpty(LinkString S){
if(S.length==0)
return 1;
else
return 0;
}
//求串的长度
int StrLength(LinkString S){
return S.length;
}
//串的转换操作。将串S的内容转换为字符串,将串S中的字符复制到cstr。成功返回1,否则返回0
int ToChars(LinkString S,char **cstr){
Chunk *p=S.head; //将p指向串S中的第1个结点
int i;
char *q;
*cstr=(char *)malloc((S.length+1)*sizeof(char));
if(!cstr||!S.length)
return 0;
q=*cstr; //将q指向cstr
while(p){ //块链没结束
for(i=0;i<ChunkSize;i++)
if(p->ch[i]!=stuff) //假设当前字符不是填充的特殊字符'#'。则将S中字符赋值给q
*q++=(p->ch[i]);
p=p->next;
}
(*cstr)[S.length]='\0'; //在字符串的末尾加入结束标志
return 1;
}
//串的复制操作
int StrCopy(LinkString *T,LinkString S){
char *str;
int flag;
if(!ToChars(S,&str)) //将串S中的字符复制到字符串str中
return 0;
flag=StrAssign(T,str); //将字符串str的字符赋值到串T中
free(str); //释放str的空间
return flag;
}
//串的比較操作。若S的值大于T。则返回正值。若S的值等于T,则返回0。若S的值小于T,则返回负值
int StrCompare(LinkString S,LinkString T){
char *p,*q;
int flag;
if(!ToChars(S,&p)) //将串S转换为字符串p
return 0;
if(!ToChars(T,&q)) //将串T转换为字符串q
return 0;
for(;*p!='\0'&&*q!='\0';)
if(*p==*q){
p++;
q++;
}else{
flag=*p-*q;
free(p); //释放p的空间
free(q); //释放q的空间
return flag;
}
if(*p=='\0'||*q=='\0'){
free(p);
free(q);
return S.length-T.length;
}
}
//串的链接操作。将串S连接在串T的尾部
int StrConcat(LinkString *T,LinkString S){
int flag1,flag2;
LinkString S1,S2;
InitString(&S1);
InitString(&S2);
flag1=StrCopy(&S1,*T); //将串T的内容复制到S1中
flag2=StrCopy(&S2,S); //将串S的内容复制到S2中
if(flag1==0||flag2==0) //假设有一个串拷贝不成功,则返回0
return 0;
T->head=S1.head; //改动串T的头指针
S1.tail->next=S2.head; //将串S1和S2首尾相连
T->tail=S2.tail; //改动串T的尾指针
T->length=S.length+T->length; //改动串T的长度
return 1;
}
//串的插入操作。 在串S的第pos个位置插入串T
int StrInsert(LinkString *S,int pos,LinkString T){
char *t1,*s1;
int i,j;
int flag;
if(pos<1||pos>S->length+1)
return 0;
if(!ToChars(*S,&s1)) //将串S转换为字符串s1
return 0;
if(!ToChars(T,&t1)) //将串T转换为字符串t1
return 0;
j=strlen(s1); //j为字符串s1的长度
s1=(char *)realloc(s1,(j+strlen(t1)+1)*sizeof(char)); //为s1又一次分配空间
for(i=j;i>=pos-1;i--) //将字符串s1中的第pos以后的字符向后移动strlen(t1)个位置
s1[i+strlen(t1)]=s1[i];
for(i=0;i<(int)strlen(t1);i++) //在字符串s1中插入t1
s1[pos+i-1]=t1[i];
InitString(S);
flag=StrAssign(S,s1); //由s1生成串S
free(t1);
free(s1);
return flag;
}
//串的删除操作。 将串S中的第pos个字符起长度为len的子串删除
int StrDelete(LinkString *S,int pos,int len){
char *str;
int i;
int flag;
if(pos<1||pos>S->length-len+1||len<0)
return 0;
if(!ToChars(*S,&str)) //将串S转换为字符串str
return 0;
for(i=pos+len-1;i<=(int)strlen(str);i++) //将字符串中第pos个字符起的长度为len的子串删除
str[i-len]=str[i];
InitString(S); //释放S的原有存储空间
flag=StrAssign(S,str); //将字符串str转换为串S
free(str);
return flag;
}
//取子串操作。 用Sub返回串S的第pos个字符起长度为len的子串
int SubString(LinkString *Sub,LinkString S,int pos,int len){
char *t,*str;
int flag;
if(pos<1||pos>S.length||len<0||len>S.length-pos+1)
return 0;
if(!ToChars(S,&str)) //将串S转换为字符串str
return 0;
t=str+pos-1; //t指向字符串str中的pos个字符
t[len]='\0'; //将Sub结束处置为'\0'
flag=StrAssign(Sub,t); //将字符串t转换为Sub
free(str);
return flag;
}
//清空串操作。将串的空间释放
void ClearString(LinkString *S){
Chunk *p,*q;
p=S->head;
while(p){
q=p->next;
free(p);
p=q;
}
S->head=S->tail=NULL;
S->length=0;
}
//链串的输出
void StrPrint(LinkString S){
int i=0,j;
Chunk *h;
h=S.head;
while(i<S.length){
for(j=0;j<ChunkSize;j++)
if(*(h->ch+j)!=stuff){
printf("%c",*(h->ch+j));
i++;
}
h=h->next;
}
printf("\n");
}
void main(){
int i,j;
int flag;
LinkString S1,S2,S3,Sub;
char *str1="Welcome to";
char *str2=" Data Structure";
char *str3="Computer Architecture";
printf("串的初始化和赋值操作:\n");
InitString(&S1); /*串S1,S2。S3的初始化*/
InitString(&S2);
InitString(&S3);
InitString(&Sub);
StrAssign(&S1,str1); /*串S1。S2,S3的赋值操作*/
StrAssign(&S2,str2);
StrAssign(&S3,str3);
printf("串S1的值是:");
StrPrint(S1);
printf("串S2的值是:");
StrPrint(S2);
printf("串S3的值是:");
StrPrint(S3);
printf("%d\n",StrCompare(S1,S2));
printf("将串S2连接在串S1的末尾:\n");
StrConcat(&S1,S2); /*将串S2连接在串S1的末尾*/
printf("S1是:");
StrPrint(S1);
printf("将串S1的第12个位置后的14个字符删除:\n");
StrDelete(&S1,12,14); /*将串S1中的第12个位置后的14个字符删除*/
printf("S1是:");
StrPrint(S1);
printf("将串S3插入到串S1中的第12个字符后:\n");
StrInsert(&S1,12,S3); /*将串S3插入到串S1的第12个字符后*/
printf("S1是:");
StrPrint(S1);
printf("将串S1中的第12个字符后的8个字符取出并赋值给串Sub:\n");
SubString(&Sub,S1,12,8); /*将串S1中的第12个位置后的8个字符取出赋值给Sub*/
printf("Sub是:");
StrPrint(Sub);
system("pause");
}

_DataStructure_C_Impl:链串的更多相关文章

  1. YTU 3008: 链串的基本运算

    3008: 链串的基本运算 时间限制: 1 Sec  内存限制: 128 MB 提交: 1  解决: 1 题目描述 编写一个程序,实现链串的各种基本运算,主函数已给出,请补充每一种方法. 1.建立串s ...

  2. 数据结构(c语言第2版)-----了解链表,栈,队列,串

    关于链表我觉得这都是最基本的东西,但是不常见,在实际的应用中很少的使用,了解它会用就OK,不需要研究的那么深,除非做那种内存压缩,存储方面工作. C语言中动态申请空间 malloc() q=(dlin ...

  3. JavaScript责任链模式

    介绍 责任链模式(Chain of responsibility)是使多个对象都有机会处理请求,从而避免请求的发送者和接受者之间的耦合关系.将对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理 ...

  4. 【数据结构(C语言版)系列四】 串

    串类型的定义 串(或字符串)是由零个或多个字符组成的有限序列,一般记为 s = 'a1a2...an',s为串名.子串在主串中的位置以子串的第一个字符在主串中的位置来表示. 串和表示和实现——定长顺序 ...

  5. RxJava如何结合观察者与链式处理

    RxJava如何结合观察者与链式处理 Author: Dorae Date: 2018年12月3日17:10:31 转载请注明出处 一.概述 首先问自己几个问题,如果非常清楚这几个问题的目的与答案,那 ...

  6. nginx应用总结(1)--基础认识和应用配置

    在linux系统下使用nginx作为web应用服务,用来提升网站访问速度的经验已五年多了,今天在此对nginx的使用做一简单总结. 一.nginx服务简介Nginx是一个高性能的HTTP和反向代理服务 ...

  7. nginx常用命令

    ps -ef | grep nginx在进程列表里面找master进程,它的编号就是主进程号了. 步骤2:发送信号 从容停止Nginx: kill -QUIT 主进程号 快速停止Nginx: kill ...

  8. HDU 2609 最小表示法

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2609 题意:给定n个循环链[串],问有多少个本质不同的链[串](如果一个循环链可以通过找一个起点使得和 ...

  9. nginx全局变量实例对照 rewrite参考手册

    http://dwz.stamhe.com/index.php?_a=index&_m=show&count=10 remote_addr 客户端ip,如:192.168.4.2 bi ...

随机推荐

  1. java 实现yaml 数据转json与map

    首先引入snakeyaml-1.16.jar的包. 直接上代码: package com.ming.yaml; import java.util.Map; import org.yaml.snakey ...

  2. list用法(用到了再补充)

    之前学list吧,也知道很多,但是到用的时候却无从下手,还是不熟悉的缘故,看来基础知识应该再加强,要达到信手拈来的程度才行. 先说下list的特性:有序可重复,也可以存储多个空值. 我用到的方法: L ...

  3. Assembly之Instruction之Byte and Word

    Byte and word issues The MSP430 is byte-addressed, and little-endian. Word operands must be located ...

  4. OpenCV:OpenCV目标检测Adaboost+haar源代码分析

    使用OpenCV作图像检测, Adaboost+haar决策过程,其中一部分源代码如下: 函数调用堆栈的底层为: 1.使用有序决策桩进行预测 template<class FEval> i ...

  5. AI:狄拉克之海上的涟漪

    延陵季子2011年  8月27日 19:02   借鉴英文原文:Ripples in the Dirac Sea 当他试着用一种轻松的口吻诉说一些事情时,我会明白,其实我们都明白,在他的心里绝对不是平 ...

  6. 身份认证防止重放攻击的challenge-response方法

    或者叫询问-应答机制. 基于挑战/应答(Challenge/Response)方式的身份认证系统就是每次认证时认证服务器端都给客户端发送一个不同的"挑战"字串,客户端程序收到这个& ...

  7. sessionStorage和localStorage存储的转换不了json

    先说说localStorage与sessionStorage的差别 sessionStorage是存储浏览器的暂时性的数据,当关闭浏览器下次再打开的时候就不能拿到之前存储的缓存了 localStora ...

  8. Robot Framework(六)变量

    变量 2.5.1简介 变量是Robot Framework的一个不可或缺的特性,它们可以在测试数据的大多数地方使用.最常见的是,它们用于测试用例表和关键字表中关键字的参数,但所有设置都允许在其值中使用 ...

  9. Log4net日志发布到服务器上日志无法写入

    log4net在本地执行时候,日志正常写入,但是发布到服务器上的时候,日志就无法正常写入 解决方案: 1.文件权限 在发布到服务器上的时候,可能文件没有写入权限,导致日志无法正常写入 打开IIS 找到 ...

  10. 单调队列 && 单调栈

    单调队列 && 单调栈 单调队列 维护某个滑动区间的min or max,可用于dp的优化 以维护min为例,采用STL双端队列实现 每次加入元素x前 先检查队首元素==滑动后要删除的 ...