/* * Copyright 2014-2015 JKOOL, LLC. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.jkoolcloud.tnt4j.tracker; import java.util.Map; import java.util.concurrent.ConcurrentMap; import java.util.concurrent.TimeUnit; import com.google.common.cache.Cache; import com.google.common.cache.CacheBuilder; /** * This class implements time tracker for a set of keys. * The class maintains a cache of time stamp hits and measures time * since the last hit on a set of keys or by thread. * * @version $Revision: 1$ */ public class TimeTracker { /* * Timing thread local maintains timing since last hit for specific thread */ private static final ThreadLocal<TimeStats> THREAD_TIMER = new ThreadLocal<TimeStats>() { @Override protected TimeStats initialValue() { return new TimeStats(); } }; /* * Timing map maintains timing since last hit for a specific key */ final ConcurrentMap<String, TimeStats> EVENT_MAP; /* * Timing cache maintains timing since last hit for a specific key */ final Cache<String, TimeStats> EVENT_CACHE; /** * Create a time tracker with specified capacity and life span * * @param capacity * maximum capacity * @param lifeSpan life span in milliseconds */ private TimeTracker(int capacity, long lifeSpan) { EVENT_CACHE = CacheBuilder.newBuilder().concurrencyLevel(Runtime.getRuntime().availableProcessors()).recordStats() .maximumSize(capacity).expireAfterWrite(lifeSpan, TimeUnit.MILLISECONDS).build(); EVENT_MAP = EVENT_CACHE.asMap(); } /** * Create a default time tracker with specified capacity and life span * * @param capacity * maximum capacity * @param lifeSpan life span in milliseconds */ public static TimeTracker newTracker(int capacity, long lifeSpan) { return new TimeTracker(capacity, lifeSpan); } /** * Hit and obtain elapsed nanoseconds since last hit based. * Time statistics is maintained per thread. * * @return elapsed nanoseconds since last hit */ public static long hitAndGet() { return THREAD_TIMER.get().hit(); } /** * Obtain time statistics maintained per thread * * @return time statistics maintained per thread */ public static TimeStats getStats() { return THREAD_TIMER.get(); } /** * Hit and obtain elapsed nanoseconds since last hit * * @param key * timer key * @return elapsed nanoseconds since last hit */ public long hitAndGet(String key) { TimeStats timeStats = EVENT_MAP.get(key); if (timeStats == null) { timeStats = EVENT_MAP.putIfAbsent(key, new TimeStats()); timeStats = timeStats == null? EVENT_MAP.get(key): timeStats; } return timeStats.hit(); } /** * obtain hit count for a specific key * * @param key * timer key * @return hit count for a specific key */ public long getHitCount(String key) { TimeStats last = EVENT_MAP.get(key); return last != null? last.getHitCount(): 0; } /** * obtain elapsed nanoseconds for a specific key * * @param key * timer key * @return hit count for a specific key */ public long getElapsedNanos(String key) { TimeStats last = EVENT_MAP.get(key); return last != null? last.getAgeNanos(): 0; } /** * Obtain time statistics for a specific key * * @return time statistics for a specific key */ public TimeStats getStats(String key) { return EVENT_MAP.get(key); } /** * Get map of all time statistics maintained by this tracker * * @return map of all time statistics maintained by this tracker */ public Map<String, TimeStats> getTimeStats() { return EVENT_MAP; } }