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

org.apache.log4j.pattern.CachedDateFormat Maven / Gradle / Ivy

There is a newer version: 6.1.3
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 org.apache.log4j.pattern;

import java.text.DateFormat;
import java.text.FieldPosition;
import java.text.NumberFormat;
import java.text.ParsePosition;
import java.util.Date;
import java.util.TimeZone;

/**
 * CachedDateFormat optimizes the performance of a wrapped DateFormat. The
 * implementation is not thread-safe. If the millisecond pattern is not
 * recognized, the class will only use the cache if the same value is requested.
 *
 */
public final class CachedDateFormat extends DateFormat {
    /**
     * Serialization version.
     */
    private static final long serialVersionUID = 1;
    /**
     * Constant used to represent that there was no change observed when changing
     * the millisecond count.
     */
    public static final int NO_MILLISECONDS = -2;

    /**
     * Supported digit set. If the wrapped DateFormat uses a different unit set, the
     * millisecond pattern will not be recognized and duplicate requests will use
     * the cache.
     */
    private static final String DIGITS = "0123456789";

    /**
     * Constant used to represent that there was an observed change, but was an
     * expected change.
     */
    public static final int UNRECOGNIZED_MILLISECONDS = -1;

    /**
     * First magic number used to detect the millisecond position.
     */
    private static final int MAGIC1 = 654;

    /**
     * Expected representation of first magic number.
     */
    private static final String MAGICSTRING1 = "654";

    /**
     * Second magic number used to detect the millisecond position.
     */
    private static final int MAGIC2 = 987;

    /**
     * Expected representation of second magic number.
     */
    private static final String MAGICSTRING2 = "987";

    /**
     * Expected representation of 0 milliseconds.
     */
    private static final String ZERO_STRING = "000";

    /**
     * Wrapped formatter.
     */
    private final DateFormat formatter;

    /**
     * Index of initial digit of millisecond pattern or UNRECOGNIZED_MILLISECONDS or
     * NO_MILLISECONDS.
     */
    private int millisecondStart;

    /**
     * Integral second preceding the previous convered Date.
     */
    private long slotBegin;

    /**
     * Cache of previous conversion.
     */
    private StringBuffer cache = new StringBuffer(50);

    /**
     * Maximum validity period for the cache. Typically 1, use cache for duplicate
     * requests only, or 1000, use cache for requests within the same integral
     * second.
     */
    private final int expiration;

    /**
     * Date requested in previous conversion.
     */
    private long previousTime;

    /**
     * Scratch date object used to minimize date object creation.
     */
    private final Date tmpDate = new Date(0);

    /**
     * Creates a new CachedDateFormat object.
     * 
     * @param dateFormat Date format, may not be null.
     * @param expiration maximum cached range in milliseconds. If the dateFormat is
     *                   known to be incompatible with the caching algorithm, use a
     *                   value of 0 to totally disable caching or 1 to only use
     *                   cache for duplicate requests.
     */
    public CachedDateFormat(final DateFormat dateFormat, final int expiration) {
	if (dateFormat == null) {
	    throw new IllegalArgumentException("dateFormat cannot be null");
	}

	if (expiration < 0) {
	    throw new IllegalArgumentException("expiration must be non-negative");
	}

	formatter = dateFormat;
	this.expiration = expiration;
	millisecondStart = 0;

	//
	// set the previousTime so the cache will be invalid
	// for the next request.
	previousTime = Long.MIN_VALUE;
	slotBegin = Long.MIN_VALUE;
    }

    /**
     * Finds start of millisecond field in formatted time.
     * 
     * @param time      long time, must be integral number of seconds
     * @param formatted String corresponding formatted string
     * @param formatter DateFormat date format
     * @return int position in string of first digit of milliseconds, -1 indicates
     *         no millisecond field, -2 indicates unrecognized field (likely
     *         RelativeTimeDateFormat)
     */
    public static int findMillisecondStart(final long time, final String formatted, final DateFormat formatter) {
	long slotBegin = (time / 1000) * 1000;

	if (slotBegin > time) {
	    slotBegin -= 1000;
	}

	int millis = (int) (time - slotBegin);

	int magic = MAGIC1;
	String magicString = MAGICSTRING1;

	if (millis == MAGIC1) {
	    magic = MAGIC2;
	    magicString = MAGICSTRING2;
	}

	String plusMagic = formatter.format(new Date(slotBegin + magic));

	/**
	 * If the string lengths differ then we can't use the cache except for duplicate
	 * requests.
	 */
	if (plusMagic.length() != formatted.length()) {
	    return UNRECOGNIZED_MILLISECONDS;
	} else {
	    // find first difference between values
	    for (int i = 0; i < formatted.length(); i++) {
		if (formatted.charAt(i) != plusMagic.charAt(i)) {
		    //
		    // determine the expected digits for the base time
		    StringBuffer formattedMillis = new StringBuffer("ABC");
		    millisecondFormat(millis, formattedMillis, 0);

		    String plusZero = formatter.format(new Date(slotBegin));

		    // If the next 3 characters match the magic
		    // string and the expected string
		    if ((plusZero.length() == formatted.length())
			    && magicString.regionMatches(0, plusMagic, i, magicString.length())
			    && formattedMillis.toString().regionMatches(0, formatted, i, magicString.length())
			    && ZERO_STRING.regionMatches(0, plusZero, i, ZERO_STRING.length())) {
			return i;
		    } else {
			return UNRECOGNIZED_MILLISECONDS;
		    }
		}
	    }
	}

	return NO_MILLISECONDS;
    }

    /**
     * Formats a Date into a date/time string.
     *
     * @param date          the date to format.
     * @param sbuf          the string buffer to write to.
     * @param fieldPosition remains untouched.
     * @return the formatted time string.
     */
    public StringBuffer format(Date date, StringBuffer sbuf, FieldPosition fieldPosition) {
	format(date.getTime(), sbuf);

	return sbuf;
    }

    /**
     * Formats a millisecond count into a date/time string.
     *
     * @param now Number of milliseconds after midnight 1 Jan 1970 GMT.
     * @param buf the string buffer to write to.
     * @return the formatted time string.
     */
    public StringBuffer format(long now, StringBuffer buf) {
	//
	// If the current requested time is identical to the previously
	// requested time, then append the cache contents.
	//
	if (now == previousTime) {
	    buf.append(cache);

	    return buf;
	}

	//
	// If millisecond pattern was not unrecognized
	// (that is if it was found or milliseconds did not appear)
	//
	if (millisecondStart != UNRECOGNIZED_MILLISECONDS &&
	// Check if the cache is still valid.
	// If the requested time is within the same integral second
	// as the last request and a shorter expiration was not requested.
		(now < (slotBegin + expiration)) && (now >= slotBegin) && (now < (slotBegin + 1000L))) {
	    //
	    // if there was a millisecond field then update it
	    //
	    if (millisecondStart >= 0) {
		millisecondFormat((int) (now - slotBegin), cache, millisecondStart);
	    }

	    //
	    // update the previously requested time
	    // (the slot begin should be unchanged)
	    previousTime = now;
	    buf.append(cache);

	    return buf;
	}

	//
	// could not use previous value.
	// Call underlying formatter to format date.
	cache.setLength(0);
	tmpDate.setTime(now);
	cache.append(formatter.format(tmpDate));
	buf.append(cache);
	previousTime = now;
	slotBegin = (previousTime / 1000) * 1000;

	if (slotBegin > previousTime) {
	    slotBegin -= 1000;
	}

	//
	// if the milliseconds field was previous found
	// then reevaluate in case it moved.
	//
	if (millisecondStart >= 0) {
	    millisecondStart = findMillisecondStart(now, cache.toString(), formatter);
	}

	return buf;
    }

    /**
     * Formats a count of milliseconds (0-999) into a numeric representation.
     * 
     * @param millis Millisecond coun between 0 and 999.
     * @param buf    String buffer, may not be null.
     * @param offset Starting position in buffer, the length of the buffer must be
     *               at least offset + 3.
     */
    private static void millisecondFormat(final int millis, final StringBuffer buf, final int offset) {
	buf.setCharAt(offset, DIGITS.charAt(millis / 100));
	buf.setCharAt(offset + 1, DIGITS.charAt((millis / 10) % 10));
	buf.setCharAt(offset + 2, DIGITS.charAt(millis % 10));
    }

    /**
     * Set timezone.
     *
     * Setting the timezone using getCalendar().setTimeZone() will likely cause
     * caching to misbehave.
     * 
     * @param timeZone TimeZone new timezone
     */
    public void setTimeZone(final TimeZone timeZone) {
	formatter.setTimeZone(timeZone);
	previousTime = Long.MIN_VALUE;
	slotBegin = Long.MIN_VALUE;
    }

    /**
     * This method is delegated to the formatter which most likely returns null.
     * 
     * @param s   string representation of date.
     * @param pos field position, unused.
     * @return parsed date, likely null.
     */
    public Date parse(String s, ParsePosition pos) {
	return formatter.parse(s, pos);
    }

    /**
     * Gets number formatter.
     *
     * @return NumberFormat number formatter
     */
    public NumberFormat getNumberFormat() {
	return formatter.getNumberFormat();
    }

    /**
     * Gets maximum cache validity for the specified SimpleDateTime conversion
     * pattern.
     * 
     * @param pattern conversion pattern, may not be null.
     * @return Duration in milliseconds from an integral second that the cache will
     *         return consistent results.
     */
    public static int getMaximumCacheValidity(final String pattern) {
	//
	// If there are more "S" in the pattern than just one "SSS" then
	// (for example, "HH:mm:ss,SSS SSS"), then set the expiration to
	// one millisecond which should only perform duplicate request caching.
	//
	int firstS = pattern.indexOf('S');

	if ((firstS >= 0) && (firstS != pattern.lastIndexOf("SSS"))) {
	    return 1;
	}

	return 1000;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy