node的实现是PostgreSQL的查询解析的基础,实现的关键是两个宏,makeNode和newNode。其他节点继承自Node节点,如果增加新的结构体,需要添加NodeTag中添加对应的枚举值,并在equal和nodetoString中添加对于的处理代码。当结构体少是很容易处理,如果结构体过多,维护会比较麻烦。PostgreSQL中大约有300个继承自node的结构体,写代码的人真是需要相当的勇气和毅力呀。

 

#include
<iostream>

#include
<string.h>

#include
<assert.h>

#include
<stdlib.h>

#include
<stdio.h>

using
namespace std;

 

enum NodeTag

{

    T_Stmt,

    T_Value

};

 

typedef
struct
Node

{

    NodeTag type;

}Node;

 

Node *newNodeMacroHolder ;

#define newNode(size, tag) \

    ( \

         assert((size) >= sizeof(Node)),        /* need the tag, at least */ \

         newNodeMacroHolder = (Node *) malloc(size), \

         newNodeMacroHolder->type = (tag), \

         newNodeMacroHolder \

    )

 

#define makeNode(_type_) ((_type_ *)newNode(sizeof(_type_),T_##_type_))

#define nodeTag(nodeptr) (((const Node *)(nodeptr))->type)

 

typedef
struct
Stmt

{

    NodeTag type;

    char *text;

}Stmt;

typedef
struct
Value

{

    NodeTag type;

    long
val;

}Value;

 

bool
equal(void *a,void *b)

{

    if(a == b)

        return
true;

    if (a == NULL || b == NULL)

        return
false;

    if(nodeTag(a) != nodeTag(b))

        return
false;

 

    switch(nodeTag(a)){

        case
T_Stmt:

            return
strcmp(((const
Stmt*)a)->text,((const
Stmt *)b)->text)==0? true:false;

        case
T_Value:

            return ((const
Value *)a)->val==((const
Value *)b)->val;

        default:

            cout<<"error:unknown type"<<endl;

    }

 

    return
false;

}

char * nodetoString(void *obj)

{

    char *r =(char *)malloc(1024);

 

    if (obj == NULL){

        strcpy(r,"<>");

    }

 

    switch(nodeTag(obj)){

        case
T_Stmt:

            sprintf(r,"<Stmt:%s>",((const
Stmt *)obj)->text);

            break;

        case
T_Value:

            sprintf(r,"<Value:%ld>",((const
Value *)obj)->val);

            break;

        default:

            strcpy(r,"<unknown node type>");

    }

 

    return r;

}

 

int
main(int argc,char *argv[])

{

    Stmt *s= makeNode(Stmt);

    if(s){

        char str[]="select * from a";

        s->text=str;

    }

        

    Stmt *t= makeNode(Stmt);

    if(t){

        char str[]="select * from b";

        t->text=str;

    }

    Value *v=makeNode(Value);

    if(v){

        v->val=100;

    }

 

    cout<<"t->text:"<<t->text<<endl;

    

    cout<<"equal:"<<equal(s,t)<<endl;

 

    cout<<nodetoString(t)<<endl;

 

    free(s);

    free(t);

    free(v);

    return 0;

}

 

如果准备使用C语言来实现node结构体的话,尤其是准备采用gcc编译的话,要注意将newNode的宏设置为

/* 针对gcc版本的newNode */

#define newNode(size, tag) \

({    Node *_result; \

    AssertMacro((size) >= sizeof(Node));/* 检测申请的内存大小,>>=sizeof(Node) */ \

    _result = (Node *) palloc0fast(size); /* 申请内存 */ \

    _result->type = (tag); /*设置TypeTag */ \

    _result; /*返回值*/\

})

 

参见我的另一篇笔记《PostgreSQL源码解读 基础结构 node

PostgreSQL 源码解读 node的模拟实现的更多相关文章

  1. PostgreSQL源码解读 基础结构 node

    一.node节点的定义 源代码路径postgresql-9.2.3/src/include/nodes/nodes.h 在查询解析SQL的查询部分,要用到大量的结构体,许多函数处理的逻辑类似,就是传入 ...

  2. Spark jdbc postgresql数据库连接和写入操作源码解读

    概述:Spark postgresql jdbc 数据库连接和写入操作源码解读,详细记录了SparkSQL对数据库的操作,通过java程序,在本地开发和运行.整体为,Spark建立数据库连接,读取数据 ...

  3. node.js require() 源码解读

    时至今日,Node.js 的模块仓库 npmjs.com ,已经存放了15万个模块,其中绝大部分都是 CommonJS 格式.这种格式的核心就是 require 语句,模块通过它加载.学习 Node. ...

  4. fastclick.js源码解读分析

    阅读优秀的js插件和库源码,可以加深我们对web开发的理解和提高js能力,本人能力有限,只能粗略读懂一些小型插件,这里带来对fastclick源码的解读,望各位大神不吝指教~! fastclick诞生 ...

  5. AFNetworking 3.0 源码解读(十一)之 UIButton/UIProgressView/UIWebView + AFNetworking

    AFNetworking的源码解读马上就结束了,这一篇应该算是倒数第二篇,下一篇会是对AFNetworking中的技术点进行总结. 前言 上一篇我们总结了 UIActivityIndicatorVie ...

  6. 第二十五课:jQuery.event.trigger的源码解读

    本课主要来讲解jQuery.event.trigger的源码解读. trigger = function(event, data, elem, onlyHandlers){ if(elem & ...

  7. 第二十四课:jQuery.event.remove,dispatch的源码解读

    本课还是来讲解一下jQuery是如何实现它的事件系统的.这一课我们先来讲一下jQuery.event.remove的源码解读. remove方法的目的是,根据用户传参,找到事件队列,从里面把匹配的ha ...

  8. 第二十三课:jQuery.event.add的原理以及源码解读

    本课主要来讲解一下jQuery是如何实现它的事件系统的. 我们先来看一个问题: 如果有一个表格有100个tr元素,每个都要绑定mouseover/mouseout事件,改成事件代理的方式,可以节省99 ...

  9. nodeJS之eventproxy源码解读

    1.源码缩影 !(function (name, definition) { var hasDefine = typeof define === 'function', //检查上下文环境是否为AMD ...

随机推荐

  1. MFC中关于CListBox控件添加水平滚动条

    首先是设置listbox控件的属性  Horizontal Scroll设为TRUE: 然后添加函数到CUighurRecognitionDlg.cpp(在CUighurRecognitionDlg. ...

  2. 【01】markdown语法

    [02]段落和换行 一个 Markdown 段落是由一个或多个连续的文本行组成,它的前后要有一个以上的空行(空行的定义是显示上看起来像是空的,便会被视为空行.比方说,若某一行只包含空格和制表符,则该行 ...

  3. json分享

    JSON是什么? JavaScript Object Notation (JSON) is a text format for the serialization of structured data ...

  4. Educational Codeforces Round 33 (Rated for Div. 2)

    A. Chess For Three time limit per test 1 second memory limit per test 256 megabytes input standard i ...

  5. Codeforces Round #362 (Div. 2)

    闲来无事一套CF啊,我觉得这几个题还是有套路的,但是很明显,这个题并不难 A. Pineapple Incident time limit per test 1 second memory limit ...

  6. MySQL主从复制报错一致性问题解决

    当MySQL主从复制中因为不一致报错的情况,我们第一时间想到的就是使用pt-table-checksum来进行检查主从一致并进行修复,但是使用此工具的前提是主从复制线程都为on状态, 所以这种情况下可 ...

  7. Java类方法 类变量

    类变量就是静态变量,类方法就是静态方法. 在理解类变量.类方法之前先看一段代码: class Person{ int age ; String name; static int totalFee; p ...

  8. iOS学习笔记15-序列化、偏好设置和归档

    一.本地持久化 所谓的持久化,就是将数据保存到硬盘中,使得在应用程序或机器重启后可以继续访问之前保存的数据.在iOS开发中,有很多数据持久化的方案,接下来我将尝试着介绍一下5种方案: plist文件( ...

  9. P3799 妖梦拼木棒 (组合数学)

    题目背景 上道题中,妖梦斩了一地的木棒,现在她想要将木棒拼起来. 题目描述 有n根木棒,现在从中选4根,想要组成一个正三角形,问有几种选法? 输入输出格式 输入格式: 第一行一个整数n 第二行n个整数 ...

  10. LA 3644 简单并查集

    题目大意:有一些简单的化合物,每个化合物由两种元素组成,把这些化合物按顺序装车,若k个化合物正好包含k种元素,那么就会爆炸.避免爆炸,有些化合物就不能装车.求有多少个不能装车. 题目分析:若k个化合物 ...