解析json字符串有很多方式, 1 : 在网上下载json解析的dll类库并添加引用, 调用相关方法; 2 : 使用自带类库JavaScriptSerializer的序列号和反序列化; 对于以上两个方法我没有试用过, 应该很方便很简洁性能很高吧!

自己根据遍历字符串找json字符串规律, 自己写了一个类库, 只有一个方法只提供解析, 没有其他方法. 缺点 : 可能比较死板, 可能性能也不及网上下载解析类库.

经测试和调试后可以遍历大部分json字符串数据, json字符串可以嵌套, 但要符合json的规律, 数据中不能出现json字符串敏感关键字符 " 和 , 和 [ ] 和 { } ,数据中如果需要使用可以使用中文字符代替.

数据返回结果存放在 Dictionary<string, object> 键 值对中, 如果 值为字符串, 那么object就为字符串,为了嵌套, 如果 值为数组, 那么object就为 List<object> , 如果值为一个对象, 那么object就存放在 Dictionary<string, object> 如此嵌套下去, 最终数据我们根据自己的json数据结构遍历Dictionary<string, object>集合即可. (注 : 每个Dictionary中键必须唯一)

1. 解析类 : AnalyzeJSON 全部代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace System.AnalyzeJSON
{
/// <summary>
/// 对于 JSON数据进行解析
/// Date:2019/6/25
/// Author:weloglog
/// </summary>
public class AnalyzeJSON
{
/// <summary>
/// 最大嵌套深度
/// </summary>
public int MaxNum = ; /// <summary>
/// 解析JSON字符串
/// </summary>
/// <param name="strJSON">JSON字符串</param>
/// <returns>返回Dictionary数据</returns>
public Dictionary<string, object> AnalyzeString(string strJSON)
{
if (strJSON == null || strJSON.Trim() == "" || strJSON.Trim().Length < )
{
return null;
} #region 筛选判断并赋值 [此步骤可以省略]
int idF = -;//第一个 { 下标索引
int idL = -;//最后一个 } 下标索引
int mD = ;//记录 { } 的个数对
int mZ = ;//记录 [ ] 的个数对
for (int i = ; i < strJSON.Length; i++)
{
if (mD > MaxNum || mZ > MaxNum)
{
break;//不满足条件退出循环
}
if (idF == - && strJSON[i] == '{')
{
idF = i;//取第一个 { 下标
}
if (strJSON[i] == '{')
{
mD++;
if (idL > )
{
break;
}
}
if (strJSON[i] == '}')
{
mD--;
if (mD == )
{
idL = i;
}
}
if (strJSON[i] == '[')
{
mZ++;
}
if (strJSON[i] == ']')
{
mZ--;
}
}
if (mD == && mZ == && idL > && idL - idF > )
{
strJSON = strJSON.Substring(idF, idL - idF + );//重新赋值json字符串数据, 去掉{ }前后多余部分
}
else
{
return null;//条件不满足, JSON字符串不规范
}
#endregion //遍历 并返回
return obj(strJSON);
} //遇到 { } 的处理函数
private Dictionary<string, object> obj(string str)
{
Dictionary<string, object> ro = new Dictionary<string, object>(); int dc = ;//{ } 的对数
int len = str.Length;
for (int i = ; i < len; i++)
{
if (str[i] == '{')
{
dc++;
}
if (str[i] == '}')
{
dc--;
}
if (str[i] != '{' && dc > )
{
StringBuilder tem = new StringBuilder();
StringBuilder ojtem = new StringBuilder();
bool isstr = false;
object oj = "";
int c = ;//次数
bool iskey = true;//是否为键赋值
bool isString = true;//值是否为字符串类型
while (i < len && str[i] != ',')
{
if (iskey) //给键 赋值
{
if (str[i] != '\"')
{
if (str[i] == ':')
{
iskey = false;
c = -;//重置
}
else
{
//tem += str[i];
tem.Append(str[i]);
}
}
}
else //给值 赋值
{
//特殊情况, 遇到 { } 和 [ ] 的情况
if (isString && str[i] == '[')//只允许第一次进入
{
isString = false;
int idxs = ;//记录 [ ] 出现的次数
StringBuilder tm = new StringBuilder();
while (i < len)
{
if (str[i] == '[')
{
idxs++;
}
if (str[i] == ']')
{
idxs--;
}
tm.Append(str[i]);
i++;
if (idxs == )//变成一个完整的组合
{
break;
}
}
oj = arr(tm.ToString());
break;
}
else if (isString && str[i] == '{')//只允许第一次进入
{
isString = false;
int idxs = ;//记录 { } 出现的次数
StringBuilder tm = new StringBuilder();
while (i < len)
{
if (str[i] == '{')
{
idxs++;
}
if (str[i] == '}')
{
idxs--;
}
tm.Append(str[i]);
i++;
if (idxs == )//变成一个完整的组合
{
break;
}
}
oj = obj(tm.ToString());
break;
}
else
{
if (str[i] != '\"')
{
if (str[i] == ',' || str[i] == '}' || str[i] == ']')//跳出循环
{
break;
}
else
{
isstr = true;
ojtem.Append(str[i]);
}
}
}
}
c++;
i++;
}
c = ;
try//键 唯一
{
if (tem != null && tem.ToString().Length > )
{
if (isstr)
{
ro.Add(tem.ToString(), ojtem);//添加
isstr = false;
}
else
{
ro.Add(tem.ToString(), oj);//添加
}
}
}
catch { }
}
}
return ro;
} //遇到 [ ] 的处理函数
private object arr(string str)
{
object ojj = new object();
//去掉首位 [ ] 符号
str = str.Substring(, str.Length - );
int len = str.Length;
int c = ;//双引号索引
List<object> lst = new List<object>();
bool ists = false;//是否为特殊 for (int i = ; i < len; i++)
{
object tem = "";
StringBuilder sb = new StringBuilder();
bool isstr = false;
while (i < len)
{
if (str[i] == '[')//特殊处理
{
int idxs = ;//记录 [ ] 出现的次数
StringBuilder tm = new StringBuilder();
while (i < len)
{
if (str[i] == '[')
{
idxs++;
}
if (str[i] == ']')
{
idxs--;
}
tm.Append(str[i]);
i++;
if (idxs == )//变成一个完整的组合
{
break;
}
}
lst.Add(arr(tm.ToString()));
ists = true;
i++;
continue;
}
else if (str[i] == '{')//特殊处理
{
int idxs = ;//记录 [ ] 出现的次数
StringBuilder tm = new StringBuilder();
while (i < len)
{
if (str[i] == '{')
{
idxs++;
}
if (str[i] == '}')
{
idxs--;
}
tm.Append(str[i]);
i++;
if (idxs == )//变成一个完整的组合
{
break;
}
}
lst.Add(obj(tm.ToString()));
ists = true;
i++;
continue;
}
else
{
ists = false;
if (c == && str[i] == '\"')
{
i++;
c++;
continue;
}
if (str[i] == '\"' && i + < len && str[i + ] == ',' || i + == len)
{
i++;
c++;
break;
}
if (str[i] == '\"' && i + < len && str[i + ] == ']' || i + == len)
{
i++;
c++;
continue;
}
if (i + < len && str[i + ] == ']')
{
i++;
c++;
continue;
}
isstr = true;
sb.Append(str[i]);
i++;
c++;
}
}
if (!ists)
{
if (isstr)
{
lst.Add(sb);// [ ] 的值存入List<string> 中
isstr = false;
}
else
{
lst.Add(tem);// [ ] 的值存入List<string> 中
}
}
c = ;//归零
}
ojj = lst;
return ojj;
}
}
}

AnalyzeJSON 类

2. 方法的调用和数据的使用

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.AnalyzeJSON;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.IO;
using System.Text.RegularExpressions; namespace WindowsFormsApplication1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
} //按钮点击事件
private void button1_Click(object sender, EventArgs e)
{
//
//窗体中一个 TextBox 文本框[多行] 和 一个 Button 测试按钮
// //
//在网上随便找了一个json数据字符串的js
//[在这里感谢'4399游戏资讯'平台提供的这些数据供测试,该js只供学习不可用作商业用途]
//
//该js是一个游戏类英雄相关属性的数据
//js共4行, 每一行数据也非常大, 我们使用第一行进行测试,每一行使用回车键[\n]分割
//
string src = "//newsimg.5054399.com/dtzzq/static/zrmnq/wap/js/data.js";
StreamReader reader = new StreamReader(System.Net.WebRequest.Create("http:" + src).GetResponse().GetResponseStream());
//Regex.Unescape("") 方法是将字符串含有 \uxxxx 的16进制转化为我们识别的字符
string[] zongstrs = Regex.Unescape(reader.ReadToEnd()).Split('\n'); //创建一个解析对象
AnalyzeJSON aj = new AnalyzeJSON(); //调用方法 AnalyzeString("JSON数据字符串") 进行解析 并返回 解析后的数据集合Dictionary<string, object>
Dictionary<string, object> obj = aj.AnalyzeString(zongstrs[]); //定义一个字符串进行拼接显示得到的数据
StringBuilder sb = new StringBuilder();
//调用拼接处理方法
zx(obj, sb); //把得到的数据以字符串的形式展示出来
textBox1.Text = sb.ToString();
} //
//数据遍历解析方法
//根据需要可以为自己定义数据处理赋值方法,此处只作为显示使用
//
private void zx(Dictionary<string, object> obj, StringBuilder sb)
{
foreach (var item in obj)
{
if ((item.Value as Dictionary<string, object>) != null)
{
zx((item.Value as Dictionary<string, object>), sb);
}
else
{
if ((item.Value as List<object>) != null && (item.Value as List<object>).Count > )
{
List<object> lst = item.Value as List<object>;
sb.Append(item.Key + ":\r\n");
for (int i = ; i < lst.Count; i++)
{
if ((lst[i] as Dictionary<string, object>) != null)
{
zx((lst[i] as Dictionary<string, object>), sb);
}
else
{
sb.Append("\t" + lst[i] + ",");
} }
sb.Append("\r\n");
}
else
{
sb.Append(item.Key + ":" + item.Value.ToString() + "\r\n");
}
}
}
} }
}

页面展示代码

代码可能还有很多需要改进的地方, 希望各位大神指出来, 共同学习进步!^_^

c# 自定义解析JSON字符串数据的更多相关文章

  1. C# 解析JSON格式数据

    JSON简介 JSON(全称为JavaScript ObjectNotation) 是一种轻量级的数据交换格式.它是基于JavaScript语法标准的一个子集.JSON采用完全独立于语言的文本格式,可 ...

  2. 【转】C# 解析JSON格式数据

    http://blog.csdn.net/coolszy/article/details/8606803 JSON简介 JSON(全称为JavaScript ObjectNotation) 是一种轻量 ...

  3. C#解析JSON字符串总结

    JSON文件读取到内存中就是字符串,.NET操作JSON就是生成与解析JSON字符串. 操作JSON通常有以下几种方式: 1. 原始方式:按照JSON字符串自己来解析. 2. 通用方式[★★★★★]: ...

  4. Json转model对象,model转json,解析json字符串

    GitHub链接: https://github.com/mozhenhau/D3Json D3Json 通过swift的反射特性,把json数据转换为model对象,本类最主要是解决了其他一般jso ...

  5. C#解析JSON字符串总结(转载)

    JSON文件读取到内存中就是字符串,.NET操作JSON就是生成与解析JSON字符串. 操作JSON通常有以下几种方式: 1. 原始方式:按照JSON字符串自己来解析. 2. 通用方式[★★★★★]: ...

  6. 使用 dynamic 标记解析JSON字符串 JDynamic :支持Json反序列化为Dynamic对象

    使用 dynamic 标记解析JSON字符串  http://www.cnblogs.com/taotaodetuer/p/4171327.html 1 string jsonStr = " ...

  7. 使用 dynamic 标记解析JSON字符串

    string jsonStr = "{\"data\": {\"ssoToken\": \"70abd3d8a6654ff189c482fc ...

  8. Objective-C-使用NSMutableURLRequest发送POST请求,使用NSJSONSerialization解析JSON字符串

    NSString *reqData = @"Data="; NSData *postDatas = nil; NSString *urlPath = @"url" ...

  9. Objective-C——NSMutableURLRequest发送POST请求,使用NSJSONSerialization解析JSON字符串

    NSString *reqData = @"Data="; NSData *postDatas = nil; NSString *urlPath = @"url" ...

随机推荐

  1. ios TCP抓包

    1.工具安装 Mac15.1 ,Xcode11.2.1 这一步有个小坑,我第一次执行rvictls -s 提示 rvictl: command not found.Stack Overflow上提供了 ...

  2. mysqldump定时任务生成备份文件内容为空解决方法

    1问题:写好了一个mysqldump备份脚本(如图)直接执行可以正常生成备份文件,但在用crontab运行时却生成内容为空 2原因分析:由于mysqldump存在于全局环境变量mysql的bin下面, ...

  3. 前端知识点回顾——koa和模板引擎

    koa 基于Node.js的web框架,koa1只兼容ES5,koa2兼容ES6及以后. const Koa = requier("koa"); const koa = new K ...

  4. python中的cls到底指的是什么

    python中的cls到底指的是什么,与self有什么区别? 2018年07月31日 11:13:09 rs勿忘初心 阅读数:7769   作者:秦风链接:https://www.zhihu.com/ ...

  5. shell编程(2)

    一.变量简介 变量是任何一种编程语言都必不可少的组成部分,变量用来存放各种数据.脚本语言在定义变量时通常不需要指明类型,直接赋值即可,shell变量也遵循这个规则. 在Bash shell 中,每一个 ...

  6. prometheus监控插件mysqld_exporter

    1,首先需要增加授权 mysql> GRANT PROCESS, REPLICATION CLIENT, SELECT ON *.* TO 'exporter'@'localhost' IDEN ...

  7. 本地文件上传到Linux服务器

    1.从服务器上下载文件scp username@servername:/path/filename /var/www/local_dir(本地目录) 例如scp root@192.168.0.101: ...

  8. IDEA在引入Maven项目后Dependencies中在出现红色波浪线

    解决方法: 移除pom.xml中相关依赖,再重新添加即可. 情况及具体解决方法如下: 1.在Maven Project中 Dependencies 出现红色波浪线,如图所示 2.查询本地仓库:jar包 ...

  9. Cannon 60D 电池卡在电池槽了,拔不出来怎么办?

    事情是这样的,本来好好的电池在电池槽里的,后来拿去充电了,充满后就准备装回去,然后一个不小心,电池掉地上了,就看了一下没摔爆,所以也没特别留意有没有什么地方摔坏摔瘸角,然后就往相机里塞,突然就发现塞不 ...

  10. imregionalmax imregionalmin imextendedmax imextendedmin imhmax imhmin 函数的详解 matlab中函数

    BW = imregionalmax(I): 该函数获得灰度图像 I 的局部极大值,返回值BW为和原图像大小相同的二值图像,BW中元素1对应极大值,其他元素为0 BW = imregionalmax( ...