
本文旨在深入探讨在groovy中处理json数据时,如何安全地删除匹配条件的元素并更新文件,同时避免常见的`concurrentmodificationexception`。我们将介绍两种核心策略:通过`findall`方法生成一个过滤后的新json对象,以及利用迭代器(iterator)在遍历过程中安全地移除现有对象中的元素。文章将提供详细的代码示例和最佳实践,帮助开发者高效、稳定地管理json数据。
在Groovy中,处理JSON数据是常见的任务。当我们需要根据特定条件删除JSON对象中的元素时,直接在迭代过程中修改原始集合(如Map)往往会导致java.util.ConcurrentModificationException。这是因为当一个集合正在被迭代器遍历时,如果其底层结构被直接修改(例如添加或删除元素),迭代器会检测到这种并发修改并抛出异常,以防止不确定的行为。
本教程将提供两种健壮且Groovy风格的解决方案来解决这个问题,并演示如何将修改后的JSON数据写回文件。
ConcurrentModificationException通常发生在以下场景:你正在使用迭代器(隐式或显式)遍历一个集合,同时又通过集合本身的add()、remove()等方法修改这个集合。Groovy的each闭包在内部也使用了迭代器,因此直接在each循环中调用json.remove(key)会触发此异常。
例如,以下代码片段会抛出异常:
import groovy.json.*
def jsonSlurper = new JsonSlurper()
def jsonContent = '''
{
"stack-0": {
"name": "foo-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "foo-DB"
},
"stack-1": {
"name": "bar-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "bar-DB"
}
}'''
def json = jsonSlurper.parseText(jsonContent)
// 错误示例:直接在each循环中修改json对象
json.each { key, val ->
if ("foo-web" == val.name) {
json.remove(key) // 这将导致ConcurrentModificationException
}
}这是推荐的Groovy风格解决方案之一。它通过findAll方法创建一个新的Map(或List),其中只包含满足指定条件的元素。这种方法不会修改原始对象,因此是完全安全的。
使用Map.findAll()方法遍历原始JSON对象(在Groovy中解析的JSON对象通常是Map类型),并根据条件筛选出需要保留的元素。findAll会返回一个新的Map,其中包含所有满足闭包条件的键值对。
import groovy.json.*
// 模拟从文件加载JSON内容
def jsonContent = '''\
{
"stack-0": {
"name": "foo-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "foo-DB"
},
"stack-1": {
"name": "bar-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "bar-DB"
},
"stack-2": {
"name": "foo-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "foo-DB"
}
}'''
def originalJson = new JsonSlurper().parseText(jsonContent)
println "原始JSON: ${originalJson}"
// 创建一个过滤后的新Map,保留所有name不等于"foo-web"的元素
def filteredJson = originalJson.findAll { key, val ->
'foo-web' != val.name
}
println "过滤后的JSON: ${filteredJson}"
// 预期输出:
// 过滤后的JSON: [stack-1:[name:bar-web, createdAt:2022-11-30T10:56:32.551977633Z, dbName:bar-DB]]
assert filteredJson.toString() == '[stack-1:[name:bar-web, createdAt:2022-11-30T10:56:32.551977633Z, dbName:bar-DB]]'
// 此时 originalJson 仍然保持不变
println "原始JSON(未改变): ${originalJson}"如果出于性能考虑(例如处理非常大的JSON数据,避免创建新对象带来的内存开销),或者你必须修改原始JSON对象的引用,那么使用迭代器是正确的选择。
获取JSON对象(Map)的迭代器,然后通过迭代器提供的remove()方法来删除元素。Iterator.remove()是唯一在迭代过程中安全修改集合的方法。
import groovy.json.*
// 模拟从文件加载JSON内容
def jsonContent = '''\
{
"stack-0": {
"name": "foo-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "foo-DB"
},
"stack-1": {
"name": "bar-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "bar-DB"
},
"stack-2": {
"name": "foo-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "foo-DB"
}
}'''
def mutableJson = new JsonSlurper().parseText(jsonContent)
println "原始JSON (可变): ${mutableJson}"
// 使用迭代器安全地修改现有json对象
def entry
for (Iterator iter = mutableJson.entrySet().iterator(); iter.hasNext();) {
entry = iter.next()
if ('foo-web' == entry.value.name) {
iter.remove() // 使用迭代器自身的remove方法
}
}
println "修改后的JSON (原地修改): ${mutableJson}"
// 预期输出:
// 修改后的JSON (原地修改): {stack-1={name=bar-web, createdAt=2022-11-30T10:56:32.551977633Z, dbName=bar-DB}}
assert mutableJson.toString() == '{stack-1={name=bar-web, createdAt=2022-11-30T10:56:32.551977633Z, dbName=bar-DB}}'无论你选择了哪种修改JSON数据的方法,最终都需要将结果写回文件。这可以通过JsonBuilder和File.write()方法实现。
使用JsonBuilder将Groovy对象(Map)转换回JSON格式的字符串,然后使用File.write()方法将字符串写入文件。为了保持可读性,通常会使用toPrettyString()方法。
结合上述两种解决方案中的任意一种,假设我们已经得到了filteredJson或mutableJson。
import groovy.json.*
// 假设 filteredJson 是通过 findAll 得到的
// 或者 mutableJson 是通过 Iterator 修改的
def finalJsonData = [
"stack-1": [
"name": "bar-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "bar-DB"
]
] as Map // 示例数据,实际应为 filteredJson 或 mutableJson
// 将Groovy Map转换为格式化的JSON字符串
def newJsonString = new JsonBuilder(finalJsonData).toPrettyString()
// 定义文件路径
def filePath = "sarva.json"
def outputFile = new File(filePath)
// 将JSON字符串写入文件
outputFile.write(newJsonString)
println "已将更新后的JSON数据写入 ${filePath}。"
println "文件内容:\n${outputFile.text}"
// 验证文件内容
assert outputFile.text.trim() == '''{
"stack-1": {
"name": "bar-web",
"createdAt": "2022-11-30T10:56:32.551977633Z",
"dbName": "bar-DB"
}
}'''.trim()
// 清理:删除生成的文件
outputFile.delete()在Groovy中安全地从JSON对象中删除元素并更新文件,关键在于避免ConcurrentModificationException。我们提供了两种主要策略:
无论选择哪种方法,最终都可以通过JsonBuilder将修改后的数据序列化为JSON字符串,并使用File.write()方法将其持久化到文件中。理解这些策略并根据具体需求选择合适的方法,将帮助你编写出更健壮、高效的Groovy JSON处理代码。
以上就是Groovy中安全修改JSON数据:避免并发修改异常与文件更新实践的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号