cms查询系统(二)json形式参数的设计与解析

1 前言

本篇文章主要来说明下代码模块的设计。像我们这种菜鸟级别,只有平时多读读源码,多研究和探讨其中的设计才可能提升自己,写出高质量的代码。

没有最好的设计,只有更好的设计,所以在发表我自己的愚见的同时,希望小伙伴们相互探讨更好的设计,有探讨才有更大的进步。

2 题目及分析

我们维护了一个数据中心,对外提供查询API,如何能让用户随意的添加查询条件,而不用修改后台的查询代码呢?用户如何配置查询条件,从而达到如下的sql效果呢?:

a.name='lg' or b.age>12
b.id in (12,34,45)
c.updateTime>'2015-3-28' and (b.id=2 or d.age<23)
e.age>f.age

2.1 查询参数的传递方式

我们作为API设计者,该如何让用户方便的传递他们任意的查询需求呢?这是我们要思考的地方。

目前来看比较好的方式莫过于:用户通过json来表达他们的查询需求。

2.2 查询的本质分析

从上面的查询来看,我们可以总结出来查询条件无非就是某个字段满足什么样的条件。这里有三个对象:

  • 查询的字段 如 b.age
  • 条件值 如 12
  • 怎样满足条件值 如 >

2.3 查询的配置分析

这样我们就可以清晰明了了,一个查询条件无非就是三个内容,所以可以如下配置:

{
    "columns":"b.age",
    "oper":">",
    "value":12
}

很显然,上面的确很麻烦,我们无非是要表达这三个内容,所以就要简化配置:

{
    "b.age":12,
    "oper":">"
}

还是不够简化,如何把操作符 > 也放置进去呢?如下

{
    "b.age@>":12
}

这样我们就可以把三个对象表达清楚了,将查询的字段和操作符合并起来作为key,并使用分隔符@分割两者,条件值作为value。这样就做到了,非常简化的查询配置。

接下来又面临一个问题,如何表达查询条件之间的and or 关系呢?即如何表达下面的内容呢?

c.age>14 and (b.id=2 or d.age<23)

借鉴mongodb的查询方案,可以如下配置:

{
    "c.age@>":14,
    "$or":{
        "b.id@=":2,
        "d.age@<":23
    }
}

通过配置一个$or作为key表明里面的几个查询条件是or的关系,如果是$and表明里面的查询条件之间是and的关系,外层默认是and的关系。

同时我们再回顾下,mongodb所作出的查询设计,也是通过用户配置json形式来表达查询意图,但是我们来看下它是如何查询

a.age>12 对应的mongodb的查询方式为:
{
    a.age : {
        $gt : 22
    }
}

我们的查询方式是

{
    "a.age@>":22
}

虽然看似我们的更加简单,mongodb的更加繁琐,主要是mongodb认为对于一个字段,可以有多个查询条件的,为了支持更加复杂的查询,如下:

{
    a.age : {
        $lt :24,
        $gt : 17
    }
}

然而我们也可以对此进行拆分,如下,同样满足:

{
    "a.age@>":17,
    "a.age@<":24,
}

各有各的好处和缺点,我就是我,颜色不一样的烟火。哈哈。。。

3 代码设计与实现

3.1 解析器接口的设计

对题目进行分析完了之后,就要考虑如何实现这样的json配置到sql的转化。实现起来不难,最重要的是如何做出一个高扩展性的实现?

再来看下下面的例子:

{
    "a.name@=":"lg",
    "b.age@>":12,
    "c.id@in":[12,13,14],
    "d.time@time>":"2015-3-1"
}

其实就是针对每个自定义的操作符进行相应的处理,所以就有了解析器接口:

public interface SqlParamsParser {

    //这里表示该解析器是否支持对应的操作符
    public boolean support(String oper);

    public String getParams(String key,Object value,String oper);

    public SqlParamsParseItemResult getParamsResult(String key,Object value,String oper);

}

其中SqlParamsParseItemResult,则是把解析后的结果分别存储起来,而不是直接拼接成一个字符串,主要为了直接拼接字符串式的sql注入,它的内容如下:

public class SqlParamsParseItemResult {
    private String key;
    private String oper;
    private Object value;
}

上面的key oper value 则是解析后的内容。下面举例说明

以"b.age@>“:12 为例,其中getParams方法中的 key就是b.age, value就是12, oper就是> 而这个方法的返回的字符串结果为:

b.age>12

返回的SqlParamsParseItemResult存储的内容为分别为 key=b.age ; oper=> ; value=12

以"c.id@in”:[12,13,14]为例,其中getParams方法中的 key就是c.id,value就是一个List集合,oper就是in ,这个方法的返回结果为:

c.id in (12,13,14)

返回的SqlParamsParseItemResult存储的内容为分别为 key=c.id ; oper=in ; value=(12,13,14)

以"d.time@time>“:“2015-3-1"为例,其中getParams方法中的 key就是c.id,value就是一个List集合,oper就是in,这个方法的返回结果为:

unix_timestamp(d.time) > 1425139200     (2015-3-1对应的秒数)

返回的SqlParamsParseItemResult存储的内容为分别为 key=unix_timestamp(d.time) ; oper=> ; value=1425139200

3.2 解析器接口的抽象类

解析器有很多相同的地方,这就需要我们进行抽象,抽出共性部分,留给子类去实现不同的部分。所以有了抽象类AbstractSqlParamsParser

有哪些共性部分和非共性部分呢?

  • 共性部分: 就是support方法。每个解析器支持某几种操作符,所以判断该解析器是否支持当前的操作符的逻辑是共同的,所以如下:

    public abstract class AbstractSqlParamsParser implements SqlParamsParser{
    
        private String[] opers;
        private boolean ignoreCase=true;
    
        protected void setOpers(String[] opers){
            this.opers=opers;
        }
        protected void setIgnoreCase(boolean ignoreCase){
            this.ignoreCase=ignoreCase;
        }
        @Override
        public boolean support(String oper) {
            if(opers!=null && oper!=null){
                for(String operItem:opers){
                    if(ignoreCase){
                        operItem=operItem.toLowerCase();
                        oper=oper.toLowerCase();
                    }
                    if(operItem.equals(oper)){
                        return true;
                    }
                }
            }
            return false;
        }
    }
    

    opers属性表示当前解析器所支持的所有操作符。ignoreCase表示在匹配操作符的时候是否忽略大小写。这两个属性都设置成private,然后对子类开放了protected类型的set方法,用于子类来设置这两个属性。

  • 非共性部分:留出了doParams方法供子类来具体实现
    @Override
    public SqlParamsParseItemResult getParamsResult(String key, Object value,
            String oper) {
        return doParams(key, processStringValue(value), oper);
    }
    
    protected abstract SqlParamsParseItemResult doParams(String key, Object value, String oper);
    

3.3 解析器接口的实现类

目前内置了几个常用的解析器实现,类图如下: 

以TimeSqlParamsParser为例来简单说明下:

它主要是用于解析如下形式的:

{
    "d.time@time>":"2015-3-1"
}

最终想达到的效果是:

unix_timestamp(d.time) > 1425139200

它的解析过程如下:

/**
 * 以d.time@time>'2015-3-1'为例
 * 初始参数 key=d.time; value='2015-3-1'; oper=time>
 * 解析后的key=unix_timestamp(d.time); value=1425139200('2015-3-1'对应的秒数); oper=>
 */
@Override
protected SqlParamsParseItemResult doParams(String key, Object value, String oper) {
    String timeKey="unix_timestamp("+key+")";
    String realOper=oper.substring(4+fullTimeFlag.length());
    if(value instanceof String){
        String tmp=(String)value;
        Assert.isLarger(tmp.length(),2,"时间参数不合法");
        //默认进行了字符串处理,即加上了'',现在要去掉,然后解析成时间的秒数
        value=tmp.substring(1,tmp.length()-1);
        try {
            SimpleDateFormat format=new SimpleDateFormat(timeFormat);
            Date date=format.parse((String)value);
            value=date.getTime()/1000;
        } catch (ParseException e) {
            e.printStackTrace();
            throw new IllegalArgumentException("timeFormat为"+timeFormat+";value="+value+";出现了解析异常");
        }
    }else{
        Assert.isInstanceof(value,Number.class,"时间参数必须为时间的秒数");
    }
    return new SqlParamsParseItemResult(timeKey,realOper,value);
}

解析过程其实就是对key value oper 进行了不同程度的转换。

同时TimeSqlParamsParser还支持其他时间形式的解析,如"2015-3-1 12:23:12”,只需如下方式创建一个解析器:

new TimeSqlParamsParser("yyyy-MM-dd HH:mm:ss","full_")

然后他就能够解析下面的形式:

{
    "d.time@full_time>":"2015-3-1 12:23:12"
}

同时又能保留原有的形式,两者互不干扰。

4 DefaultSqlParamsHandler使用解析器

有了解析器的一系列实现,下面就需要一个综合的类来使用这些解析器。这就是DefaultSqlParamsHandler:

4.1 注册使用解析器

public class DefaultSqlParamsHandler {

private List<SqlParamsParser> sqlParamsParsers;

public DefaultSqlParamsHandler(){
    sqlParamsParsers=new ArrayList<SqlParamsParser>();
    sqlParamsParsers.add(new DefaultSqlParamsParser());
    sqlParamsParsers.add(new InSqlParamsParser());
    sqlParamsParsers.add(new TimeSqlParamsParser());
    sqlParamsParsers.add(new TimeSqlParamsParser("yyyy-MM-dd HH:mm:ss","full_"));
    sqlParamsParsers.add(new DefaultColumnSqlParamsParser());
}

内部已经注册了几个解析器。同时需要对外留出注册自定义解析器的方法:

public void registerSqlParamsHandler(SqlParamsParser sqlParamsParser){
    if(sqlParamsParser!=null){
        sqlParamsParsers.add(sqlParamsParser);
    }
}

public void registerSqlParamsHandler(List<SqlParamsParser> sqlParamsParsers){
    if(sqlParamsParsers!=null){
        for(SqlParamsParser sqlParamsParser:sqlParamsParsers){
            registerSqlParamsHandler(sqlParamsParser);
        }
    }
}

4.2 解析过程

这个过程不仅需要使用已经注册的解析器来解析,还包含对解析条件之间的and or 关系的递归处理。代码如下,不再详细说明:

private SqlParamsParseResult getSqlWhereParamsResultByAndOr(Map<String,Object> params,String andOr,
        boolean isPlaceHolder,SqlParamsParseResult sqlParamsParseResult){
    if(params!=null){
        String andOrDelititer=" "+andOr+" ";
        for(String key:params.keySet()){
            Object value=params.get(key);
            if(value instanceof Map){
                //这里需要进行递归处理嵌套的查询条件
                SqlParamsParseResult SqlParamsParseResultModel=null;
                if(key.equals(andKey)){
                    SqlParamsParseResultModel=processModelSqlWhereParams((Map<String,Object>)value,AND,isPlaceHolder);
                }else if(key.equals(orKey)){
                    SqlParamsParseResultModel=processModelSqlWhereParams((Map<String,Object>)value,OR,isPlaceHolder);
                }
                if(SqlParamsParseResultModel!=null && StringUtils.isNotEmpty(SqlParamsParseResultModel.getBaseWhereSql())){
                    sqlParamsParseResult.addSqlModel(andOrDelititer);
                    sqlParamsParseResult.addSqlModel("("+SqlParamsParseResultModel.getBaseWhereSql()+")");
                    sqlParamsParseResult.addArguments(SqlParamsParseResultModel.getArguments());
                }
            }else{
                //这里才是使用已经注册的解析器进行解析
                SqlParamsParseItemResult sqlParamsParseItemResult=processNormalSqlWhereParams(key,value,isPlaceHolder);
                if(sqlParamsParseItemResult!=null){
                    sqlParamsParseResult.addSqlModel(andOrDelititer);
                    sqlParamsParseResult.addSqlModel(sqlParamsParseItemResult.getSqlModel(isPlaceHolder,PLACE_HOLDER));
                    sqlParamsParseResult.addArgument(sqlParamsParseItemResult.getValue());
                }
            }
        }
        StringBuilder baseWhereSql=sqlParamsParseResult.getBaseWhereSql();
        if(StringUtils.isNotEmpty(baseWhereSql)){
            sqlParamsParseResult.setBaseWhereSql(new StringBuilder(baseWhereSql.substring(andOrDelititer.length())));
        }
    }
    return sqlParamsParseResult;
}

这里进行了递归调用,主要用于处理 $and $or 的嵌套查询,getSqlWhereParamsResultByAndOr可能内部调用了processModelSqlWhereParams,processModelSqlWhereParams内部又调用了getSqlWhereParamsResultByAndOr

private SqlParamsParseResult processModelSqlWhereParams(Map<String,Object> params,String andOr,boolean isPlaceHolder){
    return getSqlWhereParamsResultByAndOr(params,andOr,isPlaceHolder,new SqlParamsParseResult());
}

这里就是使用解析器进行解析的过程,先遍历每个解析器是否支持当前的操作符,如果支持则进行相应的解析

private SqlParamsParseItemResult processNormalSqlWhereParams(String key,Object value,boolean isPlaceHolder) {
    SqlParamsParseItemResult sqlParamsParseItemResult=null;
    String[] parts=key.split(separatorFlag);
    if(parts.length==2){
        for(SqlParamsParser sqlParamsParser:sqlParamsParsers){
            if(sqlParamsParser.support(parts[1])){
                sqlParamsParseItemResult=sqlParamsParser.getParamsResult(parts[0],value,parts[1]);
                break;
            }
        }
    }else{
        sqlParamsParseItemResult=new SqlParamsParseItemResult(key,"=",SqlStringUtils.processString(value));
    }
    return sqlParamsParseItemResult;
}

4.3 对外留出的扩展

{
    "c.age@>":14,
    "$or":{
        "b.id@=":2,
        "d.age@<":23
    }
}

这里面的@ $or 以及 $and 都是可以自己设定的,默认值是上述形式。

5 工程项目

这个小项目已经发布到osc上,见 osc的search-sqlparams项目

时间: 2024-10-01 03:11:22

cms查询系统(二)json形式参数的设计与解析的相关文章

cms查询系统(三)查询demo体验

安装相应的jar包到maven仓库 安装2个项目 有2个项目的jar包需要安装,分别是: search-sqlparams-1.3.0.jar 主要用于查询参数的解析 search-core-1.3.0.jar 整体的查询流程体系,需要使用上述参数解析包 上述两个项目地址分别如下: search-sqlparams search-core 以search-sqlparams-1.3.0.jar为例来说下安装步骤: 第一步:fork search-sqlparams 项目(这两个项目都是maven

cms查询系统(一)背景以及需求分析与设计

本人想做一个cms查询框架,用于解决实际的业务问题,顺便锻炼下能力 1 背景介绍 在一个配置管理系统即cms系统中,有很多的实体,实体间有很多的关联关系,这些实体就是构建成了一张网.如下图所示:  目前面临的问题是,如何轻松应对其他用户对实体的各种各样的查询需求? 2 需求整理 站在用户的角度来分析需求 2.1 用户的输入 2.1.1 用户要查询的数据 可能只需要获取某些想要的字段 可能想获取每个实体的所有字段 2.1.2 查询条件 各种各样的查询条件,如 如 =. !=.>.<.like.i

DreamWeaver打造Web查询系统(3)

dreamweaver|web|查询系统 二.将 index.asp 转换为动态页面 从本小节开始,我们逐步将前面建立的静态页面转换为动态页面,使其和数据库相结合,以完成我们最终的页面. 首先我们要把首页转换为动态页面.在首页中,仅有右侧的"超值推荐"部分需要从数据库中读取数据,因此,我们的目的是从数据库中的 tele 表中读取 Is_Good 字段值为1的记录并按降序显示就可以了. ok,开始工作.第1步,建立数据集.在绑定面板,选择 记录集(查询) 菜单,并在对话框内根据图 6-5

VB实现的电话语音成绩查询系统

一.引言 电话语音查询系统已广泛应用于电话银行.话费查询.证券 委托.自动缴费(水.电.气等费用).语音信箱.自动声讯服务.民航.公共 考务.铁路等部门的信息查询以及各种公共场所自动回答顾客提问等领域. 电话语音查询系统的工作流程是用户拨通电话语音查询系统的热线电话, 并根据电话中的语音提示,通过按电话键来查询电脑中存储的各种信息: 电脑自 动对用户的操作进行应答,并以语音形式将信息反馈给用户.其组成结构如图1 所示. 图1 自动语音查询系统 二.组成 电话语音查询系统 是现代电信技术与计算机技

基于s/c的数据库查询系统的设计 ,毕业论文的题目 ,理不清思路。麻烦各位大神给个指导

问题描述 基于s/c的数据库查询系统的设计 ,毕业论文的题目 ,理不清思路.麻烦各位大神给个指导 建立一个数据库,并查询,这个好办.但是一加前面的条件就不太明白了.s/c也大致了解,但是这两个要怎么联系到一起呢.好晕啊.我是急着写论文,这些知识只学了皮毛,问题描述如果那里说错了还请担待.我只求一个大体的思路.帮帮忙吧,各路大神们. 解决方案 其实也就是让一个程序是客户端,里面跑着TCP客户端的程序,然后通过TCP连接到你的一个TCP的服务器 这些TCP的程序在网上很多,如果就是想做个毕设可以直接

公交车路线查询系统后台数据库设计——关联地名和站点

在<公交车路线查询系统后台数据库设计--查询算法>一文中,已经实现了查询站点到站点的路线 查询算法,但是,现实中用户不一定使用站点进行查询,而是使用地名.因此,公交车查询系统数据库必 需记录地名与站点的对应关系,在查询时将地名映射为站点.根据实际情况,某一地点附近通常有几个站 点,因此,地名与站点之间是多对多的关系.显然,只需创建一个地名站点关系表stop_spot(Stop,Spot) 用于储存这个关系即可.数据库关系图如下: 注: Route:路线表 Stop:站点表 Spot:地名表 s

【百度地图API】建立全国银行位置查询系统(二)——怎样为地图添加控件

原文:[百度地图API]建立全国银行位置查询系统(二)--怎样为地图添加控件 <摘要>你将在第二章中学会以下知识: 使用手写代码的利器--notepad++: 如何为地图添加控件--鱼骨.鹰眼.比例尺.2D3D转换控件.版权控件. ----------------------------------------------------------------------------------------------------------------- 一.安装配置notepad++ 为什么

表格中自适应成绩查询系统

菜单式成绩查询 每个学校都会用到成绩查询.常规的查询查询方式是输入待查询对象的相关信息后让系统提取成绩资料. 这种方式有两个缺点: 1.需要录入文字,对于不会打字或者字符录入速度慢者不方便: 2.如果录入了错别字.同音字,则无法查询到正确的结果. 今天教大家一种新的查询方式,不需要录入任何字符就可以查询任何资料.同时因为不需要录入字符,也就避免了错误的产生. 假设图A是学校中所有班级的平均成绩表,其中各系别的学期长度不同,分别有2年.3年.4年,所以中间存在空白区. 图A 平均成绩表 在本例中,

基于多种服务的地理位置查询系统

本文讲的是基于多种服务的地理位置查询系统, 简介 我的这篇文章讨论了 PostGIS 以及查询地理数据的几种方法.这篇文章将集中讨论构建一个免费的地理服务系统,并聚合呈现结果. 概述 总的来说,我们将会向不同的网络服务(或APIs)发起请求,对响应结果做反向地理编码后再聚合展示. 比较 Geonames 和 OpenStreetMap 下表罗列了二者之间的部分差别: 二者用途不同.Geonomes 用于城市/行政区/国家数据,可被用于地理编码.OpenStreetMap 拥有更加详尽的数据(使用