最近写的一个东西需要对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. IOS 2D游戏开发框架 SpriteKit-->续(postion,锚点,游戏滚动场景)

    一. Postion 这里的postion和app开发中的layer.postion是一样的,postion的值代表该视图在父节点中的相对位置, 比如一个试图的父节点是self.view, 这里我们想 ...

  2. NYOJ 737---石子归并(GarsiaWachs算法)

    原题链接 描述    有N堆石子排成一排,每堆石子有一定的数量.现要将N堆石子并成为一堆.合并的过程只能每次将相邻的两堆石子堆成一堆,每次合并花费的代价为这两堆石子的和,经过N-1次合并后成为一堆.求 ...

  3. Java--Exchanger用于进行线程间的数据交换

    package com; import java.util.concurrent.Exchanger; /** * Created by yangyu on 16/11/28. */ /** * Ex ...

  4. 从源码角度理清memcache缓存服务

    memcache作为缓存服务器,用来提高性能,大部分互联网公司都在使用.   前言    文章的阅读的对象是中高级开发人员.系统架构师. 本篇文章,不是侧重对memcache的基础知识的总结,比如se ...

  5. LR--Controller的Pacing设置(不容忽视的设置)

    运行时的Pacing设置主要影响什么?     Pacing主要用来设置重复迭代脚本的间隔时间.共有三种方法: A:上次迭代结束后立刻开始. B:上次迭代结束后等待固定时间. C:按固定或随机的时间间 ...

  6. PDF.NET框架学习篇之SQL-MAP使用存储过程

    最近一直在学习“深蓝医生”的PDF.NET框架,对Sql-Map使用存储过程有了点小小的体会.基础知识请到http://www.cnblogs.com/bluedoctor/archive/2010/ ...

  7. 决战大数据之二:CentOS 7 最新JDK 8安装

    决战大数据之二:CentOS 7 最新JDK 8安装 [TOC] 修改hostname # hostnamectl set-hostname node1 --static # reboot now 重 ...

  8. jQuery实用小技巧-获取选中的的下拉框和返回头部滑动动画

    //获取选中的下拉框 $('#someElement').find('option:selected'); $('#someElement option:selected'); //返回头部滑动动画 ...

  9. android listview多视图嵌套多视图

    笔记,listview视图总结 public class HomeEduMoreAdapter extends BaseAdapter { private final String TAG = &qu ...

  10. 操作系统开发系列—11.ELF格式 ●

    ELF文件的结构如下图所示: ELF文件由4部分组成,分别是ELF头(ELF header).程序头表(Program header table).节(Sections)和节头表(Section he ...