最近写的一个东西需要对json字符串进行格式化然后显示在网页上面。

我就想去网上找找有没有这样的api可以直接调用。百度 json api ,搜索结果都是那种只能在网页上进行校验的工具,没有api。

那只有自己去实现一个json 格式化工具。

仔细分析,实现起来并不是很困难,至少思路很清晰。

需要解决的几个问题:

  1. 对json的校验:主要是符号的匹配;
  2. 格式化预处理:去除键值对之间的空白字符;
  3. 格式化:主要是缩进的问题,要符合json通常展示的格式。

解决的办法:

  1. 针对A问题:

可以采用栈去匹配符号,大括号、中括号、双引号等。

  1. 针对B问题:

可以用fastjson的JSONObject.toString()方法实现,自动去除键值对之间的空白字符。

  1. 针对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的更多相关文章

  1. 『练手』手写一个独立Json算法 JsonHelper

    背景: > 一直使用 Newtonsoft.Json.dll 也算挺稳定的. > 但这个框架也挺闹心的: > 1.影响编译失败:https://www.cnblogs.com/zih ...

  2. 放弃antd table,基于React手写一个虚拟滚动的表格

    缘起 标题有点夸张,并不是完全放弃antd-table,毕竟在react的生态圈里,对国人来说,比较好用的PC端组件库,也就antd了.即便经历了2018年圣诞彩蛋事件,antd的使用者也不仅不减,反 ...

  3. 只会用就out了,手写一个符合规范的Promise

    Promise是什么 所谓Promise,简单说就是一个容器,里面保存着某个未来才会结束的事件(通常是一个异步操作)的结果.从语法上说,Promise 是一个对象,从它可以获取异步操作的消息.Prom ...

  4. 搞定redis面试--Redis的过期策略?手写一个LRU?

    1 面试题 Redis的过期策略都有哪些?内存淘汰机制都有哪些?手写一下LRU代码实现? 2 考点分析 1)我往redis里写的数据怎么没了? 我们生产环境的redis怎么经常会丢掉一些数据?写进去了 ...

  5. 手写一个简单的ElasticSearch SQL转换器(一)

    一.前言 之前有个需求,是使ElasticSearch支持使用SQL进行简单查询,较新版本的ES已经支持该特性(不过貌似还是实验性质的?) ,而且git上也有elasticsearch-sql 插件, ...

  6. 摊牌了!我要手写一个“Spring Boot”

    目前的话,已经把 Spring MVC 相关常用的注解比如@GetMapping .@PostMapping .@PathVariable 写完了.我也已经将项目开源出来了,地址:https://gi ...

  7. 手写一个最迷你的Web服务器

    今天我们就仿照Tomcat服务器来手写一个最简单最迷你版的web服务器,仅供学习交流. 1. 在你windows系统盘的F盘下,创建一个文件夹webroot,用来存放前端代码.  2. 代码介绍: ( ...

  8. 手把手教你手写一个最简单的 Spring Boot Starter

    欢迎关注微信公众号:「Java之言」技术文章持续更新,请持续关注...... 第一时间学习最新技术文章 领取最新技术学习资料视频 最新互联网资讯和面试经验 何为 Starter ? 想必大家都使用过 ...

  9. 教你如何使用Java手写一个基于链表的队列

    在上一篇博客[教你如何使用Java手写一个基于数组的队列]中已经介绍了队列,以及Java语言中对队列的实现,对队列不是很了解的可以我上一篇文章.那么,现在就直接进入主题吧. 这篇博客主要讲解的是如何使 ...

随机推荐

  1. 基于Typecho CMS框架开发大中型应用

    基于Typecho CMS框架开发大中型应用 大中型应用暂且定义为:大于等于3个数据表的应用!汗吧! Typecho原本是一款博客系统,其框架体系有别于市面上一般意义MVC框架,主体代码以自创的Wid ...

  2. 巧用loadrunner代理,录制手机APP脚本

    利用loadrunner代理方式录制手机(iPhone.android)应用程序HTTP脚本 1.利用笔记本网卡或者类似360随身wifi,在安装loadrunner的电脑上共享网络,在手机上连接共享 ...

  3. 七个结构模式之桥接模式(Bridge Pattern)

    问题: 当存在多个独立的变化维度时,如果仍采用多层继承结构,会急剧的增加类的个数,因此可以考虑将各个维度分类,使他们不相互影响. 定义: 将抽象部分与它的实现部分进行分离,抽象部分只保留最为本质的部分 ...

  4. JPA persistence

    Play provides a set of very useful helpers to simplify the management of your JPA entities. Note tha ...

  5. 命令行工具解析Crash文件,dSYM文件进行符号化

    备份   文/爱掏蜂窝的熊(简书作者)原文链接:http://www.jianshu.com/p/0b6f5148dab8著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”. 序 在日常开发 ...

  6. jQuery cxSlide 焦点图轮换

    cxSlide 是一个简单易用的焦点图展示插件,支持水平.纵向切换,透明过渡切换. 已支持 CSS 动画过渡切换.通过 CSS 动画切换,可以展示更多效果. 版本: jQuery v1.7+ jQue ...

  7. Photopile JS – 帮助你实现精致的照片堆叠效果

    Photopile JS 是模拟照片散布堆叠在一起的 JavaScript/jQuery 图片库.点击缩略图,照片会弹出放大 ,再次点击照片会返回.缩略图是可拖动的,允许照片深深的堆在一起而不被覆盖, ...

  8. 一款实用的viewer.js 图片相册

    Viewer.js 是一款强大的图片相册插件,像SNS交友网站一般都会用到点击缩略图,弹出层大图片,而且弹出层有多个控制按钮,比如放大缩小.旋转.撤回等,底部有缩略图列表可切换. 支持移动设备触摸事件 ...

  9. div,span,p等转换成可编辑

    当前它能够将任意不可编辑的标签(span.div.p...等)转换成可编辑的text input.password.textarea.下拉列表(drop-down list)等标签.你可以利用它的ed ...

  10. 退出多个activity的方法

    1.使用List集合方式 用list保存activity实例,然后逐一干掉 import java.util.LinkedList; import java.util.List; import and ...