在日志分析中,经常会对记录的sql进行分析,所以将一整行sql格式化,进行多行缩就显得很有必要,许多数据库客户端都提供sql的格式化功能,但复杂的多层嵌套sql往往格式化的l还不够友好,所以就自己造了个。

import java.util.HashSet;
import java.util.LinkedList;
import java.util.Set;
import java.util.StringTokenizer; public class BasicFormatterImpl {
private static final Set<String> BEGIN_CLAUSES = new HashSet<String>();
private static final Set<String> END_CLAUSES = new HashSet<String>();
private static final Set<String> LOGICAL = new HashSet<String>();
private static final Set<String> QUANTIFIERS = new HashSet<String>();
private static final Set<String> DML = new HashSet<String>();
private static final Set<String> MISC = new HashSet<String>();
static final String indentString = " ";
static final String initial = "\n "; public String format(String source) {
return new FormatProcess(source).perform().trim();
} static {
BEGIN_CLAUSES.add("left");
BEGIN_CLAUSES.add("right");
BEGIN_CLAUSES.add("inner");
BEGIN_CLAUSES.add("outer");
BEGIN_CLAUSES.add("group");
BEGIN_CLAUSES.add("order"); END_CLAUSES.add("where");
END_CLAUSES.add("set");
END_CLAUSES.add("having");
END_CLAUSES.add("join");
END_CLAUSES.add("from");
END_CLAUSES.add("by");
END_CLAUSES.add("join");
END_CLAUSES.add("into");
END_CLAUSES.add("union"); LOGICAL.add("and");
LOGICAL.add("or");
LOGICAL.add("when");
LOGICAL.add("else");
LOGICAL.add("end"); QUANTIFIERS.add("in");
QUANTIFIERS.add("all");
QUANTIFIERS.add("exists");
QUANTIFIERS.add("some");
QUANTIFIERS.add("any"); DML.add("insert");
DML.add("update");
DML.add("delete"); MISC.add("select");
MISC.add("on");
} private static class FormatProcess {
boolean beginLine = true;
boolean afterBeginBeforeEnd = false;
boolean afterByOrSetOrFromOrSelect = false;
boolean afterValues = false;
boolean afterOn = false;
boolean afterBetween = false;
boolean afterInsert = false;
int inFunction = 0;
int parensSinceSelect = 0;
private LinkedList<Integer> parenCounts = new LinkedList<Integer>();
private LinkedList<Boolean> afterByOrFromOrSelects = new LinkedList<Boolean>(); int indent = 1; StringBuffer result = new StringBuffer();
StringTokenizer tokens;
String lastToken;
String token;
String lcToken; public FormatProcess(String sql) {
this.tokens = new StringTokenizer(sql, "()+*/-=<>'`\"[], \n\r\f\t", true);
} public String perform() {
this.result.append("\n "); while (this.tokens.hasMoreTokens()) {
this.token = this.tokens.nextToken();
this.lcToken = this.token.toLowerCase(); if ("'".equals(this.token)) {
String t;
do {
t = this.tokens.nextToken();
this.token += t;
} while ((!"'".equals(t)) && (this.tokens.hasMoreTokens()));
} else if ("\"".equals(this.token)) {
String t;
do {
t = this.tokens.nextToken();
this.token += t;
} while (!"\"".equals(t));
} if ((this.afterByOrSetOrFromOrSelect) && (",".equals(this.token))) {
commaAfterByOrFromOrSelect();
} else if ((this.afterOn) && (",".equals(this.token))) {
commaAfterOn();
} else if ("(".equals(this.token)) {
openParen();
} else if (")".equals(this.token)) {
closeParen();
} else if (BasicFormatterImpl.BEGIN_CLAUSES.contains(this.lcToken)) {
beginNewClause();
} else if (BasicFormatterImpl.END_CLAUSES.contains(this.lcToken)) {
endNewClause();
} else if ("select".equals(this.lcToken)) {
select();
} else if (BasicFormatterImpl.DML.contains(this.lcToken)) {
updateOrInsertOrDelete();
} else if ("values".equals(this.lcToken)) {
values();
} else if ("on".equals(this.lcToken)) {
on();
} else if ((this.afterBetween) && (this.lcToken.equals("and"))) {
misc();
this.afterBetween = false;
} else if (BasicFormatterImpl.LOGICAL.contains(this.lcToken)) {
logical();
} else if (isWhitespace(this.token)) {
white();
} else {
misc();
} if (!isWhitespace(this.token)) {
this.lastToken = this.lcToken;
}
} return this.result.toString();
} private void commaAfterOn() {
out();
this.indent -= 1;
newline();
this.afterOn = false;
this.afterByOrSetOrFromOrSelect = true;
} private void commaAfterByOrFromOrSelect() {
out();
newline();
} private void logical() {
if ("end".equals(this.lcToken)) {
this.indent -= 1;
}
newline();
out();
this.beginLine = false;
} private void on() {
this.indent += 1;
this.afterOn = true;
newline();
out();
this.beginLine = false;
} private void misc() {
out();
if ("between".equals(this.lcToken)) {
this.afterBetween = true;
}
if (this.afterInsert) {
newline();
this.afterInsert = false;
} else {
this.beginLine = false;
if ("case".equals(this.lcToken))
this.indent += 1;
}
} private void white() {
if (!this.beginLine)
this.result.append(" ");
} private void updateOrInsertOrDelete() {
out();
this.indent += 1;
this.beginLine = false;
if ("update".equals(this.lcToken)) {
newline();
}
if ("insert".equals(this.lcToken))
this.afterInsert = true;
} private void select() {
out();
this.indent += 1;
newline();
this.parenCounts.addLast(new Integer(this.parensSinceSelect));
this.afterByOrFromOrSelects.addLast(Boolean.valueOf(this.afterByOrSetOrFromOrSelect));
this.parensSinceSelect = 0;
this.afterByOrSetOrFromOrSelect = true;
} private void out() {
this.result.append(this.token);
} private void endNewClause() {
if (!this.afterBeginBeforeEnd) {
this.indent -= 1;
if (this.afterOn) {
this.indent -= 1;
this.afterOn = false;
}
newline();
}
out();
if (!"union".equals(this.lcToken)) {
this.indent += 1;
}
newline();
this.afterBeginBeforeEnd = false;
this.afterByOrSetOrFromOrSelect = (("by".equals(this.lcToken)) || ("set".equals(this.lcToken))
|| ("from".equals(this.lcToken)));
} private void beginNewClause() {
if (!this.afterBeginBeforeEnd) {
if (this.afterOn) {
this.indent -= 1;
this.afterOn = false;
}
this.indent -= 1;
newline();
}
out();
this.beginLine = false;
this.afterBeginBeforeEnd = true;
} private void values() {
this.indent -= 1;
newline();
out();
this.indent += 1;
newline();
this.afterValues = true;
} private void closeParen() {
this.parensSinceSelect -= 1;
if (this.parensSinceSelect < 0) {
this.indent -= 1;
this.parensSinceSelect = ((Integer) this.parenCounts.removeLast()).intValue();
this.afterByOrSetOrFromOrSelect = ((Boolean) this.afterByOrFromOrSelects.removeLast()).booleanValue();
}
if (this.inFunction > 0) {
this.inFunction -= 1;
out();
} else {
if (!this.afterByOrSetOrFromOrSelect) {
this.indent -= 1;
newline();
}
out();
}
this.beginLine = false;
} private void openParen() {
if ((isFunctionName(this.lastToken)) || (this.inFunction > 0)) {
this.inFunction += 1;
}
this.beginLine = false;
if (this.inFunction > 0) {
out();
} else {
out();
if (!this.afterByOrSetOrFromOrSelect) {
this.indent += 1;
newline();
this.beginLine = true;
}
}
this.parensSinceSelect += 1;
} private static boolean isFunctionName(String tok) {
char begin = tok.charAt(0);
boolean isIdentifier = (Character.isJavaIdentifierStart(begin)) || ('"' == begin);
return (isIdentifier) && (!BasicFormatterImpl.LOGICAL.contains(tok))
&& (!BasicFormatterImpl.END_CLAUSES.contains(tok))
&& (!BasicFormatterImpl.QUANTIFIERS.contains(tok)) && (!BasicFormatterImpl.DML.contains(tok))
&& (!BasicFormatterImpl.MISC.contains(tok));
} private static boolean isWhitespace(String token) {
return " \n\r\f\t".indexOf(token) >= 0;
} private void newline() {
this.result.append("\n");
for (int i = 0; i < this.indent; i++) {
this.result.append(" ");
}
this.beginLine = true;
}
} public static void main(String[] args) {
System.out.println(new BasicFormatterImpl()
.format("select aa,bb,cc,dd from ta1,(select ee,ff,gg from ta2 where ee=ff) ta3 where aa=bb and cc=dd group by dd order by createtime desc limit 3 "));
}
}

  

运行结果:

java格式化sql的更多相关文章

  1. mybatis自定义打印执行时间并格式化sql插件

    打印SQL的执行时间,我们可以实现mybatis官方我们提供的接口org.apache.ibatis.plugin.Interceptor,我们可以拦截的类有多个Executor,StatementH ...

  2. atitit.java解析sql语言解析器解释器的实现

    atitit.java解析sql语言解析器解释器的实现 1. 解析sql的本质:实现一个4gl dsl编程语言的编译器 1 2. 解析sql的主要的流程,词法分析,而后进行语法分析,语义分析,构建sq ...

  3. java连接sql server2000/2005

    接触Java或者JSP,难免会使用到数据库SQL Server 2000/2005(我使用2005标准版[9.0.3054]测试),经过自己的搜索和研究,使用JDBC连接SQL Server成功,特此 ...

  4. JAVA格式化时间日期

    JAVA格式化时间日期 import java.util.Date; import java.text.DateFormat; /** * 格式化时间类 * DateFormat.FULL = 0 * ...

  5. 使用java对sql server进行增删改查

    import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import ...

  6. jTDS Java连接SQL Server 2000数据库

    Java连接SQL Server 2000数据库时,有两种方法: (1)通过Microsoft的JDBC驱动连接.此JDBC驱动共有三个文件,分别 是mssqlserver.jar.msutil.ja ...

  7. 7.20.01 java格式化输出 printf 例子

    java格式化输出 printf 例子 importjava.util.Date; publicclassPrintf { publicstaticvoidmain(String[] args) { ...

  8. Java 获取SQL查询语句结果

    step1:构造连接Class.forName("com.mysql.jdbc.Driver"); Connection con = DriverManager.getConnec ...

  9. Java格式化时间

    Java格式化时间 将秒或者毫秒值格式化成指定格式的时间 效果图 工具类 工具类里我只列出了一种格式的格式化方式,可以根据自己的需求,修改"yyyy-MM-dd hh:mm:ss" ...

随机推荐

  1. Attribute(特性)

    一向都觉得.NET的Attribute好神秘.一个方框框住的东西,置身于类.方法的头部,本身不在类或方法里面,但又会起作用,有时作用还很大,仿佛充满了魔力.简直给人一种无冕之王,幕后之黑手的感觉! 某 ...

  2. bzoj3330: [BeiJing2013]分数

    口胡 题目hint都给你是一个三分函数了 还不会上三分套三分吗 exp函数又卡 精度又卡 什么sb毒瘤题 浪费时间

  3. YTU 2801: 用数字造数字(II)

    2801: 用数字造数字(II) 时间限制: 1 Sec  内存限制: 128 MB 提交: 244  解决: 168 题目描述 输入一个3位以上的整数,求其中最大的两个数字之和与最小的数字之和之间的 ...

  4. bootstrap3.0学习笔记记录1

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  5. Python基础第七天

    一.内容 二.练习 练习1 题目:编写函数,函数执行的时间是随机的 图示: 代码: import time import random def func(): s = 1 l = [] for i i ...

  6. Spring注解详细

    1.@controller 控制器(注入服务) 2.@service 服务(注入dao) 3.@repository dao(实现dao访问) 4.@component (把普通pojo实例化到spr ...

  7. Swift4 类与继承, 类型转换, 类型判断

    创建: 2018/03/05 完成: 2018/03/07 更新: 2018/03/09 完善标题 [Swift4 类与继承, 类型转换] -> [Swift4 类与继承, 类型转换与判断] 补 ...

  8. CCF2016.4 - A题

    思路:枚举每个点,看看它是否同时小于/大于前一个点和后一个点 import java.util.Scanner; public class Main { public static void main ...

  9. P3308 [SDOI2014]LIS(最小割+退流)

    传送门 设\(f[i]\)为以\(i\)结尾的最长上升子序列.可以考虑建这样一张图,对于所有的\(i<j,f[j]=f[i+1]\)连边\((i,j)\),\(f[i]=1\)的话连边\((S, ...

  10. Golang 入门 : 理解并发与并行

    Golang 的语法和运行时直接内置了对并发的支持.Golang 里的并发指的是能让某个函数独立于其他函数运行的能力.当一个函数创建为 goroutine 时,Golang 会将其视为一个独立的工作单 ...