JSON格式字符串解析思路及基本实现

0,本文主要涉及

使用Java语言将JSON格式字符串解析为Map或List等Java数据结构。

1,一些说明

网上有很多示例,但是有的将词法解析和语法解析同时实现,有的则关注了很多细节的部分(错误处理,转义字符校验等),整体的思路不是很明显,这里提供一版思路清晰,但是略过细节的实现代码及思路。

2,解析思路

假定JSON格式字符串格式无错误
首先解析字符串获取关键字符 { } [ ] : , 以及各种值(字符串,布尔值,null,数值)
然后用递归的方式解析三种不同的数据,JSONObject(通过{识别),JSONArray(通过[识别),以及具体值

3,实现代码

package com.zhangbohun;

import java.io.*;
import java.util.*;

/**
 * @author zhangbohun
 * Create Date 2019/12/20 17:42
 * Modify Date 2019/12/20 17:42
 */
public class JSONParser
{

    private List scan(String str)
    {
        List l = new LinkedList();
        for(int i = 0; i < str.length(); i++)
        {
            switch(str.charAt(i))
            {
                case '{':
                case '}':
                case '[':
                case ']':
                case ',':
                case ':':
                    l.add(str.charAt(i));//关键字符
                    break;
                case 'n':
                    l.add(null);
                    i += 3;
                    break;
                case 't':
                    l.add(true);
                    i += 3;
                    break;
                case 'f':
                    l.add(false);
                    i += 4;
                    break;
                case '"':
                    //获取字符串
                    StringBuilder sb = new StringBuilder();
                    for(int j = i + 1; j < str.length(); j++)
                    {
                        char ch = str.charAt(j);
                        if(ch == '\\')
                        {   // 处理转义字符
                            sb.append('\\');
                            sb.append(str.charAt(j += 1));
                        }
                        else if(ch == '"')
                        {     // 双引号结束
                            l.add(sb.toString());
                            break;
                        }
                        else
                        {
                            sb.append(ch);
                        }
                    }
                    i += sb.length() + 1;
                    break;
                default:
                    //跳过空白字符
                    if(Objects.equals(str.charAt(i), ' ') || Objects.equals(str.charAt(i), '\t') || Objects.equals(str.charAt(i), '\r') || Objects.equals(str.charAt(i), '\n'))
                    {
                        break;
                    }
                    //获取数字(普通数字和科学记数法数字)
                    int j = i;
                    while(Character.isDigit(str.charAt(i)) || Objects.equals(str.charAt(i), '-') || Objects.equals(str.charAt(i), 'e') || Objects.equals(str.charAt(i), 'E') || Objects.equals(str.charAt(i), '+'))
                    {
                        i++;
                    }
                    l.add(Double.parseDouble(str.substring(j, i)));
                    i--;
                    break;
            }
        }
        return l;
    }


    private Object parse(List tokenList)
    {
        if(Objects.equals(tokenList.get(0), '{'))
        {
            tokenList.remove(0);//{
            Object value = parseObject(tokenList);
            tokenList.remove(0);//}
            return value;

        }
        else if(Objects.equals(tokenList.get(0), '['))
        {
            tokenList.remove(0);//[
            Object value = parseArray(tokenList);
            tokenList.remove(0);//]
            return value;
        }
        else
        {
            Object value = tokenList.get(0);
            tokenList.remove(0);
            return value;
        }
    }

    private Map parseObject(List tokenList)
    {
        Map m = new HashMap();
        while(!Objects.equals(tokenList.get(0), '}'))
        {
            Object key = tokenList.get(0);
            tokenList.remove(0);//key
            tokenList.remove(0);//:
            m.put(key, parse(tokenList));
            if(Objects.equals(tokenList.get(0), ','))
            {
                tokenList.remove(0);//,
            }
        }
        return m;
    }

    private List parseArray(List tokenList)
    {
        List l = new ArrayList();
        while(!Objects.equals(tokenList.get(0), ']'))
        {
            l.add(parse(tokenList));
            if(Objects.equals(tokenList.get(0), ','))
            {
                tokenList.remove(0);//,
            }
        }
        return l;
    }


    public Object parseFromString(String jsonString)
    {
        return parse(scan(jsonString));
    }

    public Object parseFromFile(File file) throws IOException
    {
        return parseFromStream(new FileInputStream(file));
    }

    public Object parseFromStream(InputStream inputStream) throws IOException
    {
        StringBuilder sb = new StringBuilder();

        InputStreamReader reader = new InputStreamReader(inputStream);
        BufferedReader br = new BufferedReader(reader);
        String s = br.readLine();
        while(s != null)
        {
            sb.append(s);
            s = br.readLine();
        }

        return parseFromString(sb.toString());
    }
}

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值