//_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]=' '; //在字符串的末尾添加结束标志 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!=' '&&*q!=' ';) if(*p==*q){ p++; q++; }else{ flag=*p-*q; free(p); //释放p的空间 free(q); //释放q的空间 return flag; } if(*p==' '||*q==' '){ 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]=' '; //将Sub结束处置为' ' 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(" "); } 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("串的初始化和赋值操作: "); 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 ",StrCompare(S1,S2)); printf("将串S2连接在串S1的末尾: "); StrConcat(&S1,S2); /*将串S2连接在串S1的末尾*/ printf("S1是:"); StrPrint(S1); printf("将串S1的第12个位置后的14个字符删除: "); StrDelete(&S1,12,14); /*将串S1中的第12个位置后的14个字符删除*/ printf("S1是:"); StrPrint(S1); printf("将串S3插入到串S1中的第12个字符后: "); StrInsert(&S1,12,S3); /*将串S3插入到串S1的第12个字符后*/ printf("S1是:"); StrPrint(S1); printf("将串S1中的第12个字符后的8个字符取出并赋值给串Sub: "); SubString(&Sub,S1,12,8); /*将串S1中的第12个位置后的8个字符取出赋值给Sub*/ printf("Sub是:"); StrPrint(Sub); system("pause"); }