在日志分析中,经常会对记录的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. Django's CSRF mechanism

    Forbidden (403) CSRF verification failed. Request aborted. You are seeing this message because this ...

  2. 反爬统计 数据库 sql CASE

    -- 经排查日志,发现ordertest.com下的url检测,频繁<Response [403]>,Forbidden;再进一步查询数据库数据:逐日统计错误临时表test_error_t ...

  3. php判断手机号码

      //PHP判断手机号码 public function isMobile($params) { $pattern = "/^(13[0-9]|14[0-9]|15[0-9]|17[0-9 ...

  4. linux下的git安装及配置

    一.yum安装方式 1.安装 $ yum install curl-devel expat-devel gettext-devel openssl-devel zlib-devel $ yum -y ...

  5. Lightoj 1068(数位DP)

    求一段区间中被k整除,各个位数相加之和被k整除的数的个数. 这不是重点,重点是k太大了,最大值有10000,所以不能直接开那么大的数组. 仔细分析一下可以发现,由于数最大是2的31次方(2147483 ...

  6. bzoj3696

    3696: 化合物 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 245  Solved: 131[Submit][Status][Discuss] ...

  7. UI:通讯录实现

    通讯录实现草图: 代码: #pragma mark (.h文件)-------------------------------------------------------------------- ...

  8. ThinkPHP3.2.3学习笔记2---模型

    一.模型实例化1.直接实例化可以和实例化其他类库一样实例化模型类,例如:$User = new \Home\Model\UserModel();$Info = new \Admin\Model\Inf ...

  9. Java实现Excel数据批量导入数据库

    Java实现Excel数据批量导入数据库 概述: 这个小工具类是工作中的一个小插曲哦,因为提数的时候需要跨数据库导数... 有的是需要从oracle导入mysql ,有的是从mysql导入oracle ...

  10. 一个 Java 对象到底有多大?

    阅读本文大概需要 2.8 分钟. 出处:http://u6.gg/swLPg 编写 Java 代码的时候,大多数情况下,我们很少关注一个 Java 对象究竟有多大(占据多少内存),更多的是关注业务与逻 ...