jsqlparser(四):解析SQL语句部件

4 篇文章 0 订阅

jsqlparser不仅可以解析完整的SQL语句也可以用来解析SQL语法片段:

如下示例,我们将"ORDER BY column"这个用于定义排序的ORDER BY语句解析为
net.sf.jsqlparser.statement.select.OrderByElement 对象

	@Test
	public void test14ParseOrderByElements() {
		try {
			/** 创建解析器对象 */
			CCJSqlParser parser = new CCJSqlParser(new StringProvider("ORDER BY column"));
			/** 调用 OrderByElements 方法返回 net.sf.jsqlparser.statement.select.OrderByElement 对象 */
			System.out.printf("elements:%s\n",parser.OrderByElements());
		} catch (ParseException e) {
			e.printStackTrace();
			assertTrue(false);
		}
	}

对每个SQL部件,CCJSqlParser都有一个对应的无参解析方法,一般情况下,这个方法名是解析对象的类名。所以我们可以基于此将这个解析SQL语句部件的方法改造一下变成一个通用方法:

    /**
	 * 调用{@link CCJSqlParser}解析SQL语句部件返回解析生成的对象,如{@code 'ORDER BY id DESC'}
	 * @param <T>
	 * @param input 
	 * @param method 指定调用的{@link CCJSqlParser}解析方法
	 * @param targetType 返回的解析对象类型
	 * @return
	 * @since 3.18.3
	 */
	public static <T> T parseComponent(String input,String method,Class<T> targetType){
		try {
			CCJSqlParser parser = new CCJSqlParser(new StringProvider(input));
			try {
				return checkNotNull(targetType,"targetType is null").cast(parser.getClass().getMethod(method).invoke(parser));
			} catch (InvocationTargetException e) {
				Throwables.throwIfUnchecked(e.getTargetException());
				throw new RuntimeException(e.getTargetException());
			}
		} catch (IllegalAccessException | NoSuchMethodException | SecurityException e) {
			Throwables.throwIfUnchecked(e);
			throw new RuntimeException(e);
		}
	}
	/**
	 * 调用{@link CCJSqlParser}解析SQL语句部件返回解析生成的对象,
	 * 使用{@code targetType}的类名作为解析方法名
	 * @see #parseComponent(String, String, Class)
	 * @since 3.18.3
	 */
	public static <T> T parseComponent(String input,Class<T> targetType){
		return parseComponent(input,
					checkNotNull(targetType,"targetType is null").getSimpleName(),
					targetType);
	}

调用示例:

	@Test
	public void test15ParseOrderByElement() {
		try {
			OrderByElement orderByElement= ParserSupport.parseComponent("column desc", OrderByElement.class);
			System.out.printf("element:%s\n",orderByElement);
		} catch (ParseException e) {
			e.printStackTrace();
			assertTrue(false);
		}
	}

应用示例:

	/**
	 * 
	 * 将指定的字段加入SQL语句ORDER BY 排序字段,作为最优先排序字段 
	 * @param sql
	 * @param column
	 * @return 返回加入column排序的SQL语句
	 */
	private String addOrderByColumn(String sql,String column) {
		if(!isNullOrEmpty(sql) && !isNullOrEmpty(column)) {
			Select select = ParserSupport.parseSelect(sql);
			if(select.getSelectBody() instanceof PlainSelect) {
				PlainSelect plainSelect = (PlainSelect) select.getSelectBody();
				List<OrderByElement> orderByElements = plainSelect.getOrderByElements();
				if(null == orderByElements) {
					orderByElements = Lists.newArrayListWithCapacity(1);
					plainSelect.setOrderByElements(orderByElements);
				}
				OrderByElement orderBy = parseComponent(column,OrderByElement.class);
				orderByElements.add(0,orderBy);
				return select.toString();
			}			
		}
		return sql;
	}

完整代码参见我的码云仓库:
https://gitee.com/l0km/sql2java/blob/dev/sql2java-manager/src/main/java/gu/sql2java/parser/ParserSupport.java
测试代码
https://gitee.com/l0km/sql2java/blob/dev/sql2java-manager/src/test/java/gu/sql2java/pagehelper/parser/JsqlParserTest.java

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是使用jsqlparser解析子查询sql的示例代码: ```java import java.io.StringReader; import java.util.List; import net.sf.jsqlparser.JSQLParserException; import net.sf.jsqlparser.expression.Expression; import net.sf.jsqlparser.parser.CCJSqlParserUtil; import net.sf.jsqlparser.statement.Statement; import net.sf.jsqlparser.statement.select.PlainSelect; import net.sf.jsqlparser.statement.select.Select; import net.sf.jsqlparser.statement.select.SubSelect; public class JSqlParserDemo { public static void main(String[] args) throws JSQLParserException { String sql = "SELECT * FROM (SELECT * FROM table1 WHERE id IN (SELECT id FROM table2)) AS subquery"; Statement statement = CCJSqlParserUtil.parse(sql); Select selectStatement = (Select) statement; PlainSelect plainSelect = (PlainSelect) selectStatement.getSelectBody(); SubSelect subSelect = (SubSelect) plainSelect.getFromItem(); Select subquery = (Select) subSelect.getSelectBody(); PlainSelect subqueryPlainSelect = (PlainSelect) subquery.getSelectBody(); List<Expression> expressions = subqueryPlainSelect.getSelectItems(); for (Expression expression : expressions) { System.out.println(expression.toString()); } } } ``` 上述代码中,我们首先将子查询sql语句作为字符串传入JSqlParser进行解析。然后,我们通过获取Select对象和PlainSelect对象来获取子查询的Select对象和PlainSelect对象。最后,我们可以通过获取子查询的PlainSelect对象来获取子查询中的Select字段。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

10km

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值