All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.github.ddth.commons.utils.DPathUtils Maven / Gradle / Ivy

There is a newer version: 1.1.0
Show newest version
package com.github.ddth.commons.utils;

import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import org.apache.commons.lang3.ArrayUtils;
import org.apache.commons.lang3.StringUtils;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.ContainerNode;
import com.fasterxml.jackson.databind.node.MissingNode;
import com.fasterxml.jackson.databind.node.NullNode;
import com.fasterxml.jackson.databind.node.ObjectNode;

/**
 * Utility to access data from a hierarchy structure.
 * 
 * 

* Notation: *

* *
    *
  • . (the dot character): path separator
  • *
  • name: access a map's attribute specified by * name
  • *
  • [i]: access i'th element of a list/array (0-based)
  • *
* (example: employees.[1].first_name). * *

* Sample usage: assuming you have the following data structure: *

* *
 * Map<String, Object> company = new HashMap<String, Object>();
 * company.put("name", "Monster Corp.");
 * company.put("year", 2003);
 * 
 * List<Map<String, Object>> employees = new ArrayList<Map<String, Object>>();
 * company.put("employees", employees);
 * 
 * Map<String, Object> employee1 = new HashMap<String, Object>();
 * employee1.put("first_name", "Mike");
 * employee1.put("last_name", "Wazowski");
 * employee1.put("email", "[email protected]");
 * employee1.put("age", 29);
 * employees.add(employee1);
 * 
 * Map<String, Object> employee2 = new HashMap<String, Object>();
 * employee2.put("first_name", "Sulley");
 * employee2.put("last_name", "Sullivan");
 * employee2.put("email", "[email protected]");
 * employee2.put("age", 30);
 * employees.add(employee2);
 * 
* *

* You can access company's attributes: *

* *
 * String companyName = DPathUtils.getValue(company, "name", String.class);
 * // got string "Monster Corp."
 * 
 * Integer companyYear = DPathUtils.getValue(company, "year", Integer.class);
 * // got integer 2003
 * 
* *

* You can access the two employee: *

* *
 * Object user1 = DPathUtils.getValue(company, "employees.[0]");
 * Map<String, Object> user2 = DPathUtils.getValue(company, "employees.[1]", Map.class);
 * 
* *

* Or, employee's attributes: *

* *
 * String firstName1 = DPathUtils.getValue(company, "employees.[0].first_name", String.class);
 * Object email2 = DPathUtils.getValue(company, "employees.[1].email");
 * Long age2 = DPathUtils.getValue(company, "employees.[1].age", Long.class);
 * 
* * @author Thanh Nguyen * @since 0.1.0 */ public class DPathUtils { private final static Pattern PATTERN_INDEX = Pattern.compile("^\\[(.*?)\\]$"); public final static char PATH_SEPARATOR = '.'; private final static Pattern PATTERN_END_INDEX = Pattern.compile("^(.*)(\\[.*?\\])$"); /** * Split {@code DPath} string to tokens. * *
    *
  • {@code "a.b.c.[i].d"} or {@code "a.b.c[i].d"} will be split into * {@code ["a","b","c","[i]","d"]}
  • *
  • {@code "a.b.c.[i].[j].d"} or {@code "a.b.c[i].[j].d"} or * {@code "a.b.c[i][j].d"} or {@code "a.b.c.[i][j].d"} will be split into * {@code ["a","b","c","[i]","[j]","d"]}
  • *
* * @param dpath * @return * @since 0.6.1 */ @SuppressWarnings("serial") public final static String[] splitDpath(String dpath) { String[] tokens = StringUtils.split(dpath, PATH_SEPARATOR); List tokenList = new ArrayList() { { for (String token : tokens) { List temp = new ArrayList() { { String _token = token; Matcher m = PATTERN_END_INDEX.matcher(_token); while (m.matches()) { _token = m.group(1); add(0, m.group(2)); m = PATTERN_END_INDEX.matcher(_token); } if (!StringUtils.isBlank(_token)) { add(0, _token); } } }; addAll(temp); } } }; return tokenList.toArray(ArrayUtils.EMPTY_STRING_ARRAY); } private static Object extractValue(Object target, String index) { if (target == null) { return null; } Matcher m = PATTERN_INDEX.matcher(index); if (m.matches()) { try { int i = Integer.parseInt(m.group(1)); if (target instanceof Object[]) { return ((Object[]) target)[i]; } if (target instanceof List) { return ((List) target).get(i); } throw new IllegalArgumentException("Expect an array or list for index [" + index + "] but received [" + target.getClass() + "] instead."); } catch (IndexOutOfBoundsException e) { // rethrow for now throw e; } catch (NumberFormatException e) { throw new IllegalArgumentException("Invalid index value: " + index, e); } } if (target instanceof Map) { return ((Map) target).get(index); } throw new IllegalArgumentException( "Unsupported type [" + target.getClass() + "] or invalid index [" + index + "]"); } private static JsonNode extractValue(JsonNode node, String index) { if (node == null || node instanceof NullNode || node instanceof MissingNode) { return null; } Matcher m = PATTERN_INDEX.matcher(index); if (m.matches()) { try { int i = Integer.parseInt(m.group(1)); if (node instanceof ArrayNode) { if (i >= 0 && i < node.size()) { return node.get(i); } else { throw new IndexOutOfBoundsException(String.valueOf(i)); } } throw new IllegalArgumentException("Expect an ArrayNode for index [" + index + "] but received [" + node.getClass() + "] instead."); } catch (NumberFormatException e) { throw new IllegalArgumentException("Invalid index value: " + index, e); } } if (node instanceof ObjectNode) { return node.get(index); } throw new IllegalArgumentException( "Unsupported type [" + node.getClass() + "] or invalid index [" + index + "]"); } /*----------------------------------------------------------------------*/ /** * Extract a date value from the target object using DPath expression. If the extracted value is * a string, parse it as a {@link Date} using the specified date-time format. * * @param target * @param dPath * @param dateTimeFormat * see {@link SimpleDateFormat} * @return * @since 0.6.2 */ public static Date getDate(Object target, String dPath, String dateTimeFormat) { Object obj = getValue(target, dPath); return ValueUtils.convertDate(obj, dateTimeFormat); } /** * Extract a value from the target object using DPath expression (generic * version). * * @param target * @param dPath * @param clazz * @return */ public static T getValue(Object target, String dPath, Class clazz) { if (clazz == null) { throw new NullPointerException("Class parameter is null!"); } Object temp = getValue(target, dPath); return ValueUtils.convertValue(temp, clazz); } /** * Extract a value from the target object using DPath expression (generic * version). * * @param target * @param dPath * @param clazz * @return * @since 0.7.1 */ public static Optional getValueOptional(Object target, String dPath, Class clazz) { return Optional.ofNullable(getValue(target, dPath, clazz)); } /** * Extract a value from the target object using DPath expression. * * @param target * @param dPath */ public static Object getValue(Object target, String dPath) { String[] paths = splitDpath(dPath); Object result = target; for (String path : paths) { result = extractValue(result, path); } return result; } /** * Extract a value from the target object using DPath expression. * * @param target * @param dPath * @return * @since 0.7.1 */ public static Optional getValueOptional(Object target, String dPath) { return Optional.ofNullable(getValue(target, dPath)); } /*----------------------------------------------------------------------*/ /** * Extract a date value from the target {@link JsonNode} using DPath expression. If the * extracted value is a string, parse it as a {@link Date} using the specified date-time format. * * @param node * @param dPath * @param dateTimeFormat * see {@link SimpleDateFormat} * @return * @since 0.6.2 */ public static Date getDate(JsonNode node, String dPath, String dateTimeFormat) { JsonNode obj = getValue(node, dPath); return ValueUtils.convertDate(obj, dateTimeFormat); } /** * Extract a value from the target {@link JsonNode} using DPath expression (generic * version). * * @param node * @param dPath * @param clazz * @return * @since 0.6.2 */ public static T getValue(JsonNode node, String dPath, Class clazz) { if (clazz == null) { throw new NullPointerException("Class parameter is null!"); } JsonNode temp = getValue(node, dPath); return ValueUtils.convertValue(temp, clazz); } /** * Extract a value from the target {@link JsonNode} using DPath expression (generic * version). * * @param node * @param dPath * @param clazz * @return * @since 0.7.1 */ public static Optional getValueOptional(JsonNode node, String dPath, Class clazz) { return Optional.ofNullable(getValue(node, dPath, clazz)); } /** * Extract a value from the target {@link JsonNode} using DPath expression. * * @param node * @param dPath * @since 0.6.2 */ public static JsonNode getValue(JsonNode node, String dPath) { String[] paths = splitDpath(dPath); JsonNode result = node; for (String path : paths) { result = extractValue(result, path); } return result; } /** * Extract a value from the target {@link JsonNode} using DPath expression. * * @param node * @param dPath * @return * @since 0.7.1 */ public static Optional getValueOptional(JsonNode node, String dPath) { return Optional.ofNullable(getValue(node, dPath)); } /*----------------------------------------------------------------------*/ private static Object createIntermediate(Object target, StringBuffer pathSofar, String index, String nextIndex) { Object value = PATTERN_INDEX.matcher(nextIndex).matches() ? new ArrayList() : new HashMap(); return createIntermediate(target, pathSofar, index, value); } @SuppressWarnings("unchecked") private static Object createIntermediate(Object target, StringBuffer pathSofar, String index, Object value) { if (target == null) { return null; } Matcher m = PATTERN_INDEX.matcher(index); if (m.matches()) { try { int i = Integer.parseInt(m.group(1)); if (target instanceof List) { List temp = (List) target; if (i >= 0 && i < temp.size()) { // the middle item is null (in some cases) temp.set(i, value); } else if (i == temp.size()) { // special case: add the last item temp.add(value); } else { throw new IllegalArgumentException("Error: Index out of bound. Path [" + pathSofar + "], target [" + target.getClass() + "]."); } return value; } else if (target instanceof Object[]) { Object[] temp = (Object[]) target; if (i >= 0 && i < temp.length) { // the middle item is null (in some cases) temp[i] = value; } else { throw new IllegalArgumentException("Error: Index out of bound. Path [" + pathSofar + "], target [" + target.getClass() + "]."); } return value; } else { throw new IllegalArgumentException("Expect an array or list for path [" + pathSofar + "] but received [" + target.getClass() + "] instead."); } } catch (IndexOutOfBoundsException e) { // rethrow for now throw e; } catch (NumberFormatException e) { throw new IllegalArgumentException("Error: Invalid index. Path [" + pathSofar + "], target [" + target.getClass() + "].", e); } } else if (target instanceof Map) { Map temp = (Map) target; temp.put(index, value); return value; } else { throw new IllegalArgumentException("Target object of type [" + target.getClass() + "] is not writable with path [" + pathSofar + "]!"); } } /** * Set a value to the target object specified by DPath expression. * *

* Note: intermediated nodes will NOT be created. *

* *

* Note: if {@code value} is {@code null}: *

    *
  • If the specified item's parent is a list or array, the item * (specified by {@code dPath}) will be set to {@code null}.
  • *
  • If the specified item's parent is a map, the item (specified by * {@code dPath}) will be removed.
  • *
*

* * @param target * @param dPath * @param value */ public static void setValue(Object target, String dPath, Object value) { setValue(target, dPath, value, false); } /** * Set a value to the target object specified by DPath expression. * *

* Note: intermediated nodes will be created if * {@code createIntermediatePath} is true. *

* *

* Note: if {@code value} is {@code null}: *

    *
  • If the specified item's parent is a list or array, the item * (specified by {@code dPath}) will be set to {@code null}.
  • *
  • If the specified item's parent is a map, the item (specified by * {@code dPath}) will be removed.
  • *
*

* * @param target * @param dPath * @param value * @param createIntermediatePath * @since 0.6.1 */ @SuppressWarnings("unchecked") public static void setValue(Object target, String dPath, Object value, boolean createIntermediatePath) { if (target == null) { throw new IllegalArgumentException("Target is null!"); } String[] paths = splitDpath(dPath); Object cursor = target, prevCursor = target; StringBuffer pathSofar = new StringBuffer(); // "seek"to the correct position for (int i = 0; i < paths.length - 1; i++) { if (i > 0) { pathSofar.append(PATH_SEPARATOR); } String index = paths[i], nextIndex = paths[i + 1]; pathSofar.append(index); try { cursor = extractValue(cursor, index); } catch (IndexOutOfBoundsException e) { cursor = null; } if (cursor == null && createIntermediatePath) { // creating intermediate value cursor = createIntermediate(prevCursor, pathSofar, index, nextIndex); } prevCursor = cursor; } if (cursor == null) { throw new IllegalArgumentException("Path not found [" + dPath + "]!"); } String index = paths[paths.length - 1]; Matcher m = PATTERN_INDEX.matcher(index); if (m.matches() || "[]".equals(index)) { try { int i = "[]".equals(index) ? Integer.MAX_VALUE : Integer.parseInt(m.group(1)); if (cursor instanceof List) { List temp = (List) cursor; if (i >= temp.size()) { temp.add(value); } else if (i >= 0) { temp.set(i, value); } else { throw new IllegalArgumentException("Error: Index out of bound. Path [" + dPath + "], target [" + cursor.getClass() + "]."); } } else if (cursor instanceof Object[]) { Object[] temp = (Object[]) cursor; if (i >= 0 && i < temp.length) { temp[i] = value; } else { throw new IllegalArgumentException("Error: Index out of bound. Path [" + dPath + "], target [" + cursor.getClass() + "]."); } } else { throw new IllegalArgumentException( "Target object [" + cursor.getClass() + "] is not a list or array!"); } } catch (IndexOutOfBoundsException e) { // rethrow for now throw e; } catch (NumberFormatException e) { throw new IllegalArgumentException("Error: Invalid index. Path [" + dPath + "], target [" + cursor.getClass() + "].", e); } } else if (cursor instanceof Map) { Map temp = (Map) cursor; if (value == null) { temp.remove(index); } else { temp.put(index, value); } } else { throw new IllegalArgumentException("Target object of type [" + cursor.getClass() + "] is not writable with path [" + dPath + "]!"); } } /** * Delete a value from the target object specified by DPath expression. * *
    *
  • If the specified item's parent is a list or map, the item (specified * by {@code dPath}) will be removed (for list, next items will be shifted up one * position).
  • *
  • If the specified item's parent is an array, the item (specified by * {@code dPath}) will be set to {@code null}.
  • *
* * @param target * @param dPath * @since 0.6.1 */ @SuppressWarnings("unchecked") public static void deleteValue(Object target, String dPath) { String[] paths = splitDpath(dPath); Object cursor = target; // "seek"to the correct position for (int i = 0; i < paths.length - 1; i++) { cursor = extractValue(cursor, paths[i]); } if (cursor == null) { return; } String index = paths[paths.length - 1]; Matcher m = PATTERN_INDEX.matcher(index); if (m.matches()) { try { int i = Integer.parseInt(m.group(1)); if (cursor instanceof List) { List temp = (List) cursor; if (i >= 0 && i < temp.size()) { temp.remove(i); } else { throw new IllegalArgumentException("Error: Index out of bound. Path [" + dPath + "], target [" + cursor.getClass() + "]."); } } else { throw new IllegalArgumentException( "Target object [" + cursor.getClass() + "] is not a list!"); } } catch (IndexOutOfBoundsException e) { // rethrow for now throw e; } catch (NumberFormatException e) { throw new IllegalArgumentException("Error: Invalid index. Path [" + dPath + "], target [" + cursor.getClass() + "].", e); } } else if (cursor instanceof Map) { Map temp = (Map) cursor; temp.remove(index); } else { throw new IllegalArgumentException("Target object of type [" + cursor.getClass() + "] is not writable with path [" + dPath + "]!"); } } /*----------------------------------------------------------------------*/ @SuppressWarnings("rawtypes") private static JsonNode createIntermediate(JsonNode node, StringBuffer pathSofar, String index, String nextIndex) { if (node instanceof ContainerNode) { ContainerNode temp = (ContainerNode) node; JsonNode value = PATTERN_INDEX.matcher(nextIndex).matches() ? temp.arrayNode() : temp.objectNode(); return createIntermediate(temp, pathSofar, index, value); } return null; } private static JsonNode createIntermediate(JsonNode node, StringBuffer pathSofar, String index, JsonNode value) { if (node == null) { return null; } Matcher m = PATTERN_INDEX.matcher(index); if (m.matches()) { try { int i = Integer.parseInt(m.group(1)); if (node instanceof ArrayNode) { ArrayNode temp = (ArrayNode) node; if (i >= 0 && i < temp.size()) { // the middle item is null (in some cases) temp.set(i, value); } else if (i == temp.size()) { // special case: add the last item temp.add(value); } else { throw new IllegalArgumentException("Error: Index out of bound. Path [" + pathSofar + "], target [" + node.getClass() + "]."); } return value; } else { throw new IllegalArgumentException("Expect an ArrayNode for path [" + pathSofar + "] but received [" + node.getClass() + "] instead."); } } catch (IndexOutOfBoundsException e) { // rethrow for now throw e; } catch (NumberFormatException e) { throw new IllegalArgumentException("Error: Invalid index. Path [" + pathSofar + "], target [" + node.getClass() + "].", e); } } else if (node instanceof ObjectNode) { ObjectNode temp = (ObjectNode) node; temp.set(index, value); return value; } else { throw new IllegalArgumentException("Target object of type [" + node.getClass() + "] is not writable with path [" + pathSofar + "]!"); } } /** * Set a value to the target {@link JsonNode} specified by DPath expression. * *

* Note: intermediated nodes will NOT be created. *

* *

* Note: if {@code value} is {@code null}: *

    *
  • If the specified item's parent is an {@link ArrayNode}, the item * (specified by {@code dPath}) will be set to {@code null}.
  • *
  • If the specified item's parent is an {@link ObjectNode}, the item (specified by * {@code dPath}) will be removed.
  • *
*

* * @param node * @param dPath * @param value * @since 0.6.2 */ public static void setValue(JsonNode node, String dPath, Object value) { setValue(node, dPath, value, false); } /** * Set a value to the target {@link JsonNode} specified by DPath expression. * *

* Note: intermediated nodes will be created if {@code createIntermediatePath} is true. *

* *

* Note: if {@code value} is {@code null}: *

    *
  • If the specified item's parent is an {@link ArrayNode}, the item * (specified by {@code dPath}) will be set to {@code null}.
  • *
  • If the specified item's parent is an {@link ObjectNode}, the item (specified by * {@code dPath}) will be removed.
  • *
*

* * @param node * @param dPath * @param value * @param createIntermediatePath * @since 0.6.2 */ public static void setValue(JsonNode node, String dPath, Object value, boolean createIntermediatePath) { if (node == null) { throw new IllegalArgumentException("Target is null!"); } String[] paths = splitDpath(dPath); JsonNode cursor = node, prevCursor = node; StringBuffer pathSofar = new StringBuffer(); // "seek"to the correct position for (int i = 0; i < paths.length - 1; i++) { if (i > 0) { pathSofar.append(PATH_SEPARATOR); } String index = paths[i], nextIndex = paths[i + 1]; pathSofar.append(index); try { cursor = extractValue(cursor, index); } catch (IndexOutOfBoundsException e) { cursor = null; } if ((cursor == null || cursor instanceof NullNode || cursor instanceof MissingNode) && createIntermediatePath) { // creating intermediate value cursor = createIntermediate(prevCursor, pathSofar, index, nextIndex); } prevCursor = cursor; } if (cursor == null || cursor instanceof NullNode || cursor instanceof MissingNode) { throw new IllegalArgumentException("Path not found [" + dPath + "]!"); } JsonNode valueNode = value instanceof JsonNode ? (JsonNode) value : JacksonUtils.toJson(value); String index = paths[paths.length - 1]; Matcher m = PATTERN_INDEX.matcher(index); if (m.matches() || "[]".equals(index)) { try { int i = "[]".equals(index) ? Integer.MAX_VALUE : Integer.parseInt(m.group(1)); if (cursor instanceof ArrayNode) { ArrayNode temp = (ArrayNode) cursor; if (i >= temp.size() && value != null) { // add to end temp.add(valueNode); } else if (i >= 0) { // set middle (can be null) temp.set(i, valueNode); } else { throw new IllegalArgumentException("Error: Index out of bound. Path [" + dPath + "], target [" + cursor.getClass() + "]."); } } else { throw new IllegalArgumentException( "Target object [" + cursor.getClass() + "] is not a ArrayNode!"); } } catch (IndexOutOfBoundsException e) { // rethrow for now throw e; } catch (NumberFormatException e) { throw new IllegalArgumentException("Error: Invalid index. Path [" + dPath + "], target [" + cursor.getClass() + "].", e); } } else if (cursor instanceof ObjectNode) { ObjectNode temp = (ObjectNode) cursor; if (value == null) { temp.remove(index); } else { temp.set(index, valueNode); } } else { throw new IllegalArgumentException("Target object of type [" + cursor.getClass() + "] is not writable with path [" + dPath + "]!"); } } /** * Delete a value from the target {@link JsonNode} specified by DPath expression. * * @param node * @param dPath * @since 0.6.2 */ public static void deleteValue(JsonNode node, String dPath) { String[] paths = splitDpath(dPath); JsonNode cursor = node; // "seek"to the correct position for (int i = 0; i < paths.length - 1; i++) { cursor = extractValue(cursor, paths[i]); } if (cursor == null || cursor instanceof NullNode || cursor instanceof MissingNode) { return; } String index = paths[paths.length - 1]; Matcher m = PATTERN_INDEX.matcher(index); if (m.matches()) { try { int i = Integer.parseInt(m.group(1)); if (cursor instanceof ArrayNode) { ArrayNode temp = (ArrayNode) cursor; if (i >= 0 && i < temp.size()) { temp.remove(i); } else { throw new IllegalArgumentException("Error: Index out of bound. Path [" + dPath + "], target [" + cursor.getClass() + "]."); } } else { throw new IllegalArgumentException( "Target object [" + cursor.getClass() + "] is not an ArrayNode!"); } } catch (IndexOutOfBoundsException e) { // rethrow for now throw e; } catch (NumberFormatException e) { throw new IllegalArgumentException("Error: Invalid index. Path [" + dPath + "], target [" + cursor.getClass() + "].", e); } } else if (cursor instanceof ObjectNode) { ObjectNode temp = (ObjectNode) cursor; temp.remove(index); } else { throw new IllegalArgumentException("Target object of type [" + cursor.getClass() + "] is not writable with path [" + dPath + "]!"); } } @SuppressWarnings("unchecked") public static void main(String[] args) { Object sample1 = new int[0]; Object sample2 = new Integer[0]; Object sample3 = new Object[0]; System.out.println("int[] data : " + sample1); System.out.println("int[] class : " + sample1.getClass()); System.out.println("Integer[] data : " + sample2); System.out.println("Integer[] class: " + sample2.getClass()); System.out.println("Object[] data : " + sample3); System.out.println("Object[] class : " + sample3.getClass()); System.out.println(); System.out.println("int[] is Object[] : " + (sample1 instanceof Object[])); System.out.println("Integer[] is Object[]: " + (sample2 instanceof Object[])); System.out.println("Object[] is Object[] : " + (sample3 instanceof Object[])); System.out.println(); // init data Map company = new HashMap(); { company.put("name", "Monster Corp."); company.put("year", 2003); List> employees = new ArrayList>(); company.put("employees", employees); Map employee1 = new HashMap(); employee1.put("first_name", "Mike"); employee1.put("last_name", "Wazowski"); employee1.put("email", "[email protected]"); employee1.put("age", 29); employees.add(employee1); Map employee2 = new HashMap(); employee2.put("first_name", "Sulley"); employee2.put("last_name", "Sullivan"); employee2.put("email", "[email protected]"); employee2.put("age", 30); employees.add(employee2); } // access company's attributes: String companyName = DPathUtils.getValue(company, "name", String.class); System.out.println(companyName); Integer companyYear = DPathUtils.getValue(company, "year", Integer.class); System.out.println(companyYear); // access the two employee: Object user1 = DPathUtils.getValue(company, "employees.[0]"); System.out.println(user1); Map user2 = DPathUtils.getValue(company, "employees.[1]", Map.class); System.out.println(user2); // access employee's attributes: String firstName1 = DPathUtils.getValue(company, "employees.[0].first_name", String.class); System.out.println(firstName1); Object email2 = DPathUtils.getValue(company, "employees.[1].email"); System.out.println(email2); Long age2 = DPathUtils.getValue(company, "employees.[1].age", Long.class); System.out.println(age2); } }