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

org.protempa.proposition.value.NominalValue Maven / Gradle / Ivy

There is a newer version: 5.2-Alpha-2
Show newest version
/*
 * #%L
 * Protempa Framework
 * %%
 * Copyright (C) 2012 - 2013 Emory University
 * %%
 * 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.
 * #L%
 */
package org.protempa.proposition.value;

import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.text.Format;
import java.util.Map;
import org.apache.commons.collections4.map.ReferenceMap;
import org.apache.commons.lang3.builder.ToStringBuilder;

/**
 * Represents a {@link String} value.
 *
 * @author Andrew Post
 */
public final class NominalValue implements Value, Comparable,
        Serializable {

    private static final long serialVersionUID = 440118249272295573L;

    private String val;
    private transient volatile int hashCode;

    private static final Map cache = new ReferenceMap<>();

    /**
     * Creates a new nominal value. If val's length is less than 20
     * characters, the created {@link NominalValue} object will be cached and
     * reused.
     *
     * @param val a {@link String}. If null, {@link NominalValue}
     * with "" will be created.
     * @return a {@link NominalValue}.
     */
    public static NominalValue getInstance(String val) {
        if (val != null && val.length() < 20) {
            NominalValue result;
            synchronized (cache) {
                result = cache.get(val);
                if (result == null) {
                    result = new NominalValue(val);
                    cache.put(val, result);
                }
            }
            return result;
        } else {
            return new NominalValue(val);
        }
    }

    /**
     * Creates a new nominal value with the given value. The recommended way to
     * create nominal values is with {@link #getInstance(java.lang.String) }.
     *
     * @param val a String. If null, the default
     * string is used ("").
     */
    public NominalValue(String val) {
        if (val != null) {
            this.val = val;
        } else {
            this.val = "";
        }
    }

    @Override
    public NominalValue replace() {
        NominalValue result;
        synchronized (cache) {
            result = cache.get(this.val);
        }
        if (result != null) {
            return result;
        } else {
            return this;
        }
    }

    @Override
    public int hashCode() {
        if (this.hashCode == 0) {
            this.hashCode = val.hashCode();
        }
        return this.hashCode;
    }

    /**
     * Compares this nominal value and another for string equality.
     *
     * @param obj another object.
     * @return true if the two nominal values' strings are equal,
     * false if not.
     */
    @Override
    public boolean equals(Object obj) {
        if (this == obj) {
            return true;
        }
        if (obj == null) {
            return false;
        }
        if (getClass() != obj.getClass()) {
            return false;
        }
        final NominalValue other = (NominalValue) obj;
        return this.val.equals(other.val);
    }

    /**
     * Gets the value as a Java {@link String}.
     *
     * @return a {@link String}.
     */
    public String getString() {
        return val;
    }

    /**
     * Returns the same thing as {@link #getString}.
     */
    @Override
    public String getFormatted() {
        return val;
    }
    
    @Override
    public String format(Format format) {
        if (format == null) {
            return getFormatted();
        } else {
            return format.format(this.val);
        }
    }

    @Override
    public ValueType getType() {
        return ValueType.NOMINALVALUE;
    }

    /**
     * Compares this nominal value and another lexically. If the argument is a
     * {@link ListValue}, it checks this string for membership in the list.
     *
     * @param o another {@link Value}.
     * @return If the provided value is a {@link NominalValue}, returns null     {@link ValueComparator#GREATER_THAN},
     * {@link ValueComparator#LESS_THAN} or {@link ValueComparator#EQUAL_TO}
     * depending on whether this nominal value is lexically greater than, less
     * than or equal to the nominal value provided as argument. If the provided
     * value is a {@link ValueList}, returns {@link ValueComparator#IN} if this
     * object is a member of the list, or {@link ValueComparator#NOT_IN} if not.
     * Otherwise, returns {@link ValueComparator#UNKNOWN}.
     */
    @Override
    public ValueComparator compare(Value o) {
        if (o == null) {
            return ValueComparator.NOT_EQUAL_TO;
        }
        switch (o.getType()) {
            case NOMINALVALUE:
                NominalValue other = (NominalValue) o;
                int comp = compareTo(other);
                return comp > 0 ? ValueComparator.GREATER_THAN
                        : (comp < 0 ? ValueComparator.LESS_THAN
                                : ValueComparator.EQUAL_TO);
            case VALUELIST:
                ValueList vl = (ValueList) o;
                return vl.contains(this) ? ValueComparator.IN : ValueComparator.NOT_IN;
            default:
                return ValueComparator.NOT_EQUAL_TO;
        }

    }

    @Override
    public void accept(ValueVisitor valueVisitor) {
        if (valueVisitor == null) {
            throw new IllegalArgumentException("valueVisitor cannot be null");
        }
        valueVisitor.visit(this);
    }

    @Override
    public String toString() {
        return ToStringBuilder.reflectionToString(this);
    }

    private void writeObject(ObjectOutputStream s) throws IOException {
        s.writeObject(this.val);
    }

    private void readObject(ObjectInputStream s) throws IOException,
            ClassNotFoundException {
        String tmpVal = (String) s.readObject();
        this.val = tmpVal != null ? tmpVal : "";
        if (tmpVal != null && tmpVal.length() < 20) {
            if (!cache.containsKey(tmpVal)) {
                cache.put(tmpVal, this);
            }
        }
    }

    @Override
    public int compareTo(NominalValue t) {
        return this.val.compareTo(t.val);
    }

    @Override
    public NominalValueBuilder asBuilder() {
        return new NominalValueBuilder(this);
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy