json字符串很有用,有时候一些后台接口返回的信息是字符串格式的,可读性很差,这个时候要是有个可以格式化并高亮显示json串的方法那就好多了,下面看看一个正则表达式完成的json字符串的格式化与高亮显示
首先是对输入进行转换,如果是对象则转化为规范的json字符串,不是对象时,先将字符串转化为对象(防止不规范的字符串),然后再次转化为json串。其中json为输入。
if (typeof json !== 'string') {
json = JSON.stringify(json);
} else {
json = JSON.parse(json);
json = JSON.stringify(json);
}
等规范完数据之后对字符串进行标记,为了后面的切分、重新组合
这里有几个地方要添加标记,包括大括号、小括号的前后和逗号的后面都要添加标记,我这里使用的是换行
\n(这样在命令行下测试时效果会比较好看)。
// 在大括号前后添加换行
reg = /([\{\}])/g;
json = json.replace(reg, '\r\n$1\r\n');
// 中括号前后添加换行
reg = /([\[\]])/g;
json = json.replace(reg, '\r\n$1\r\n');
// 逗号后面添加换行
reg = /(\,)/g;
json = json.replace(reg, '$1\r\n');
添加完成标记之后就要做一些优化处理,去掉多余的换行、去掉逗号前面的换行,这样做是为了在切分是免得出现空串浪费一次循环处理,最后在冒号后面添加空格,看起来更漂亮。
// 去除多余的换行
reg = /(\r\n\r\n)/g;
json = json.replace(reg, '\r\n');
// 逗号前面的换行去掉
reg = /\r\n\,/g;
json = json.replace(reg, ',');
//冒号前面缩进
reg = /\:/g;
json = json.replace(reg, ': ');
接下来就是对这个初步处理过的串进行进一步处理了,我会在function(index, node) {}函数中添加逻辑,对每一个切分单元进行处理,包括缩进和美化格式。
$.each(json.split('\r\n'), function(index, node) {});
首先说下缩进,缩进的方法很简单,遇到{、[符号时缩进增加1,遇到}、]符号时缩进减少1,否则缩进量不变。
//这里遇到{、[时缩进等级加1,遇到}、]时缩进等级减1,没遇到时缩进等级不变
if (node.match(/\{$/) || node.match(/\[$/)) {
indent = 1;
} else if (node.match(/\}/) || node.match(/\]/)) {
if (pad !== 0) {
pad -= 1;
}
} else {
indent = 0;
}
完成缩进后就该美化高亮显示代码了,这里要用到几个css规则,下面可以看到,对切分单元进行高亮显示的时候这里用正则进行判断,如果匹配到大括号标记为对象class、中括号标记为数组class、属性名称、属性值,一次对这些进行css规则添加,添加完成之后拼接起来就可以了。
.ObjectBrace{color:#00AA00;font-weight:bold;}
.ArrayBrace{color:#0033FF;font-weight:bold;}
.PropertyName{color:#CC0000;font-weight:bold;}
.String{color:#007777;}
.Number{color:#AA00AA;}
.Comma{color:#000000;font-weight:bold;}
//添加代码高亮
node = node.replace(/([\{\}])/g,"$1");
node = node.replace(/([\[\]])/g,"$1");
node = node.replace(/(\".*\")(\:)(.*)(\,)?/g,"$1$2$3$4");
node = node.replace(/\"([^"]*)\"(\,)?$/g,"\"$1\"$2");
node = node.replace(/(-?\d+)(\,)?$/g,"$1$2");
最后我们看看完整的方法代码(这里我使用了jquery类库),以及测试地址:
要对jsonstr进行美化,这样就可以了APP.format(jsonstr),直接输出至
标签中就可以看到效果,
下面是一个测试地址,http://iforever.sinaapp.com/ 可以进去试一下,看看完整的源代码
<script><br />
var APP=function(){<br />
var format=function(json){<br />
var reg=null,<br />
result='';<br />
pad=0,<br />
PADDING=' ';<br />
if (typeof json !== 'string') {<br />
json = JSON.stringify(json);<br />
} else {<br />
json = JSON.parse(json);<br />
json = JSON.stringify(json);<br />
}<br />
// 在大括号前后添加换行<br />
reg = /([\{\}])/g;<br />
json = json.replace(reg, '\r\n$1\r\n');<br />
// 中括号前后添加换行<br />
reg = /([\[\]])/g;<br />
json = json.replace(reg, '\r\n$1\r\n');<br />
// 逗号后面添加换行<br />
reg = /(\,)/g;<br />
json = json.replace(reg, '$1\r\n');<br />
// 去除多余的换行<br />
reg = /(\r\n\r\n)/g;<br />
json = json.replace(reg, '\r\n');<br />
// 逗号前面的换行去掉<br />
reg = /\r\n\,/g;<br />
json = json.replace(reg, ',');<br />
//冒号前面缩进<br />
reg = /\:/g;<br />
json = json.replace(reg, ': ');<br />
//对json按照换行进行切分然后处理每一个小块<br />
$.each(json.split('\r\n'), function(index, node) {<br />
var i = 0,<br />
indent = 0,<br />
padding = '';<br />
//这里遇到{、[时缩进等级加1,遇到}、]时缩进等级减1,没遇到时缩进等级不变<br />
if (node.match(/\{$/) || node.match(/\[$/)) {<br />
indent = 1;<br />
} else if (node.match(/\}/) || node.match(/\]/)) {<br />
if (pad !== 0) {<br />
pad -= 1;<br />
}<br />
} else {<br />
indent = 0;<br />
}<br />
//padding保存实际的缩进<br />
for (i = 0; i < pad; i++) {<br />
padding += PADDING;<br />
}<br />
//添加代码高亮<br />
node = node.replace(/([\{\}])/g,"<span class='ObjectBrace'>$1");<br />
node = node.replace(/([\[\]])/g,"<span class='ArrayBrace'>$1");<br />
node = node.replace(/(\".*\")(\:)(.*)(\,)?/g,"<span class='PropertyName'>$1$2$3$4");<br />
node = node.replace(/\"([^"]*)\"(\,)?$/g,"<span class='String'>\"$1\"<span class='Comma'>$2");<br />
node = node.replace(/(-?\d+)(\,)?$/g,"<span class='Number'>$1<span class='Comma'>$2");<br />
result += padding + node + '<br>';<br />
pad += indent;<br />
});<br />
return result;<br />
};<br />
return {<br />
"format":format,<br />
};<br />
}();<br />
</script>
怎么样,json字符串是不是美观了很多呢,超级实用吧,这么好的东东,当然不能独享,这里推荐给小伙伴们。