简单模拟数据的多表查询的选、投、连操作

主要介绍的是数据库的查询操作的原理,当然是非常浅显易懂的那种原理,更深层次的细节操作,调优操作还没有去做

想实现一个数据库查的操作,如果不考虑性能上的问题,并且不考虑词法、语法分析的问题。换言之,我们已经明白用户的语义了,并且只想实现这个功能,不考虑性能上的代价的话。那操作只分为三步:
1 . 连接操作:将多个表,两两做笛卡尔积
2 . 选择操作:将符合语义的行留下
3 . 投影操作:将每行中,需要输出的属性留下

当然,这里面我们忽略了很多的问题,如:
1 . 我们并没有考虑索引的存在
2 . 没有考虑性能上的问题,如果可以的话我们要尽可能的先选择,再做连接


以上便是,我模拟的过程,我们还是说一下,数据库正常查找的过程吧:
1.它会将用户输入的sql语句,进行语法、词法的分析
2.知道要干什么之后,会建立一棵二叉树,把所有要操作的内容,都放在叶子节点上面,把操作的函数放在双亲节点上面,然后会分析处理时间,然后调优,分析时间,最后会选择一种最为省时的时间的处理方式


下面看一下我处理选投连的方式吧:

    /**
     * 选择
     */
    public static List<Integer> choose(List<String> keys, String[][] datas) {
        List<Integer> result = new ArrayList<Integer>();
        for (int i = 0; i < datas.length; i++) {
            result.add(i);
        }
        for (int i = 0; i < keys.size(); i++) {
            if (keys.get(i).contains("==")) {
                //判断属性和某一个值相等
                String lie = keys.get(i).split("==")[0];
                int _lie = -1;
                String key = keys.get(i).split("==")[1];
                for (int ii = 0; ii < datas[0].length; ii++) {
                    System.out.println("lie:--->" + lie);
                    System.out.println("data:--->" + datas[0][ii]);
                    if (lie.equals(datas[0][ii])) {
                        _lie = ii;
                        System.out.println("lie:--->  发生相等");
                    }
                }
                for (int j = 0; j < datas.length; j++) {
                    if (datas[j][_lie].equals(key)) {
                    } else {
                        result.remove((Integer) j);
                    }
                }
            } else if (keys.get(i).contains(">")) {
                String lie = keys.get(i).split(">")[0];
                int _lie = -1;
                String key = keys.get(i).split(">")[1];
                for (int ii = 0; ii < datas[0].length; ii++) {
                    if (lie.equals(datas[0][ii])) {
                        _lie = ii;
                    }
                }
                for (int j = 0; j < datas.length; j++) {
                    try {
                        if (Integer.parseInt(datas[j][_lie]) > Integer.parseInt(key)) {
                        } else {
                            result.remove((Integer) j);
                        }
                    } catch (Exception e) {
                        result.remove((Integer) j);
                    }
                }
            } else if (keys.get(i).contains("=")) { //判断两个不同的属性的值相等
                String key1 = keys.get(i).split("=")[0];
                String key2 = keys.get(i).split("=")[1];
                int _key1 = -1;
                int _key2 = -1;
                for (int ii = 0; ii < datas[0].length; ii++) {
                    if (key1.equals(datas[0][ii])) {
                        _key1 = ii;
                    }
                    if (key2.equals(datas[0][ii])) {
                        _key2 = ii;
                    }
                }
                for (int j = 0; j < datas.length; j++) {
                    if (datas[j][_key1].equals(datas[j][_key2])) {

                    } else {
                        result.remove((Integer) j);
                    }
                }
            }
        }
        return result;
    }


    /**
     * 投影
     */
    public static List<String> projection(List<Integer> result, List<String> information, String[][] datas) {
        //result是要第几行
        //information筛选的信息
        //datas原始数据
        List<String> resultList = new ArrayList<String>();//结果集合
        List<Integer> keys = new ArrayList<Integer>();//最终要选择出来的属性的列号
        String[] data1 = new String[datas[0].length];//属性信息,有几个,分别是什么
        for (int i = 0; i < data1.length; i++) {
            data1[i] = datas[0][i];//赋值
        }
        for (int i = 0; i < data1.length; i++) {
            if (information.contains(data1[i])) {
                keys.add(i);
            }
        }
        String text = "";
        for (int i = 0; i < datas.length; i++) {
            if (result.contains((Integer) i)) {
                for (int j = 0; j < datas[0].length; j++) {
                    if (keys.contains(j)) {
                        text = text + datas[i][j] + "  ";
                    }
                }
                text = text.substring(0, text.length() - 1);
                resultList.add(text);
                text = "";
            }
        }
        return resultList;
    }


    /**
     * 连接
     */
    public static String[][] connect(List<String> tables) {
        String[] table1 = Util.readFileTo1(tables.get(0));//读取第一个表
        String[] table2 = Util.readFileTo1(tables.get(1));//读取第二个表
        String[] sum = new String[table1.length * table2.length];//建立新的表,做笛卡尔积
        int count = 0;
        for (int i = 0; i < table1.length; i++) {//做笛卡尔积的过程
            for (int j = 0; j < table2.length; j++) {
                sum[count] = table1[i] + " " + table2[j];
                count++;
            }
        }
        int length = sum[0].split(" ").length;//求出笛卡尔积后,一行的属性个数
        String[][] result = new String[sum.length][length];
        for (int i = 0; i < sum.length; i++) {
            System.out.println("---->  " + sum[i]);
        }
        for (int i = 0; i < result.length; i++) {
            for (int j = 0; j < result[0].length; j++) {
                result[i][j] = sum[i].split(" ")[j];
            }
        }
        for (int i = 0; i < result.length; i++) {
            for (int j = 0; j < result[0].length; j++) {
                System.out.print(result[i][j] + "\\t");
            }
            System.out.println();
        }
        return result;
    }

我们主要处理问题的三个函数就写好了,下面可以写一个主函数来测试一下:
在主函数中,我们需要,先调用连接函数,再调用选择函数,最后调用投影函数,这样就可以返回正确的答案啦,有空的时候我再把调用的函数,和效果图粘上来~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容