org.apache.hadoop.hive.common.CopyOnFirstWriteProperties Maven / Gradle / Ivy
The 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.hadoop.hive.common;
import com.google.common.collect.Interner;
import com.google.common.collect.Interners;
import org.apache.hive.common.util.SuppressFBWarnings;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintStream;
import java.io.PrintWriter;
import java.io.Reader;
import java.lang.reflect.Field;
import java.util.Collection;
import java.util.Enumeration;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.function.BiConsumer;
import java.util.function.BiFunction;
import java.util.function.Function;
/**
* A special subclass of Properties, designed to save memory when many identical
* copies of Properties would otherwise be created. To achieve that, we use the
* 'interned' field, which points to the same Properties object for all instances
* of CopyOnFirstWriteProperties that were created with identical contents.
* However, as soon as any mutating method is called, contents are copied from
* the 'interned' properties into this instance.
*/
public class CopyOnFirstWriteProperties extends Properties {
private Properties interned;
private static Interner INTERNER = Interners.newWeakInterner();
private static Field defaultsField;
static {
try {
defaultsField = Properties.class.getDeclaredField("defaults");
defaultsField.setAccessible(true);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
public CopyOnFirstWriteProperties(Properties p) {
setInterned(p);
}
/************* Public API of java.util.Properties ************/
@Override
@SuppressFBWarnings(value = "IS2_INCONSISTENT_SYNC", justification = "Intended")
public String getProperty(String key) {
if (interned != null) return interned.getProperty(key);
else return super.getProperty(key);
}
@Override
@SuppressFBWarnings(value = {"IS2_INCONSISTENT_SYNC", "UG_SYNC_SET_UNSYNC_GET"}, justification = "Intended")
public String getProperty(String key, String defaultValue) {
if (interned != null) return interned.getProperty(key, defaultValue);
else return super.getProperty(key, defaultValue);
}
@Override
@SuppressFBWarnings(value = "IS2_INCONSISTENT_SYNC", justification = "Intended")
public void list(PrintStream out) {
if (interned != null) interned.list(out);
else super.list(out);
}
@Override
@SuppressFBWarnings(value = "IS2_INCONSISTENT_SYNC", justification = "Intended")
public void list(PrintWriter out) {
if (interned != null) interned.list(out);
else super.list(out);
}
@Override
public synchronized void load(InputStream inStream) throws IOException {
if (interned != null) copyFromInternedToThis();
super.load(inStream);
}
@Override
public synchronized void load(Reader reader) throws IOException {
if (interned != null) copyFromInternedToThis();
super.load(reader);
}
@Override
public synchronized void loadFromXML(InputStream inStream) throws IOException {
if (interned != null) copyFromInternedToThis();
super.loadFromXML(inStream);
}
@Override
public Enumeration> propertyNames() {
if (interned != null) return interned.propertyNames();
else return super.propertyNames();
}
@Override
public synchronized Object setProperty(String key, String value) {
if (interned != null) copyFromInternedToThis();
return super.setProperty(key, value);
}
@Override
public void store(OutputStream out, String comments) throws IOException {
if (interned != null) interned.store(out, comments);
else super.store(out, comments);
}
@Override
public void storeToXML(OutputStream os, String comment) throws IOException {
if (interned != null) interned.storeToXML(os, comment);
else super.storeToXML(os, comment);
}
@Override
public void storeToXML(OutputStream os, String comment, String encoding)
throws IOException {
if (interned != null) interned.storeToXML(os, comment, encoding);
else super.storeToXML(os, comment, encoding);
}
@Override
public Set stringPropertyNames() {
if (interned != null) return interned.stringPropertyNames();
else return super.stringPropertyNames();
}
/************* Public API of java.util.Hashtable ************/
@Override
public synchronized void clear() {
if (interned != null) copyFromInternedToThis();
super.clear();
}
@Override
public synchronized Object clone() {
if (interned != null) return new CopyOnFirstWriteProperties(interned);
else return super.clone();
}
@Override
public synchronized Object compute(Object key, BiFunction remappingFunction) {
if (interned != null) copyFromInternedToThis(); // We do this because if function returns null,
// the mapping for key is removed, i.e. the table is mutated.
return super.compute(key, remappingFunction);
}
@Override
public synchronized Object computeIfAbsent(Object key, Function mappingFunction) {
if (interned != null) copyFromInternedToThis();
return super.computeIfAbsent(key, mappingFunction);
}
@Override
public synchronized Object computeIfPresent(Object key, BiFunction remappingFunction) {
if (interned != null) copyFromInternedToThis();
return super.computeIfPresent(key, remappingFunction);
}
@Override
public synchronized boolean contains(Object value) {
if (interned != null) return interned.contains(value);
else return super.contains(value);
}
@Override
public synchronized boolean containsKey(Object key) {
if (interned != null) return interned.containsKey(key);
else return super.containsKey(key);
}
@Override
public synchronized boolean containsValue(Object value) {
if (interned != null) return interned.containsValue(value);
else return super.containsValue(value);
}
@Override
public synchronized Enumeration
© 2015 - 2025 Weber Informatics LLC | Privacy Policy