Newer
Older
MagnetRON / src / org / ntlab / objectGraphAnalyzer / ObjectCallCounter.java
Aki Hongo on 3 Mar 2020 8 KB first commit
package org.ntlab.objectGraphAnalyzer;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.AbstractMap.SimpleEntry;

import org.ntlab.trace.IMethodExecutionVisitor;
import org.ntlab.trace.IStatementVisitor;
import org.ntlab.trace.MethodExecution;
import org.ntlab.trace.MethodInvocation;
import org.ntlab.trace.Statement;
import org.ntlab.trace.ThreadInstance;
import org.ntlab.trace.Trace;
import org.ntlab.trace.TraceJSON;

public class ObjectCallCounter {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		TraceJSON trace = new TraceJSON("traces\\jEditNormal.trace");
		
		// オブジェクト生成木の作成
		final HashMap<String, String> classNames = new HashMap<>();
		final ArrayList<SimpleEntry<String, String>> links = new ArrayList<>();
		final HashMap<String, String> parentOf = new HashMap<>();
		
		trace.traverseMethodEntriesInTraceBackward(new IMethodExecutionVisitor() {
			@Override
			public boolean preVisitThread(ThreadInstance thread) {
				return false;
			}
			@Override
			public boolean preVisitMethodExecution(MethodExecution methodExecution) {
				return false;
			}
			@Override
			public boolean postVisitThread(ThreadInstance thread) {
				return false;
			}
			@Override
			public boolean postVisitMethodExecution(MethodExecution methodExecution, ArrayList<MethodExecution> children) {
				if (methodExecution.isConstructor() && methodExecution.getCallerMethodExecution() != null) {
					String srcObjId = methodExecution.getCallerMethodExecution().getThisObjId();
					String dstObjId = methodExecution.getThisObjId();
					if (!srcObjId.equals(dstObjId)) {		// 親子クラス間のメソッド呼び出しがあった場合を考慮
						if (Trace.isNull(srcObjId)) {
							srcObjId = methodExecution.getCallerMethodExecution().getThisClassName();
						}
						classNames.put(srcObjId, methodExecution.getCallerMethodExecution().getThisClassName());
						classNames.put(dstObjId, methodExecution.getThisClassName());
						links.add(new SimpleEntry<>(srcObjId, dstObjId));
						parentOf.put(dstObjId, srcObjId);
					}
				}
				return false;
			}
		});
		
		// 各オブジェクトが所属する木のルートノードと深さの計算
		final HashMap<String, String> rootOf = new HashMap<>();
		final HashMap<String, Integer> depthOf = new HashMap<>();
		
		for (String objId: classNames.keySet()) {
			int depth = 0;
			String rootObjId = objId;
			while (parentOf.get(rootObjId) != null) {
				rootObjId = parentOf.get(rootObjId);
				depth++;
				if (depthOf.get(rootObjId) != null) {
					depth += depthOf.get(rootObjId);
					rootObjId = rootOf.get(rootObjId);
				}
			}
			rootOf.put(objId, rootObjId);
			depthOf.put(objId, depth);			
		}
		
		// オブジェクト生成木上での各呼び出し距離、呼び出し回数の計算
		final HashMap<String, Integer> maxCallDistances = new HashMap<>();
		final HashMap<String, Integer> totalCallDistances = new HashMap<>();
		final HashMap<String, HashMap<String, Integer>> callDistances = new HashMap<>();
		final HashMap<String, Integer> callCounts = new HashMap<>();
		
		trace.traverseStatementsInTrace(new IStatementVisitor() {
			@Override
			public boolean preVisitStatement(Statement statement) {
				if (statement instanceof MethodInvocation) {
					MethodInvocation c = (MethodInvocation)statement;
					String srcObjId = c.getThisObjId();
					String dstObjId = c.getCalledMethodExecution().getThisObjId();
					if (Trace.isNull(srcObjId)) {
						srcObjId = c.getThisClassName();
					}
					if (Trace.isNull(dstObjId)) {
						dstObjId = c.getCalledMethodExecution().getThisClassName();
					}						
					int distance;
					if (rootOf.get(srcObjId) == null) {
						rootOf.put(srcObjId, srcObjId);
						depthOf.put(srcObjId, 0);
						classNames.put(srcObjId, c.getThisClassName());
					}
					if (rootOf.get(dstObjId) == null) {
						rootOf.put(dstObjId, dstObjId);
						depthOf.put(dstObjId, 0);
						classNames.put(dstObjId, c.getCalledMethodExecution().getThisClassName());
					}
//					if (rootOf.get(srcObjId).equals(rootOf.get(dstObjId))) {
						ArrayList<String> srcPath = new ArrayList<>();						
						srcPath.add(srcObjId);
						String rootObjId = srcObjId;
						while (parentOf.get(rootObjId) != null) {
							rootObjId = parentOf.get(rootObjId);
							srcPath.add(0, rootObjId);
						}
						
						ArrayList<String> dstPath = new ArrayList<>();
						dstPath.add(dstObjId);
						rootObjId = dstObjId;
						while (parentOf.get(rootObjId) != null) {
							rootObjId = parentOf.get(rootObjId);
							dstPath.add(0, rootObjId);
						}
						
						int srcDepth = depthOf.get(srcObjId);
						int dstDepth = depthOf.get(dstObjId);
						int depth = 0;
						while (srcPath.get(depth).equals(dstPath.get(depth))) {
							depth++;
							if (depth >= srcPath.size() || depth >= dstPath.size()) break;
						}
						distance = srcDepth + dstDepth - 2 * (depth - 1);
//					} else {
//						if (depthOf.get(dstObjId) != null) {
//							distance = depthOf.get(dstObjId) + 2;		// システム全体のルートまでを1、システム全体のルートから各クラスオブジェクトまでを1とする
//						} else {
//							distance = 2;
//						}
//					}
					if (callDistances.get(dstObjId) == null) {
						callDistances.put(dstObjId, new HashMap<String, Integer>());
					}
					HashMap<String, Integer> callDistance = callDistances.get(dstObjId);
					if (callDistance.get(srcObjId) == null) {
						callDistance.put(srcObjId, distance);
					}
					if (totalCallDistances.get(dstObjId) != null) {
						totalCallDistances.put(dstObjId, totalCallDistances.get(dstObjId) + distance);
					} else {
						totalCallDistances.put(dstObjId, distance);
					}
					if (callCounts.get(dstObjId) != null) {
						callCounts.put(dstObjId, callCounts.get(dstObjId) + 1);
					} else {
						callCounts.put(dstObjId, 1);
					}
					if (maxCallDistances.get(dstObjId) == null || distance > maxCallDistances.get(dstObjId)) {
						maxCallDistances.put(dstObjId, distance);
					}
				}
				return false;
			}
			@Override
			public boolean postVisitStatement(Statement statement) {
				return false;
			}
		});
		
		// オブジェクトid、クラス名、総呼び出し回数、総呼び出し距離、最大呼び出し距離、呼び出し元オブジェクト数、呼び出し元オブジェクト単位の呼び出し距離の総和の出力
		for (String objId: callCounts.keySet()) {
			int variation = 0;
			int sum = 0;
			for (String srcObjId: callDistances.get(objId).keySet()) {
				sum += callDistances.get(objId).get(srcObjId);
				variation++;
			}
			System.out.println(objId + ":" + classNames.get(objId) + ":" + callCounts.get(objId) + ":" + totalCallDistances.get(objId) + ":" + maxCallDistances.get(objId)+ ":" + variation + ":" + sum);
		}
		
//		final HashMap<String, String> classNames = new HashMap<>();
//		final HashMap<String, Integer> callCounts = new HashMap<>();
//		
//		for (ThreadInstance thread: trace.getAllThreads().values()) {
//			thread.traverseMethodExecutionsBackward(new IMethodExecutionVisitor() {
//				@Override
//				public boolean preVisitThread(ThreadInstance thread) {
//					return false;
//				}
//				
//				@Override
//				public boolean preVisitMethodExecution(MethodExecution methodExecution) {
//					String objId = methodExecution.getThisObjId();
//					if (!Trace.isNull(objId)) {
//						String key = objId;
//						if (classNames.get(key) == null) {
//							classNames.put(key, methodExecution.getThisClassName());
//						}
//						if (callCounts.get(key) == null) {
//							callCounts.put(key, 1);
//						} else {
//							callCounts.put(key, callCounts.get(key) + 1);
//						}
//					} else {
//						String className = methodExecution.getThisClassName();
//						String key = className;
//						if (classNames.get(key) == null) {
//							classNames.put(key, className);
//						}
//						if (callCounts.get(key) == null) {
//							callCounts.put(key, 1);
//						} else {
//							callCounts.put(key, callCounts.get(key) + 1);
//						}
//					}
//					return false;
//				}
//				
//				@Override
//				public boolean postVisitThread(ThreadInstance thread) {
//					return false;
//				}
//				
//				@Override
//				public boolean postVisitMethodExecution(MethodExecution methodExecution, ArrayList<MethodExecution> children) {
//					return false;
//				}
//			});
//		}
//		
//		for (String objId: callCounts.keySet()) {
//			System.out.println(objId + ":" + classNames.get(objId) + ":" + callCounts.get(objId));
//		}
	}

}