广义表

列表里面有列表,比如(1,(2,(3,4)),5)

用链表可以实现

结果如图

guangyibiao.h

#ifndef __GUANGYIBIAO__
#define __GUANGYIBIAO__ #include <stdio.h>
#include <string.h>
#include <memory.h>
#include <malloc.h>
#include <assert.h>
#include <stdbool.h>
#include <stdlib.h> #define AtomType int typedef enum{ATOM, LIST}ElemTag;
typedef struct GLNode{
ElemTag tag;
union{
AtomType atom;
struct GLNode* head;
};
struct GLNode* tail;
}GLNode; typedef GLNode* GenList; void init(GenList* gl);
void createGenList(GenList* gl, char* s);
void show(GenList gl); #endif

guangyibiao.c

#include "guangyibiao.h"

void init(GenList* gl){
*gl = NULL;
} //挑出逗号前的一个元素,元素可以是原子也可以是列表
bool server1(char* sub, char* hsub){ int n = strlen(sub);
int i = 0;
char ch = sub[0];
int k = 0;
//k的作用是,识别逗号是括号里的,还是括号外的,如果是括号外的逗号就跳出while了,括号内的逗号不跳出循环,继续往下找。
while(i < n && (ch != ',' || k != 0)){
if(ch == '('){
k++;
}
else if(ch == ')'){
k--;
}
i++;
ch = sub[i];
} //逗号在sub的范围内
if(i < n){
sub[i] = '\0';
strcpy(hsub, sub);
strcpy(sub, sub+i+1);
}
//括号不匹配
else if(k != 0) return false;
//sub本身就是表,比如为(1,2)时,i会等于n,所以把sub给hsub,sub就没有以后部分了
else{
strcpy(hsub, sub);
sub[0] = '\0';
} return true;
} //思路:递归创建节点,如果sub是列表,就递归调用自己
void createGenList(GenList* gl, char* s){ int n = strlen(s); //去掉s的外括号,比如s为(1,(2, 3)),处理后sub为1,(2, 3),并在sub的最后加上'\0'
char* sub = (char*)malloc(sizeof(char) * (n - 2));
char* hsub = (char*)malloc(sizeof(char) * (n - 2));
assert(NULL != sub && NULL != hsub);
strncpy(sub, s+1, n-2);
sub[n-2] = '\0'; GLNode* p = *gl;
while(strlen(sub) != 0){
if(NULL == p){
*gl = p = (GLNode*)malloc(sizeof(GLNode));
p->head = p->tail = NULL;
}else{
p = p->tail = (GLNode*)malloc(sizeof(GLNode));
p->head = p->tail = NULL;
}
assert(NULL != p); if(server1(sub, hsub)){
if(hsub[0] == '('){
p->tag = LIST;
createGenList(&(p->head), hsub);
}
else{
p->tag = ATOM;
p->atom = atoi(hsub);
}
}
}
}
//思路:递归
void show(GenList gl){
if(gl == NULL) return; if(gl->tag == ATOM){
printf("%d", gl->atom);
if(gl->tail != NULL){
printf(",");
}
//如果gl为ATOM的话,gl就不会有head,所以只需调用show(gl->tail)
show(gl->tail);
}
//如果gl为LIST的话,gl就会有head,也会有tail,所以调用show(gl->head),show(gl->tail)
else if(gl->tag == LIST){
printf("("); show(gl->head);
printf(")");
if(gl->tail != NULL){
printf(",");
}
show(gl->tail);
}
}

guangyibiaomai.c

#include "guangyibiao.h"

int main(){
GenList gl;
init(&gl); char* a = "(1,2,3)";
char* b = "(1,(2,3))";
char* c = "(1,(2),3)";
char* d = "((1,2),3)";
char* e = "((1,2,3))";
char* f = "((),1)";
char* g = "(1,(2,(3,4)),5)";
char* h = "((),1,(2,(3,(),4)),5)";
char* i = "(())"; createGenList(&gl, i);
if(gl != NULL){
printf("(");
show(gl);
printf(")\n");
} return 0;
}

c/c++ 广义表的更多相关文章

  1. 数据结构(C语言第2版)-----数组,广义表,树,图

    任何一个算法的设计取决于选定的数据结构,而算法的实现依赖于采用的存储结构. 之前线性表的数据元素都是非结构的原子类型,元素的值是不可再分的.下面学习的这两个线性表是很特殊的,其中数据元素本身也可能是一 ...

  2. 广义表 Head Tail

    head:取非空广义表的第一个元素 tail:取非空广义表除第一个元素外剩余元素构成的广义表 L=((x,y,z),a,(u,t,w)) head(L)为(x,y,z) head(head(L))为x ...

  3. 数据结构算法C语言实现(十九)--- 5.5&5.6&5.7广义表

    一.简述 传说Lisp的基本数据结构就是广义表,广义表也是具有典型递归属性的数据结构,此外,由于建表要处理字符串,用C语言处理起来也是一脸懵逼.....最后自己还想写一个将广义表还原成字符串的函数,一 ...

  4. javascript实现数据结构:广义表

    原文:javascript实现数据结构:广义表  广义表是线性表的推广.广泛用于人工智能的表处理语言Lisp,把广义表作为基本的数据结构. 广义表一般记作: LS = (a1, a2, ..., an ...

  5. 【C/C++】实现数据结构广义表

    1. 广义表的定义     每个元素可以为Atom,原子,也可以为线性表.      线性表的推广.线性表元素有唯一的前驱和后继,为线性表,而广义表是多层次的线性表      表头:第一个元素,可能是 ...

  6. 数据结构(C语言版)-第4章 串、数组和广义表

    补充:C语言中常用的串运算 调用标准库函数 #include<string.h> 串比较,strcmp(char s1,char s2) 串复制,strcpy(char to,char f ...

  7. 广义表操作 (ava实现)——广义表深度、广义表长度、打印广义表信息

    广义表是对线性表的扩展——线性表存储的所有的数据都是原子的(一个数或者不可分割的结构),且所有的数据类型相同.而广义表是允许线性表容纳自身结构的数据结构. 广义表定义: 广义表是由n个元素组成的序列: ...

  8. 数据结构:广义表的实现(Java)

    广义表的简单理解在这篇博文中:https://blog.csdn.net/lishanleilixin/article/details/87364496,在此不做赘述. Java实现广义表: pack ...

  9. 数据结构28:广义表及M元多项式

    广义表,又称为列表.记作: LS = (a1,a2,…,an) ;( LS 为广义表的名称, an 表示广义表中的数据). 广义表可以看作是线性表的推广.两者区别是:线性表中的数据元素只能表示单个数据 ...

随机推荐

  1. kubernetes 安装kong、kong-ingress-controlor

    一.关于kong的详细内容这里不再赘述,可以查看官网. kong升级到1.0以后功能越来越完善,并切新版本的kong可以作为service-mesh使用,并可以将其作为kubernetes的ingre ...

  2. Java设计模式之《模板模式》及使用场景

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9558825.html 模板模式,顾名思义,就是通过模板拓印的方式. 定义模板,就是定义框 ...

  3. python的partial()用法说明

    在functools模块中有一个工具partial(),可以用来"冻结"一个函数的参数,并返回"冻结"参数后的新函数. 很简单的解释,也是官方手册给的示例.对于 ...

  4. websocket 初识

    websocket 初识 前言 其实很早就知道了 websocket 这个概念了,像现在大火的直播等使用的就是 websocket.之前找爬虫工作,对面问我爬过 websocket 网站没,很汗颜,那 ...

  5. 12个敏捷过程的小提示Tips

    12个敏捷过程的小提示Tips 1. 可视化一切. 在团队里使用Scrum白板.同时走廊过道上也会挂上显示信息的白板,这些信息可以是公司战略.软件缺陷等等.可视化的好处是,员工经过这些白板时,能够了解 ...

  6. C# 插件热插拔

    所谓热插拔就是插件可以 在主程序不重新启动的情况直接更新插件, 网上有很多方案: https://www.cnblogs.com/happyframework/p/3405811.html 如下: 但 ...

  7. C# Winform打包部署时添加注册表信息实现开机自启动

    1.原理:需要开机自启动的程序,需要将其启动程序的路径写到注册表中指定的文件夹下 2. 写入注册表的方式有两种 a.在生成安装程序时配置 b.在程序运行时,动态配置 方法一:使用VS2010自带的打包 ...

  8. struts2_struts.xml配置文件讲解

    1.bean Bean详细讲解:https://www.cnblogs.com/lulu638/p/4340703.html 2.constant constant属性配置,可配置的属性可以参考def ...

  9. redis.conf 常见配置介绍

    参数说明redis.conf 配置项说明如下: 1. Redis默认不是以守护进程的方式运行,可以通过该配置项修改,使用yes启用守护进程 daemonize no 2. 当Redis以守护进程方式运 ...

  10. Netty实战一之异步和事件驱动

    Netty是一款异步的事件驱动的网络应用程序框架,支持快速地开发可维护的高性能的面向协议的服务器和客户端. 使用Netty你可以并不是很需要网络编程.多线程处理.并发等专业Java知识的积蓄. Net ...