Получить количество объектов, на которые ссылается ArrayList, с размером 1, сгруппированных по классу

Я получил дамп кучи из приложения и обнаружил, что существует огромное количество ArrayLists только с 1 объектом в нем. Я знаю, как получить список таких массивов, а также показать класс содержащегося элемента:

SELECT list.elementData[0] FROM java.util.ArrayList list WHERE (list.size = 1)

Результат выглядит следующим образом:

java.lang.String [id=0x7f8e44970]
java.lang.String [id=0x7f8e44980]
java.lang.String [id=0x7f8e44572]
java.io.File [id=0x7f8e43572]
...

Я хотел бы получить что-то вроде этого:

Class             | count
=================================
java.lang.String  | 100
java.io.File      | 74
...

но я не могу обобщить результаты или сделать что-то еще с ними. Я нашел здесь, как передать значения внешнему выбору, но я не могу понять, как использовать что-нибудь еще, кроме * в первом выборе.

SELECT * from OBJECTS
(SELECT OBJECTS list.elementData[0] FROM java.util.ArrayList list WHERE (list.size = 1))

person NeplatnyUdaj    schedule 26.02.2015    source источник


Ответы (1)


В OQL VisualVM нет group by, но вы можете использовать встроенные функции для создания фрагмента кода JavaScript. и запустите его в «Консоли OQL»:

var c = {};

/* Filter to show only the first occurence (max count) */
filter(
  /* Sort by occurences desc */
  sort(
    /* Count class instances */
    map(
      heap.objects("java.util.ArrayList"),
      function(list) {
        var clazz = 'null';
        if (list.size = 1 && list.elementData[0] != null) {
          clazz = classof(list.elementData[0]).name;
        }

        c[clazz] = (c[clazz] ? c[clazz] + 1 : 1);
        return { cnt:c[clazz], type:clazz };
      }
    ), 'rhs.cnt - lhs.cnt'
  ),
  function (item) {
    if (c[item.type]) {
      c[item.type] = false;
      return true;
    } else {
      return false;
    }
  }
);

Вывод представляет собой массив объектов, например:

{
cnt = 3854.0,
type = null
}    
{
cnt = 501.0,
type = org.apache.tomcat.util.digester.CallMethodRule
}
{
cnt = 256.0,
type = java.lang.ref.WeakReference
}
{
cnt = 176.0,
type = sun.reflect.generics.tree.SimpleClassTypeSignature
}

Наконец, вы можете снова вызвать функцию map, чтобы отформатировать вывод во что-то еще, например, для exmpale как csv:

map(  
  filter(...),
  'it.type + ", " + it.cnt'
);

выход:

null, 3854    
org.apache.tomcat.util.digester.CallMethodRule, 501    
java.lang.ref.WeakReference, 256    
sun.reflect.generics.tree.SimpleClassTypeSignature, 176    
org.apache.tomcat.util.digester.CallParamRule, 144    
com.sun.tools.javac.file.ZipFileIndex$Entry, 141    
org.apache.tomcat.util.digester.ObjectCreateRule, 78
person Tobías    schedule 26.02.2015