手写一个json格式化 api
最近写的一个东西需要对json字符串进行格式化然后显示在网页上面。
我就想去网上找找有没有这样的api可以直接调用。百度 json api ,搜索结果都是那种只能在网页上进行校验的工具,没有api。
那只有自己去实现一个json 格式化工具。
仔细分析,实现起来并不是很困难,至少思路很清晰。
需要解决的几个问题:
- 对json的校验:主要是符号的匹配;
- 格式化预处理:去除键值对之间的空白字符;
- 格式化:主要是缩进的问题,要符合json通常展示的格式。
解决的办法:
- 针对A问题:
可以采用栈去匹配符号,大括号、中括号、双引号等。
- 针对B问题:
可以用fastjson的JSONObject.toString()方法实现,自动去除键值对之间的空白字符。
- 针对C问题:
在需要换行的地方追加\n,缩进的地方追加\t。通过控制追加的\t的个数从而控制缩进量。如:
"{"、"[" 后面追加 \n,并且\t在上次数量上加一;
"}"、"]" 追加\n,\t在上次的数量上减一;
",": 追加\n和上次相同数量的\t。
具体实现:
package com.lm.algorithm.json; import com.alibaba.fastjson.JSONException;
import com.alibaba.fastjson.JSONObject; /**
* Created by LiuMian on 2015/12/17.
* 将json字符串添加 换行符和制表符使其能够直接在网页上按照通常的json格式显示
*/
public class JSONFormat { private String src;
private int TABLength = 0; private final String BRACKET_LEFT = "[";
private final String BRACKET_RIGHT = "]";
private final String BRACE_LEFT = "{";
private final String BRACE_RIGHT = "}";
private final String COMMA = ",";
private final String LINE_BREAK = "\n";
private final String TAB = "\t"; public JSONFormat(String src){
this.src = src;
} public String format() throws JSONException{
try{
JSONObject json = JSONObject.parseObject(src);
}catch (JSONException e){
//对json进行简单的校验,如果不符合格式那么jsonobject在解析的时候会报错
throw e;
}
return format(src);
} private String format(String src) {
StringBuffer result = new StringBuffer();
char[] srcArray = src.toCharArray();
for (int index = 0; index < src.length(); index++) {
result.append(srcArray[index]); if (BRACE_LEFT.equals(String.valueOf(srcArray[index]))) //{
result.append(appendLINE_BREAKAndTAB(++TABLength)); if (BRACE_RIGHT.equals(String.valueOf(srcArray[index]))) //}
result.insert(result.length() - 1, appendLINE_BREAKAndTAB(--TABLength)); if (BRACKET_LEFT.equals(String.valueOf(srcArray[index]))) //[
result.append(appendLINE_BREAKAndTAB(++TABLength)); if (BRACKET_RIGHT.equals(String.valueOf(srcArray[index]))) //]
result.insert(result.length() - 1, appendLINE_BREAKAndTAB(--TABLength)); if (COMMA.equals(String.valueOf(srcArray[index]))) //,
result.append(appendLINE_BREAKAndTAB(TABLength));
}
return result.toString();
} //追加换行符和 确定长度的制表符
private String appendLINE_BREAKAndTAB(int TABTimes) {
StringBuffer temp = new StringBuffer();
temp.append(appendLINE_BREAK());
temp.append(appendTAB(TABTimes));
return temp.toString();
} private String appendLINE_BREAK() {
return LINE_BREAK;
} private String appendTAB(int TABTimes) {
StringBuffer temp = new StringBuffer();
for (int i = 0; i < TABTimes; i++) {
temp.append(TAB);
}
return temp.toString();
} public static void main(String[] args) {
String toFormat = "{\"status\": \"3\",\"message\": \"\",\"errCode\": \"0\",\"data\": [{\"time\": \"2013-02-26 16:47\",\"context\": \"客户 同事收发家人 已签收 派件员 张xx\"},{\"time\": \"2013-02-26 07:33\",\"context\": \"吉林省xx市xx公司 的派件员 张金达 派件中 派件员电话15xxx73xx87\"},{\"time\": \"2013-02-26 06:02\",\"context\": \"xx省xx市xx公司 已收入\"},{\"time\": \"2013-02-25 15:42\",\"context\": \"xx省xx转运中心公司已发出\"},{\"time\":\"2013-02-25 14:59\",\"context\":\"xx省xx转运中心公司已拆包\"},{\"time\": \"2013-02-24 18:11\",\"context\":\"辽宁省大连市中山区四部公司 已收件\"},{\"time\": \"2013-02-24 17:59\",\"context\":\"辽宁省大连市公司 已收入\"},{\"time\":\"2013-02-23 17:10\",\"context\":\"辽宁省大连市中山区xxxx公司 的收件员 王xx 已收件\" }],\"html\":\"\",\"mailNo\":\"71xxxxx624\",\"expTextName\":\"圆通快递\",\"expSpellName\":\"yuantong\",\"update\":\"1375155719\",\"cache\":\"33196560\",\"ord\":\"DESC\"}";
JSONFormat jsonFormat = new JSONFormat(toFormat);
System.out.println(jsonFormat.format());
} }
效果图:
{
"status": "3",
"message": "",
"errCode": "0",
"data": [
{
"time": "2013-02-26 16:47",
"context": "客户 同事收发家人 已签收 派件员 张xx"
},
{
"time": "2013-02-26 07:33",
"context": "吉林省xx市xx公司 的派件员 张金达 派件中 派件员电话15xxx73xx87"
},
{
"time": "2013-02-26 06:02",
"context": "xx省xx市xx公司 已收入"
},
{
"time": "2013-02-25 15:42",
"context": "xx省xx转运中心公司已发出"
},
{
"time":"2013-02-25 14:59",
"context":"xx省xx转运中心公司已拆包"
},
{
"time": "2013-02-24 18:11",
"context":"辽宁省大连市中山区四部公司 已收件"
},
{
"time": "2013-02-24 17:59",
"context":"辽宁省大连市公司 已收入"
},
{
"time":"2013-02-23 17:10",
"context":"辽宁省大连市中山区xxxx公司 的收件员 王xx 已收件"
}
],
"html":"",
"mailNo":"71xxxxx624",
"expTextName":"圆通快递",
"expSpellName":"yuantong",
"update":"1375155719",
"cache":"33196560",
"ord":"DESC"
}
如果把这个放在web项目里面,然后别人就可以根据api协议以Get或者Post请求的方式来调用你的json格式化工具啦~~
不足:没有对A问题进行实现,只是简单的借用了fastjson的JSONObject.parseObject()方法。
这样做虽然简单,但是有明显的缺陷:如果某一行不符合json格式要求你只能通过抛出一个异常的方式告知调用者,而不能准确的定位在某一行。
这个格式化工具最难的地方也是在这里,需要用栈来匹配符号,这就要求对json有比较清晰地了解。
大家有好的思路或者想法,欢迎一起交流~
手写一个json格式化 api的更多相关文章
- 『练手』手写一个独立Json算法 JsonHelper
背景: > 一直使用 Newtonsoft.Json.dll 也算挺稳定的. > 但这个框架也挺闹心的: > 1.影响编译失败:https://www.cnblogs.com/zih ...
- 放弃antd table,基于React手写一个虚拟滚动的表格
缘起 标题有点夸张,并不是完全放弃antd-table,毕竟在react的生态圈里,对国人来说,比较好用的PC端组件库,也就antd了.即便经历了2018年圣诞彩蛋事件,antd的使用者也不仅不减,反 ...
- 只会用就out了,手写一个符合规范的Promise
Promise是什么 所谓Promise,简单说就是一个容器,里面保存着某个未来才会结束的事件(通常是一个异步操作)的结果.从语法上说,Promise 是一个对象,从它可以获取异步操作的消息.Prom ...
- 搞定redis面试--Redis的过期策略?手写一个LRU?
1 面试题 Redis的过期策略都有哪些?内存淘汰机制都有哪些?手写一下LRU代码实现? 2 考点分析 1)我往redis里写的数据怎么没了? 我们生产环境的redis怎么经常会丢掉一些数据?写进去了 ...
- 手写一个简单的ElasticSearch SQL转换器(一)
一.前言 之前有个需求,是使ElasticSearch支持使用SQL进行简单查询,较新版本的ES已经支持该特性(不过貌似还是实验性质的?) ,而且git上也有elasticsearch-sql 插件, ...
- 摊牌了!我要手写一个“Spring Boot”
目前的话,已经把 Spring MVC 相关常用的注解比如@GetMapping .@PostMapping .@PathVariable 写完了.我也已经将项目开源出来了,地址:https://gi ...
- 手写一个最迷你的Web服务器
今天我们就仿照Tomcat服务器来手写一个最简单最迷你版的web服务器,仅供学习交流. 1. 在你windows系统盘的F盘下,创建一个文件夹webroot,用来存放前端代码. 2. 代码介绍: ( ...
- 手把手教你手写一个最简单的 Spring Boot Starter
欢迎关注微信公众号:「Java之言」技术文章持续更新,请持续关注...... 第一时间学习最新技术文章 领取最新技术学习资料视频 最新互联网资讯和面试经验 何为 Starter ? 想必大家都使用过 ...
- 教你如何使用Java手写一个基于链表的队列
在上一篇博客[教你如何使用Java手写一个基于数组的队列]中已经介绍了队列,以及Java语言中对队列的实现,对队列不是很了解的可以我上一篇文章.那么,现在就直接进入主题吧. 这篇博客主要讲解的是如何使 ...
随机推荐
- POJ 1061 青蛙的约会 扩展欧几里得
扩展欧几里得模板套一下就A了,不过要注意刚好整除的时候,代码中有注释 #include <iostream> #include <cstdio> #include <cs ...
- C++静态库和动态库的区别
转自http://www.cnblogs.com/skynet/p/3372855.html 什么是库? 库是写好的现有的,成熟的,可以复用的代码.现实中每个程序都要依赖很多基础的底层库,不可能每个人 ...
- PHP storm快捷键
左边文件路径看不到了,按alt+1就出来了 ctrl+j 插入活动代码提示 ctrl+alt+t 当前位置插入环绕代码 alt+insert 生成代码菜单 ctrl+q 查看代码注释 ctrl+d 复 ...
- 自己总结的web前端知识体系大全【欢迎补充】
1. 前言 大约在几个月之前,让我看完了<webkit技术内幕>这本书的时候,突然有了一个想法.想把整个web前端开发所需要的知识都之中在一个视图中,形成一个完整的web前端知识体系,目的 ...
- 【初探Underscore】再说模版引擎
前言 Underscore 是一个JavaScript实用库,提供了类似Prototype.js (或 Ruby)的一些功能,但是没有扩展任何JavaScript内置对象. 它弥补了部分jQuery没 ...
- curl命令学习笔记
下载文件并保存为默认文件名 curl -O http://www.xxx.com/xxx.exe 下载文件并保存为指定文件名 curl -o filename.exe http://www.xxx.c ...
- 如何用RadioButton做一个底部的切换栏
上面的效果是用Radio进行制作的,一般我们做底部的切换栏的时候需要让按钮和文字都有一个选中的状态,然后根据点击不同的按钮触发不同的页面,这里的页面一般都是fragment做的.这里我们不讨论复杂的东 ...
- react native与现有的应用程序集成
(1)通过cocopods 集成 ,以下内容 参考 http://wiki.jikexueyuan.com/project/react-native/integration-existing.html ...
- word第二讲(0806)
word里的长度单位 绝对长度单位(厘米,英寸) 相对长度单位(字符,像素) 样式 问题: 如何设置多个部分的格式 多次选择,多次设置 多次选择,一次设置 应用格式刷(ctrl+shift+c,ctr ...
- Android studio 如何查看当前git 分支的4种方式
1.第一种 2.第二种 3.第三种 4.第四种 前面3种都是通过android studio 操作的. 第四种是通过命令行操作.(可以在 git bash 中输入命 ...