我有一堆自动生成的JSON文件,我想在版本控制中存储。问题是每次文件被序列化时,属性都会以不同的顺序出现,这使得很难知道文件是否真的发生了变化和/或真正的差异是什么。
是否有人知道将执行此任务的现有开源工具?
如果失败了,有没有人知道一个带有解析器和生成器的JSON库,可以配置为输出具有(比如)词法顺序属性的“漂亮”JSON? (Java或Ruby库是理想的,但也欢迎其他潜在客户。)
答案 0 :(得分:15)
Python's JSON module在其他程序中非常有用:
generate_json | python -mjson.tool > canonical.json
答案 1 :(得分:4)
如果您愿意通过致电
来承担一些开销gson.toJson(canonicalize(gson.toJsonTree(obj)));
然后你可以这样做:
protected static JsonElement canonicalize(JsonElement src) {
if (src instanceof JsonArray) {
// Canonicalize each element of the array
JsonArray srcArray = (JsonArray)src;
JsonArray result = new JsonArray();
for (int i = 0; i < srcArray.size(); i++) {
result.add(canonicalize(srcArray.get(i)));
}
return result;
} else if (src instanceof JsonObject) {
// Sort the attributes by name, and the canonicalize each element of the object
JsonObject srcObject = (JsonObject)src;
JsonObject result = new JsonObject();
TreeSet<String> attributes = new TreeSet<>();
for (Map.Entry<String, JsonElement> entry : srcObject.entrySet()) {
attributes.add(entry.getKey());
}
for (String attribute : attributes) {
result.add(attribute, canonicalize(srcObject.get(attribute)));
}
return result;
} else {
return src;
}
}
答案 2 :(得分:1)
杰克逊支持这一点:
@JsonPropertyOrder(字母=真)
答案 3 :(得分:0)
我没有尝试过很多组合,但似乎google-gson保留了JSON
中属性的顺序。
删除了一个示例,因为它不再相关
我从之前项目的经验中了解到它是非常可定制的,例如,如果base object不够,可以使用GsonBuilder来创建更复杂的适配器。
我没有对你的用例进行过广泛的测试,但检查它是否具有预期的输出应该很简单
<强>更新强>
而不是使用SVN / CVS检查您的文件是否被修改,我发现GSON内置versioning support可能会或可能不会解决您的问题,来自他们的文档:
使用@Since注释可以维护同一对象的多个版本。此批注可用于类,字段以及将来的发行版中的方法。要利用此功能,必须将Gson实例配置为忽略任何大于某个版本号的字段/对象。如果没有在Gson实例上设置任何版本,那么无论版本如何,它都将序列化和反序列化所有字段和类。
<强>更新强>
我唯一能想到的是用rhino解析外部文件并使用
JSON.stringify
将解析后的JSON
转换回字符串,然后你可以确定它已经运行单个“解析器”,输出不会有所不同。
然后,您可以检测到任何可能的更改。
答案 4 :(得分:0)
开源Java库Jackson可能需要花费一些精力进行设置,但能够进行漂亮打印并且具有非常整洁的@JsonPropertyOrder
注释,支持按字母或手动指定的输出顺序。
答案 5 :(得分:0)
Ruby 1.9+维护哈希的插入顺序,而JSON则保持1.9+的荣誉。
asdf = {'a' => 1, 'b' => 2}
asdf.to_json # => "{\"a\":1,\"b\":2}"
asdf = {'b' => 1, 'a' => 2}
asdf.to_json # => "{\"b\":1,\"a\":2}"
以下是如何生成“漂亮”格式:
asdf = {'a' => 1, 'b' => 2}
puts JSON.pretty_generate(asdf)
{
"a": 1,
"b": 2
}
asdf = {'b' => 1, 'a' => 2}
irb(main):022:0> puts JSON.pretty_generate(asdf)
{
"b": 1,
"a": 2
}
...以不同的顺序插入相同的属性......
这对我来说没有多大意义,但我会去拍。
因为Ruby维护了插入顺序,所以如果按给定顺序创建哈希,数据的顺序并不重要;通过对键进行排序并重新生成哈希来强制顺序,并将其传递给JSON:
require 'json'
puts Hash[{'a' => 1, 'b' => 2}.sort_by{ |a| a }].to_json
=> {"a":1,"b":2}
puts Hash[{'b' => 2, 'a' => 1}.sort_by{ |a| a }].to_json
=> {"a":1,"b":2}
puts Hash[{'b' => 2, 'c' => 3, 'a' => 1}.sort_by{ |a| a }].to_json
=> {"a":1,"b":2,"c":3}
puts Hash[{'b' => 2, 'c' => 3, 'a' => 1}.sort_by{ |a| a }].to_json
=> {"a":1,"b":2,"c":3}
puts Hash[{'a' => 1, 'c' => 3, 'b' => 2}.sort_by{ |a| a }].to_json
=> {"a":1,"b":2,"c":3}
答案 6 :(得分:0)
这是Qt中的一个简单的JSON编码器 - 应该相对容易重新编写为Java。您真正需要做的就是确保在写出时对键进行排序 - 可以使用其他JSON包读取。
QString QvJson::encodeJson(const QVariant& jsonObject) {
QVariant::Type type = jsonObject.type();
switch (type) {
case QVariant::Map:
return encodeObject(jsonObject);
case QVariant::List:
return encodeArray(jsonObject);
case QVariant::String:
return encodeString(jsonObject);
case QVariant::Int:
case QVariant::Double:
return encodeNumeric(jsonObject);
case QVariant::Bool:
return encodeBool(jsonObject);
case QVariant::Invalid:
return encodeNull(jsonObject);
default:
return encodingError("encodeJson", jsonObject, ErrorUnrecognizedObject);
}
}
QString QvJson::encodeObject(const QVariant& jsonObject) {
QString result("{ ");
QMap<QString, QVariant> map = jsonObject.toMap();
QMapIterator<QString, QVariant> i(map);
while (i.hasNext()) {
i.next();
result.append(encodeString(i.key()));
result.append(" : ");
result.append(encodeJson(i.value()));
if (i.hasNext()) {
result.append(", ");
}
}
result.append(" }");
return result;
}
QString QvJson::encodeArray(const QVariant& jsonObject) {
QString result("[ ");
QList<QVariant> list = jsonObject.toList();
for (int i = 0; i < list.count(); i++) {
result.append(encodeJson(list.at(i)));
if (i+1 < list.count()) {
result.append(", ");
}
}
result.append(" ]");
return result;
}
QString QvJson::encodeString(const QVariant &jsonObject) {
return encodeString(jsonObject.toString());
}
QString QvJson::encodeString(const QString& value) {
QString result = "\"";
for (int i = 0; i < value.count(); i++) {
ushort chr = value.at(i).unicode();
if (chr < 32) {
switch (chr) {
case '\b':
result.append("\\b");
break;
case '\f':
result.append("\\f");
break;
case '\n':
result.append("\\n");
break;
case '\r':
result.append("\\r");
break;
case '\t':
result.append("\\t");
break;
default:
result.append("\\u");
result.append(QString::number(chr, 16).rightJustified(4, '0'));
} // End switch
}
else if (chr > 255) {
result.append("\\u");
result.append(QString::number(chr, 16).rightJustified(4, '0'));
}
else {
result.append(value.at(i));
}
}
result.append('"');
QString displayResult = result; // For debug, since "result" often doesn't show
Q_UNUSED(displayResult);
return result;
}
QString QvJson::encodeNumeric(const QVariant& jsonObject) {
return jsonObject.toString();
}
QString QvJson::encodeBool(const QVariant& jsonObject) {
return jsonObject.toString();
}
QString QvJson::encodeNull(const QVariant& jsonObject) {
return "null";
}
QString QvJson::encodingError(const QString& method, const QVariant& jsonObject, Error error) {
QString text;
switch (error) {
case ErrorUnrecognizedObject:
text = QObject::tr("Unrecognized object type");
break;
default:
Q_ASSERT(false);
}
return QObject::tr("*** Error %1 in QvJson::%2 -- %3").arg(error).arg(method).arg(text);
}
答案 7 :(得分:0)
在输出之前对要序列化的对象的键进行排序。在Ruby 1.9中默认排序哈希;在Ruby 1.8中他们不是。您可以在active_support中使用OrderedHash以确保无论哪种情况。
每当您要编写JSON数据时,请对键进行排序。请注意,在Ruby 1.8中,符号无法排序,因此您必须在排序中调用to_s
。
require 'rubygems'
require 'json'
require 'active_support/ordered_hash'
obj = {
:fig => false,
:bananas => false,
:apples => true,
:eggplant => true,
:cantaloupe => true,
:dragonfruit => false
}
def sorted_hash(hsh)
sorted_keys = hsh.keys.sort_by { |k| k.to_s }
sorted_keys.inject(ActiveSupport::OrderedHash.new) do |o_hsh, k|
o_hsh[k] = hsh[k]
o_hsh
end
end
puts JSON.pretty_generate(obj)
# Could output in any order, depending on version of Ruby
# {
# "eggplant": true,
# "cantaloupe": true,
# "dragonfruit": false,
# "fig": false,
# "bananas": false,
# "apples": true
# }
puts JSON.pretty_generate(sorted_hash(obj))
# Always output in the same order
# {
# "apples": true,
# "bananas": false,
# "cantaloupe": true,
# "dragonfruit": false,
# "eggplant": true,
# "fig": false
# }
如果您的数据由对象数组或嵌套对象组成,则需要递归创建已排序的哈希值:
nested_obj = {:a => {:d => true, :b => false}, :e => {:k => false, :f => true}, :c => {:z => false, :o => true}}
def recursive_sorted_hash(hsh)
sorted_keys = hsh.keys.sort_by { |k| k.to_s }
sorted_keys.inject(ActiveSupport::OrderedHash.new) do |o_hsh, k|
o_hsh[k] = hsh[k].is_a?(Hash) ? recursive_sorted_hash(hsh[k]) : hsh[k]
o_hsh
end
end
puts JSON.pretty_generate(nested_obj)
# Again, could be in any order
# {
# "a": {
# "b": false,
# "d": true
# },
# "e": {
# "f": true,
# "k": false
# },
# "c": {
# "z": false,
# "o": true
# }
# }
puts JSON.pretty_generate(recursive_sorted_hash(nested_obj))
# Even nested hashes are in alphabetical order
# {
# "a": {
# "b": false,
# "d": true
# },
# "c": {
# "o": true,
# "z": false
# },
# "e": {
# "f": true,
# "k": false
# }
# }