I need to get values from a string with a JSON array using regex String example:
[INFO][2022-11-11] Response body :
{
"values":[
"abc123",
"def456",
"xyz789"
]
}
So I want to get the values: abc123, def456, xyz789
Please note I'm working with a string (actually it's a log output) so I'm not sure if I can use any libraries for JSON parsing
I'm trying to use the following regex, but getting only the last array value xyz789:
"values"\s*:\s*\[(\s*"(\w+)"\s*,?)*]
My solution in Java
String pattern = "\"values\"\\s*:\\s*\\[(\\s*\"(\\w+)\"\\s*,?)*]" ;
Matcher matcher = Pattern.compile(pattern).matcher(source());
while (matcher.find()) {
System.out.println("matcher.group() = " + matcher.group());
System.out.println("matcher.group(1) = " + matcher.group(1));
System.out.println("matcher.group(1) = " + matcher.group(2)); //xyz789
}
private static String source() {
String s = "{\"values\": [\n"
+ " \"abc123\", \"def456\", "
+ " \"xyz789\" ]\n"
+ " }";
return s;
}
It can be done almost effortlessly if you would use a library for parsing JSON, like Jackson, Gson, etc., instead of trying to reinvent the wheel.
Here's an example of how it can be done using Jackson.
Assume you have the following POJO:
public class MyPojo {
private List<String> values;
// getters, setters
}
That's all you need to deserialize the incoming JSON:
String jsonStr = // incoming JSON
ObjectMapper mapper = new JsonMapper();
MyPojo pojo = mapper.readValue(jsonStr, MyPojo.class);
System.out.println(pojo.getValues());
Output:
[abc123, def456, xyz789]
More over, if you would use platforms like Spring, or Jakarta EE conversion of requests and responses would be done automatically for you.
If the data is dynamic and introducing a POJO doesn't make much sense, then you can parse it as a Tree, and work with Nodes of that Tree:
String jsonStr = // incoming JSON
ObjectMapper mapper = new JsonMapper();
JsonNode node = mapper.readTree(jsonStr).get("values");
List<String> values = mapper.readerFor(new TypeReference<List<String>>() {}).readValue(node);
values.forEach(System.out::println);
Output:
abc123
def456
xyz789
If you want to do it manually at all costs, here is a way to extract these values by capturing the JSON-array and splitting it:
String jsonStr = // incoming JSON
Pattern pattern1 = Pattern.compile("\"values\"\\s*:\\s*\\[(.+)]");
Matcher matcher = pattern1.matcher(jsonStr);
List<String> values = List.of();
if (matcher.find()) {
values = Arrays.stream(matcher.group(1).split(","))
.map(s -> s.replaceAll("\"", "").strip())
.toList();
}
values.forEach(System.out::println);
Output:
abc123
def456
xyz789