在我們系統開發過程中越來越多的使用JSON作為數據交換的載體。JSON的特點是層次結構簡潔清晰,易於人閱讀和編寫。同時也易於機器解析和生成。對於機器解析和生成這點本人表示認可,然而對於易於人閱讀和編寫可以通過下面一段JSON字符串來驗證下:
{"strValue":"here is str value","nullValue":null,"intvalue":999,"doublevalue":999,"booleanValue":true,"array":["a1","a2",true,2,33.3,null,{"innerStr":"here is a inner str","innerInteger":123456789},["Hi, found me ?"]],"innerOBJ":{"innerStr":"here is a inner str","innerInteger":123456789}}
通過上面的字符串可以看到系統間交換數據的JSON字符串通常是去除了格式的,這樣的JSON我們閱讀起來貌似並不是那么輕松。在網上有很多JSON格式和着色的插件,借助這些插件可以將JSON字符串進行格式化,這時候JSON的易於人閱讀才表現出來。
{ "strValue": "here is str value", "nullValue": null, "intvalue": 999, "doublevalue": 999, "booleanValue": true, "array": [ "a1", "a2", true, 2, 33.3, null, { "innerStr": "here is a inner str", "innerInteger": 123456789 }, [ "Hi, found me ?" ] ], "innerOBJ": { "innerStr": "here is a inner str", "innerInteger": 123456789 } }
閑來無事整了一個JSON格式和着色小工具,下面直接上圖:
主要的業務代碼即為下面的工具類,提供了JSON校驗、格式化、着色、刪除空格、刪除空格並轉義、去除轉義等功能。
import java.io.FileReader; import java.io.IOException; import java.util.regex.Matcher; import java.util.regex.Pattern; import javax.script.Invocable; import javax.script.ScriptEngine; import javax.script.ScriptEngineManager; import javax.script.ScriptException; /** * JSON格式化工具 * @author StarlightFu * */ public class JsonFormatUtil { private static final String OBJECT_BRACE_REG = "([\\{\\}])"; private static final String ARRAY_BRACE_REG = "([\\[\\]])"; private static final String PROPERTY_NAME_REG = "(\\\".*\\\")(\\:)(.*)(\\,)?"; private static final String STRING_REG = "\\\"([^\"]*)\\\"(\\,)?$"; private static final String NUMBER_REG = "(-?\\d+)(\\,)?$"; /** JSON着色樣式 */ public static final String JSON_STYLE = "<style>pre{font-family:\"微軟雅黑\";font-size:10px;}.ObjectBrace{color:#00AA00;font-weight:bold;}.ArrayBrace{color:#0033FF;font-weight:bold;}" + ".PropertyName{color:#CC0000;font-weight:bold;}.String{color:#007777;}.Number{color:#AA00AA;}.Boolean{color:#0000FF;}" + ".Function{color:#AA6633;text-decoration:italic;}.Null{color:#0000FF;}.Comma{color:#000000;font-weight:bold;</style>"; /** * 校驗JSON格式是否正確 * @param json JSON字符串 * @return true:格式正確;false:格式錯誤 */ public static boolean validation(String json){ boolean result = false; FileReader reader = null; try { String valJson = "function valJson(jsonStr){var result=false;try{eval('('+jsonStr+')');result=true;}catch(e){result=false;}return result;}"; ScriptEngineManager manager = new ScriptEngineManager(); ScriptEngine engine = manager.getEngineByName("javascript"); engine.eval(valJson); if (engine instanceof Invocable) { Invocable invoke = (Invocable) engine; result = (Boolean) invoke.invokeFunction("valJson", json); } } catch (ScriptException e) { throw new ServiceException("執行JSON校驗JS異常"); } catch (NoSuchMethodException e) { throw new ServiceException("執行JSON校驗JS異常"); } finally { try { if (reader != null) { reader.close(); } } catch (IOException e) { e.printStackTrace(); } } return result; } /** * 格式化JSON字符串 * @param jsonStr * @return * @throws ServiceException */ public static String format(String jsonStr)throws ServiceException{ boolean valResult = false; valResult = JsonFormatUtil.validation(jsonStr); if (!valResult) { throw new ServiceException("JSON格式錯誤"); } if (null == jsonStr || "".equals(jsonStr)) return ""; StringBuilder sb = new StringBuilder(); char last = '\0'; char current = '\0'; int indent = 0; boolean inString = false; char inStringBegin = '\0'; for (int i = 0; i < jsonStr.length(); i++) { last = current; current = jsonStr.charAt(i); if (inString && current == inStringBegin) { // 判斷前一個字符是否為 \ if (last != '\\') { inString = false; inStringBegin = '\0'; } sb.append(current); } else if (!inString && (current == '"' || current == '\'')) { inString = true; inStringBegin = current; sb.append(current); } else if (!inString && (current == ' ' || current == '\t' || current == '\n')) { current = '\0'; } else if (!inString && current == ':') { sb.append(current).append(" "); } else if (!inString && current == ',') { sb.append(current).append('\n').append(indentBlank(indent)); } else if (!inString && (current == '[' || current == '{')) { indent++; sb.append(current).append('\n').append(indentBlank(indent)); } else if (!inString && (current == ']' || current == '}')) { indent--; sb.append('\n').append(indentBlank(indent)).append(current); } else { sb.append(current); } } return sb.toString(); } /** * 添加縮進 * @param indent * @return */ private static String indentBlank(int indent) { StringBuilder sb = new StringBuilder(); for (int i = 0; i < indent; i++) { sb.append(" "); } return sb.toString(); } /** * JSON語法着色 * @param formatJsonStr 格式化后的JSON字符串 * @return */ public static String coloring(String formatJsonStr) { StringBuilder sb = new StringBuilder(); String[] nodes = formatJsonStr.split("\n"); for (int i = 0; i < nodes.length; i++) { String node = nodes[i]; Pattern r = Pattern.compile(JsonFormatUtil.OBJECT_BRACE_REG); Matcher m = r.matcher(node); node = m.replaceAll("<span class='ObjectBrace'>$1</span>"); r = Pattern.compile(JsonFormatUtil.ARRAY_BRACE_REG); m = r.matcher(node); node = m.replaceAll("<span class='ArrayBrace'>$1</span>"); r = Pattern.compile(JsonFormatUtil.PROPERTY_NAME_REG); m = r.matcher(node); node = m.replaceAll("<span class='PropertyName'>$1</span>$2$3$4"); r = Pattern.compile(JsonFormatUtil.STRING_REG); m = r.matcher(node); node = m.replaceAll("<span class='String'>\"$1\"</span><span class='Comma'>$2</span>"); r = Pattern.compile(JsonFormatUtil.NUMBER_REG); m = r.matcher(node); node = m.replaceAll("<span class='Number'>$1</span><span class='Comma'>$2</span>"); sb.append(node); if (i < nodes.length - 1) { sb.append("\n"); } } return sb.toString(); } /** * 格式化JSON並着色 * @param json * @return */ public static String formatAndColoring(String json){ return JsonFormatUtil.coloring(JsonFormatUtil.format(json)); } /** * 刪除空格 * @param jsonStr * @return */ public static String removeSpace(String jsonStr){ StringBuilder sb = new StringBuilder(); char last = '\0'; char current = '\0'; char inStringBegin = '\0'; boolean inString = false; for (int i = 0; i < jsonStr.length(); i++) { last = current; current = jsonStr.charAt(i); if (inString && current == inStringBegin) { // 判斷前一個字符是否為 \ if (last != '\\') { inString = false; inStringBegin = '\0'; } sb.append(current); } else if (!inString && (current == '"' || current == '\'')) { inString = true; inStringBegin = current; sb.append(current); } else if (!inString && (current == ' ' || current == '\t' || current == '\n')) { current = '\0'; }else{ sb.append(current); } } return sb.toString(); } /** * 刪除空格並轉義 * @param jsonStr * @return */ public static String removeSpaceAndEscape(String jsonStr){ jsonStr = JsonFormatUtil.removeSpace(jsonStr); return jsonStr.replaceAll("\"", "\\\\\""); } /** * 去除轉義 * @param jsonStr * @return */ public static String removeEscape(String jsonStr){ return jsonStr.replaceAll("\\\\\"", "\""); } public static void main(String[] args) { String jsonStr = "{\"strValue\":\"here is str value\",\"nullValue\":null,\"intvalue\":999,\"doublevalue\":999,\"booleanValue\":true,\"array\":[\"a1\",\"a2\",true,2,33.3,null,{\"innerStr\":\"here is a inner str\",\"innerInteger\":123456789},[\"Hi, found me ?\"]],\"innerOBJ\":{\"innerStr\":\"here is a inner str\",\"innerInteger\":123456789}}"; System.out.println(JsonFormatUtil.format(jsonStr)); System.out.println(JsonFormatUtil.coloring(JsonFormatUtil.format(jsonStr))); } }
下面分享下基於上述代碼實現的桌面小應用,有興趣的同學可以下載,URL:JSON格式化小工具