blob: 8bd10359bc6ca7ca8125b1e89c0f7465a1b4dc18 [file] [log] [blame]
/*
* Copyright (C) 2019 The Android Open Source Project
*
* 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.android.server.timezonedetector;
import android.annotation.IntRange;
import android.annotation.NonNull;
import android.annotation.Nullable;
import com.android.internal.util.IndentingPrintWriter;
import java.util.LinkedList;
/**
* A class that behaves like the following definition, except it stores the history of values set
* that can be dumped for debugging with {@link #dump(IndentingPrintWriter)}.
*
* <pre>{@code
* private static class Ref<V> {
* private V mValue;
*
* public V get() {
* return mValue;
* }
*
* public V set(V value) {
* V previous = mValue;
* mValue = value;
* return previous;
* }
* }
* }</pre>
*
* <p>This class is not thread-safe.
*
* @param <V> the type of the value
*/
public final class ReferenceWithHistory<V> {
/** The size the history linked list is allowed to grow to. */
private final int mMaxHistorySize;
@Nullable
private LinkedList<V> mValues;
/**
* Creates an instance that records, at most, the specified number of values.
*/
public ReferenceWithHistory(@IntRange(from = 1) int maxHistorySize) {
if (maxHistorySize < 1) {
throw new IllegalArgumentException("maxHistorySize < 1: " + maxHistorySize);
}
this.mMaxHistorySize = maxHistorySize;
}
/** Returns the current value, or {@code null} if it has never been set. */
@Nullable
public V get() {
return (mValues == null || mValues.isEmpty()) ? null : mValues.getFirst();
}
/** Sets the current value. Returns the previous value, or {@code null}. */
@Nullable
public V set(@Nullable V newValue) {
if (mValues == null) {
mValues = new LinkedList<>();
}
V previous = get();
mValues.addFirst(newValue);
if (mValues.size() > mMaxHistorySize) {
mValues.removeLast();
}
return previous;
}
/**
* Dumps the content of the reference, including historic values, using the supplied writer.
*/
public void dump(@NonNull IndentingPrintWriter ipw) {
if (mValues == null) {
ipw.println("{Empty}");
} else {
int i = 0;
for (V value : mValues) {
ipw.println(i + ": " + value);
i++;
}
}
ipw.flush();
}
/**
* Returns the number of historic entries stored currently.
*/
public int getHistoryCount() {
return mValues == null ? 0 : mValues.size();
}
@Override
public String toString() {
return String.valueOf(get());
}
}