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

src.com.android.server.utils.WatchedLongSparseArray Maven / Gradle / Ivy

Go to download

A library jar that provides APIs for Applications written for the Google Android Platform.

There is a newer version: 15-robolectric-12650502
Show newest version
/*
 * Copyright (C) 2020 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.utils;

import android.annotation.NonNull;
import android.annotation.Nullable;
import android.util.LongSparseArray;

/**
 * A watched variant of {@link LongSparseArray}.  If a {@link Watchable} is stored in the
 * array, the array registers with the {@link Watchable}.  The array registers only once
 * with each {@link Watchable} no matter how many times the {@link Watchable} is stored in
 * the array.
 * @param  The element type, stored in the array.
 */
public class WatchedLongSparseArray extends WatchableImpl
        implements Snappable {

    // The storage
    private final LongSparseArray mStorage;

    // If true, the array is watching its children
    private volatile boolean mWatching = false;

    // The local observer
    private final Watcher mObserver = new Watcher() {
            @Override
            public void onChange(@Nullable Watchable o) {
                WatchedLongSparseArray.this.dispatchChange(o);
            }
        };

    /**
     * A private convenience function that notifies registered listeners that an element
     * has been added to or removed from the array.  The what parameter is the array itself.
     */
    private void onChanged() {
        dispatchChange(this);
    }

    /**
     * A convenience function.  Register the object if it is {@link Watchable} and if the
     * array is currently watching.  Note that the watching flag must be true if this
     * function is to succeed.
     */
    private void registerChild(Object o) {
        if (mWatching && o instanceof Watchable) {
            ((Watchable) o).registerObserver(mObserver);
        }
    }

    /**
     * A convenience function.  Unregister the object if it is {@link Watchable} and if
     * the array is currently watching.  Note that the watching flag must be true if this
     * function is to succeed.
     */
    private void unregisterChild(Object o) {
        if (mWatching && o instanceof Watchable) {
            ((Watchable) o).unregisterObserver(mObserver);
        }
    }

    /**
     * A convenience function.  Unregister the object if it is {@link Watchable}, if the array is
     * currently watching, and if the storage does not contain the object.  Note that the watching
     * flag must be true if this function is to succeed.  This must be called after an object has
     * been removed from the storage.
     */
    private void unregisterChildIf(Object o) {
        if (mWatching && o instanceof Watchable) {
            if (mStorage.indexOfValue((E) o) == -1) {
                ((Watchable) o).unregisterObserver(mObserver);
            }
        }
    }

    /**
     * Register a {@link Watcher} with the array.  If this is the first Watcher than any
     * array values that are {@link Watchable} are registered to the array itself.
     */
    @Override
    public void registerObserver(@NonNull Watcher observer) {
        super.registerObserver(observer);
        if (registeredObserverCount() == 1) {
            // The watching flag must be set true before any children are registered.
            mWatching = true;
            final int end = mStorage.size();
            for (int i = 0; i < end; i++) {
                registerChild(mStorage.valueAt(i));
            }
        }
    }

    /**
     * Unregister a {@link Watcher} from the array.  If this is the last Watcher than any
     * array values that are {@link Watchable} are unregistered to the array itself.
     */
    @Override
    public void unregisterObserver(@NonNull Watcher observer) {
        super.unregisterObserver(observer);
        if (registeredObserverCount() == 0) {
            final int end = mStorage.size();
            for (int i = 0; i < end; i++) {
                unregisterChild(mStorage.valueAt(i));
            }
            // The watching flag must be true while children are unregistered.
            mWatching = false;
        }
    }

    /**
     * Creates a new WatchedSparseArray containing no mappings.
     */
    public WatchedLongSparseArray() {
        mStorage = new LongSparseArray();
    }

    /**
     * Creates a new WatchedSparseArray containing no mappings that
     * will not require any additional memory allocation to store the
     * specified number of mappings.  If you supply an initial capacity of
     * 0, the sparse array will be initialized with a light-weight
     * representation not requiring any additional array allocations.
     */
    public WatchedLongSparseArray(int initialCapacity) {
        mStorage = new LongSparseArray(initialCapacity);
    }

    /**
     * Create a {@link WatchedLongSparseArray} from a {@link LongSparseArray}.
     */
    public WatchedLongSparseArray(@NonNull LongSparseArray c) {
        mStorage = c.clone();
    }

    /**
     * The copy constructor does not copy the watcher data.
     */
    public WatchedLongSparseArray(@NonNull WatchedLongSparseArray r) {
        mStorage = r.mStorage.clone();
    }

    /**
     * Make  a copy of src.  Any data in  is discarded.
     */
    public void copyFrom(@NonNull LongSparseArray src) {
        clear();
        final int end = src.size();
        for (int i = 0; i < end; i++) {
            put(src.keyAt(i), src.valueAt(i));
        }
    }

    /**
     * Make dst a copy of .  Any previous data in dst is discarded.
     */
    public void copyTo(@NonNull LongSparseArray dst) {
        dst.clear();
        final int end = size();
        for (int i = 0; i < end; i++) {
            dst.put(keyAt(i), valueAt(i));
        }
    }

    /**
     * Return the underlying storage.  This breaks the wrapper but is necessary when
     * passing the array to distant methods.
     */
    public LongSparseArray untrackedStorage() {
        return mStorage;
    }

    /**
     * Gets the Object mapped from the specified key, or null
     * if no such mapping has been made.
     */
    public E get(long key) {
        return mStorage.get(key);
    }

    /**
     * Gets the Object mapped from the specified key, or the specified Object
     * if no such mapping has been made.
     */
    public E get(long key, E valueIfKeyNotFound) {
        return mStorage.get(key, valueIfKeyNotFound);
    }

    /**
     * Removes the mapping from the specified key, if there was any.
     */
    public void delete(long key) {
        E old = mStorage.get(key, null);
        mStorage.delete(key);
        unregisterChildIf(old);
        onChanged();

    }

    /**
     * Alias for {@link #delete(long)}.
     */
    public void remove(long key) {
        delete(key);
    }

    /**
     * Removes the mapping at the specified index.
     *
     * 

For indices outside of the range 0...size()-1, an * {@link ArrayIndexOutOfBoundsException} is thrown.

*/ public void removeAt(int index) { E old = mStorage.valueAt(index); mStorage.removeAt(index); unregisterChildIf(old); onChanged(); } /** * Adds a mapping from the specified key to the specified value, * replacing the previous mapping from the specified key if there * was one. */ public void put(long key, E value) { E old = mStorage.get(key); mStorage.put(key, value); unregisterChildIf(old); registerChild(value); onChanged(); } /** * Returns the number of key-value mappings that this LongSparseArray * currently stores. */ public int size() { return mStorage.size(); } /** * Given an index in the range 0...size()-1, returns * the key from the indexth key-value mapping that this * LongSparseArray stores. * *

The keys corresponding to indices in ascending order are guaranteed to * be in ascending order, e.g., keyAt(0) will return the * smallest key and keyAt(size()-1) will return the largest * key.

* *

For indices outside of the range 0...size()-1, an * {@link ArrayIndexOutOfBoundsException} is thrown.

*/ public long keyAt(int index) { return mStorage.keyAt(index); } /** * Given an index in the range 0...size()-1, returns * the value from the indexth key-value mapping that this * LongSparseArray stores. * *

The values corresponding to indices in ascending order are guaranteed * to be associated with keys in ascending order, e.g., * valueAt(0) will return the value associated with the * smallest key and valueAt(size()-1) will return the value * associated with the largest key.

* *

For indices outside of the range 0...size()-1, an * {@link ArrayIndexOutOfBoundsException} is thrown.

*/ public E valueAt(int index) { return mStorage.valueAt(index); } /** * Given an index in the range 0...size()-1, sets a new * value for the indexth key-value mapping that this * LongSparseArray stores. * *

For indices outside of the range 0...size()-1, an * {@link ArrayIndexOutOfBoundsException} is thrown.

*/ public void setValueAt(int index, E value) { E old = mStorage.valueAt(index); mStorage.setValueAt(index, value); unregisterChildIf(old); registerChild(value); onChanged(); } /** * Returns the index for which {@link #keyAt} would return the * specified key, or a negative number if the specified * key is not mapped. */ public int indexOfKey(long key) { return mStorage.indexOfKey(key); } /** * Returns an index for which {@link #valueAt} would return the * specified key, or a negative number if no keys map to the * specified value. * Beware that this is a linear search, unlike lookups by key, * and that multiple keys can map to the same value and this will * find only one of them. */ public int indexOfValue(E value) { return mStorage.indexOfValue(value); } /** * Returns an index for which {@link #valueAt} would return the * specified key, or a negative number if no keys map to the * specified value. *

Beware that this is a linear search, unlike lookups by key, * and that multiple keys can map to the same value and this will * find only one of them. *

Note also that this method uses {@code equals} unlike {@code indexOfValue}. * @hide */ public int indexOfValueByValue(E value) { return mStorage.indexOfValueByValue(value); } /** * Removes all key-value mappings from this LongSparseArray. */ public void clear() { final int end = mStorage.size(); for (int i = 0; i < end; i++) { unregisterChild(mStorage.valueAt(i)); } mStorage.clear(); onChanged(); } /** * Puts a key/value pair into the array, optimizing for the case where * the key is greater than all existing keys in the array. */ public void append(long key, E value) { mStorage.append(key, value); registerChild(value); onChanged(); } /** * {@inheritDoc} * *

This implementation composes a string by iterating over its mappings. If * this map contains itself as a value, the string "(this Map)" * will appear in its place. */ @Override public String toString() { return mStorage.toString(); } /** * Create a copy of the array. If the element is a subclass of Snapper then the copy * contains snapshots of the elements. Otherwise the copy contains references to the * elements. The returned snapshot is immutable. * @return A new array whose elements are the elements of . */ public WatchedLongSparseArray snapshot() { WatchedLongSparseArray l = new WatchedLongSparseArray<>(size()); snapshot(l, this); return l; } /** * Make a snapshot of the argument. Note that is immutable when the * method returns. must be empty when the function is called. * @param r The source array, which is copied into */ public void snapshot(@NonNull WatchedLongSparseArray r) { snapshot(this, r); } /** * Make the destination a copy of the source. If the element is a subclass of Snapper then the * copy contains snapshots of the elements. Otherwise the copy contains references to the * elements. The destination must be initially empty. Upon return, the destination is * immutable. * @param dst The destination array. It must be empty. * @param src The source array. It is not modified. */ public static void snapshot(@NonNull WatchedLongSparseArray dst, @NonNull WatchedLongSparseArray src) { if (dst.size() != 0) { throw new IllegalArgumentException("snapshot destination is not empty"); } final int end = src.size(); for (int i = 0; i < end; i++) { final E val = Snapshots.maybeSnapshot(src.valueAt(i)); final long key = src.keyAt(i); dst.put(key, val); } dst.seal(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy