ucar.nc2.dataset.CoordinateAxis Maven / Gradle / Ivy
The newest version!
/*
* Copyright (c) 1998-2018 John Caron and University Corporation for Atmospheric Research/Unidata
* See LICENSE for license information.
*/
package ucar.nc2.dataset;
import ucar.ma2.*;
import ucar.nc2.*;
import ucar.nc2.constants.AxisType;
import ucar.nc2.constants.CDM;
import ucar.nc2.constants.CF;
import ucar.nc2.constants._Coordinate;
import ucar.nc2.dataset.conv.CF1Convention;
import ucar.nc2.dataset.conv.COARDSConvention;
import ucar.nc2.time.Calendar;
import java.io.IOException;
import java.util.Formatter;
/**
* A Coordinate Axis is a Variable that specifies one of the coordinates of a CoordinateSystem.
* Mathematically it is a scalar function F from index space to S:
*
*
* F:D -> S
* where D is a product set of dimensions (aka index space), and S is the set of reals (R) or Strings.
*
*
* If its element type is char, it is considered a string-valued Coordinate Axis and rank is reduced by one,
* since the outermost dimension is considered the string length: v(i, j, .., strlen).
* If its element type is String, it is a string-valued Coordinate Axis.
* Otherwise it is numeric-valued, and isNumeric() is true.
*
* The one-dimensional case F(i) -> R is the common case which affords important optimizations.
* In that case, use the subtype CoordinateAxis1D. The factory methods will return
* either a CoordinateAxis1D if the variable is one-dimensional, a CoordinateAxis2D if its 2D, or a
* CoordinateAxis for the general case.
*
* A CoordinateAxis is optionally marked as georeferencing with an AxisType. It should have
* a units string and optionally a description string.
*
* A Structure cannot be a CoordinateAxis, although members of Structures can.
*
* @author john caron
*/
public class CoordinateAxis extends VariableDS {
private static org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(CoordinateAxis.class);
private static int axisSizeToCache = 100 * 1000; // bytes
/**
* Create a coordinate axis from an existing Variable.
*
* @param ncd the containing dataset
* @param vds an existing Variable in dataset.
* @return CoordinateAxis or one of its subclasses (CoordinateAxis1D, CoordinateAxis2D, or CoordinateAxis1DTime).
* @deprecated Use CoordinateAxis.fromVariableDS()
*/
@Deprecated
public static CoordinateAxis factory(NetcdfDataset ncd, VariableDS vds) {
if ((vds.getRank() == 0) || (vds.getRank() == 1) || (vds.getRank() == 2 && vds.getDataType() == DataType.CHAR)) {
return new CoordinateAxis1D(ncd, vds);
} else if (vds.getRank() == 2)
return new CoordinateAxis2D(ncd, vds);
else
return new CoordinateAxis(ncd, vds);
}
/**
* Create a coordinate axis from an existing Variable.Builder.
*
* @param vdsBuilder an existing Variable in dataset.
* @return CoordinateAxis or one of its subclasses (CoordinateAxis1D, CoordinateAxis2D, or CoordinateAxis1DTime).
*/
public static CoordinateAxis.Builder fromVariableDS(VariableDS.Builder> vdsBuilder) {
if ((vdsBuilder.getRank() == 0) || (vdsBuilder.getRank() == 1)
|| (vdsBuilder.getRank() == 2 && vdsBuilder.dataType == DataType.CHAR)) {
return CoordinateAxis1D.builder().copyFrom(vdsBuilder);
} else if (vdsBuilder.getRank() == 2) {
return CoordinateAxis2D.builder().copyFrom(vdsBuilder);
} else {
return CoordinateAxis.builder().copyFrom(vdsBuilder);
}
}
/**
* Create a coordinate axis from an existing Variable.
* General case.
*
* @param ncd the containing dataset
* @param vds an existing Variable
* @deprecated Use CoordinateAxis.builder()
*/
@Deprecated
protected CoordinateAxis(NetcdfDataset ncd, VariableDS vds) {
super(vds, false);
this.ncd = ncd;
if (vds instanceof CoordinateAxis) {
CoordinateAxis axis = (CoordinateAxis) vds;
this.axisType = axis.axisType;
this.boundaryRef = axis.boundaryRef;
this.isContiguous = axis.isContiguous;
this.positive = axis.positive;
}
setSizeToCache(axisSizeToCache);
}
/**
* Constructor when theres no underlying variable. You better set the values too!
*
* @param ds the containing dataset.
* @param group the containing group; if null, use rootGroup
* @param shortName axis name.
* @param dataType data type
* @param dims list of dimension names
* @param units units of coordinates, preferably udunit compatible.
* @param desc long name.
* @deprecated Use CoordinateAxis.builder()
*/
@Deprecated
public CoordinateAxis(NetcdfDataset ds, Group group, String shortName, DataType dataType, String dims, String units,
String desc) {
super(ds, group, null, shortName, dataType, dims, units, desc);
this.ncd = ds;
setSizeToCache(axisSizeToCache);
}
/**
* Make a copy, with an independent cache.
*
* @return copy of this CoordinateAxis
*/
public CoordinateAxis copyNoCache() {
CoordinateAxis axis = new CoordinateAxis(ncd, getParentGroupOrRoot(), getShortName(), getDataType(),
getDimensionsString(), getUnitsString(), getDescription());
// other state
axis.axisType = this.axisType;
axis.boundaryRef = this.boundaryRef;
axis.isContiguous = this.isContiguous;
axis.positive = this.positive;
axis.cache.reset(); // decouple cache
return axis;
}
// for section and slice
/** @deprecated Use {@link #toBuilder()} */
@Deprecated
@Override
protected CoordinateAxis copy() {
return new CoordinateAxis(this.ncd, this);
}
/**
* Get type of axis
*
* @return type of axis, or null if none.
*/
public AxisType getAxisType() {
return axisType;
}
/**
* Set type of axis, or null if none. Default is none.
*
* @param axisType set to this value
* @deprecated Use CoordinateAxis.builder()
*/
@Deprecated
public void setAxisType(AxisType axisType) {
this.axisType = axisType;
}
@Override
public String getUnitsString() {
String units = super.getUnitsString();
return units == null ? "" : units;
}
/**
* Does the axis have numeric values.
*
* @return true if the CoordAxis is numeric, false if its string valued ("nominal").
*/
public boolean isNumeric() {
return (getDataType() != DataType.CHAR) && (getDataType() != DataType.STRING)
&& (getDataType() != DataType.STRUCTURE);
}
/**
* If the edges are contiguous or disjoint
* Caution: many datasets do not explicitly specify this info, this is often a guess; default is true.
*
* @return true if the edges are contiguous or false if disjoint. Assumed true unless set otherwise.
*/
public boolean isContiguous() {
return isContiguous;
}
/**
* An interval coordinate consists of two numbers, bound1 and bound2.
* The coordinate value must lie between them, but otherwise is somewhat arbitrary.
* If not interval, then it has one number, the coordinate value.
*
* @return true if its an interval coordinate.
*/
public boolean isInterval() {
return false; // interval detection is done in subclasses
}
public boolean isIndependentCoordinate() {
if (isCoordinateVariable())
return true;
return attributes.hasAttribute(_Coordinate.AliasForDimension);
}
/*
* Set if the edges are contiguous or disjoint.
*
* @param isContiguous true if the adjacent edges touch
*
* protected void setContiguous(boolean isContiguous) {
* this.isContiguous = isContiguous;
* }
*/
/**
* Get the direction of increasing values, used only for vertical Axes.
*
* @return POSITIVE_UP, POSITIVE_DOWN, or null if unknown.
*/
public String getPositive() {
return positive;
}
/**
* Set the direction of increasing values, used only for vertical Axes.
*
* @param positive POSITIVE_UP, POSITIVE_DOWN, or null if you dont know..
* @deprecated Use CoordinateAxis.builder()
*/
@Deprecated
public void setPositive(String positive) {
this.positive = positive;
}
/**
* The name of this coordinate axis' boundary variable
*
* @return the name of this coordinate axis' boundary variable, or null if none.
*/
public String getBoundaryRef() {
return boundaryRef;
}
/**
* Set a reference to a boundary variable.
*
* @param boundaryRef the name of a boundary coordinate variable in the same dataset.
* @deprecated Use CoordinateAxis.builder()
*/
@Deprecated
public void setBoundaryRef(String boundaryRef) {
this.boundaryRef = boundaryRef;
}
////////////////////////////////
private MAMath.MinMax minmax;
private void init() {
try {
Array data = read();
minmax = MAMath.getMinMax(data);
} catch (IOException ioe) {
log.error("Error reading coordinate values ", ioe);
throw new IllegalStateException(ioe);
}
}
/**
* The smallest coordinate value. Only call if isNumeric.
*
* @return the minimum coordinate value
*/
public double getMinValue() {
if (minmax == null)
init();
return minmax.min;
}
/**
* The largest coordinate value. Only call if isNumeric.
*
* @return the maximum coordinate value
*/
public double getMaxValue() {
if (minmax == null)
init();
return minmax.max;
}
//////////////////////
/**
* Get a string representation
*
* @param buf place info here
*/
public void getInfo(Formatter buf) {
buf.format("%-30s", getNameAndDimensions());
buf.format("%-20s", getUnitsString());
if (axisType != null) {
buf.format("%-10s", axisType.toString());
}
buf.format("%s", getDescription());
/*
* if (isNumeric) {
* boolean debugCoords = ucar.util.prefs.ui.Debug.isSet("Dataset/showCoordValues");
* int ndigits = debugCoords ? 9 : 4;
* for (int i=0; i< getNumElements(); i++) {
* buf.append(Format.d(getCoordValue(i), ndigits));
* buf.append(" ");
* }
* if (debugCoords) {
* buf.append("\n ");
* for (int i=0; i<=getNumElements(); i++) {
* buf.append(Format.d(getCoordEdge(i), ndigits));
* buf.append(" ");
* }
* }
* } else {
* for (int i=0; i< getNumElements(); i++) {
* buf.append(getCoordName(i));
* buf.append(" ");
* }
* }
*/
// buf.append("\n");
}
/**
* Standard sort on Coordinate Axes
*/
public static class AxisComparator implements java.util.Comparator {
public int compare(CoordinateAxis c1, CoordinateAxis c2) {
AxisType t1 = c1.getAxisType();
AxisType t2 = c2.getAxisType();
if ((t1 == null) && (t2 == null))
return c1.getShortName().compareTo(c2.getShortName());
if (t1 == null)
return -1;
if (t2 == null)
return 1;
return t1.axisOrder() - t2.axisOrder();
}
}
/**
* Instances which have same content are equal.
*/
public boolean equals(Object oo) {
if (this == oo)
return true;
if (!(oo instanceof CoordinateAxis))
return false;
if (!super.equals(oo))
return false;
CoordinateAxis o = (CoordinateAxis) oo;
if (getAxisType() != null)
if (getAxisType() != o.getAxisType())
return false;
if (getPositive() != null)
return getPositive().equals(o.getPositive());
return true;
}
/**
* Override Object.hashCode() to implement equals.
*/
public int hashCode() {
int result = super.hashCode();
if (getAxisType() != null)
result = 37 * result + getAxisType().hashCode();
if (getPositive() != null)
result = 37 * result + getPositive().hashCode();
return result;
}
/////////////////////////////////////
// needed by time coordinates
public ucar.nc2.time.Calendar getCalendarFromAttribute() {
String cal = attributes.findAttributeString(CF.CALENDAR, null);
if (cal == null) { // default for CF and COARDS
Attribute convention = (ncd == null) ? null : ncd.getRootGroup().findAttribute(CDM.CONVENTIONS);
if (convention != null && convention.isString()) {
String hasName = convention.getStringValue();
int version = CF1Convention.getVersion(hasName);
if (version >= 0) {
return Calendar.gregorian;
// if (version < 7 ) return Calendar.gregorian;
// if (version >= 7 ) return Calendar.proleptic_gregorian; //
}
if (COARDSConvention.isMine(hasName))
return Calendar.gregorian;
}
}
return ucar.nc2.time.Calendar.get(cal);
}
////////////////////////////////////////////////////////////////////////////////////////////
// TODO make these final and immutable in 6.
protected NetcdfDataset ncd; // needed?
protected AxisType axisType;
protected String positive;
protected String boundaryRef;
protected boolean isContiguous = true;
protected CoordinateAxis(Builder> builder, Group parentGroup) {
super(builder, parentGroup);
this.ncd = (NetcdfDataset) this.ncfile;
this.axisType = builder.axisType;
this.positive = builder.positive;
this.boundaryRef = builder.boundaryRef;
this.isContiguous = builder.isContiguous;
}
public Builder> toBuilder() {
return addLocalFieldsToBuilder(builder());
}
// Add local fields to the passed - in builder.
protected Builder> addLocalFieldsToBuilder(Builder extends Builder>> b) {
b.setAxisType(this.axisType).setPositive(this.positive).setBoundary(this.boundaryRef)
.setIsContiguous(this.isContiguous);
return (Builder>) super.addLocalFieldsToBuilder(b);
}
/**
* Get Builder for this class that allows subclassing.
*
* @see "https://community.oracle.com/blogs/emcmanus/2010/10/24/using-builder-pattern-subclasses"
*/
public static Builder> builder() {
return new Builder2();
}
private static class Builder2 extends Builder {
@Override
protected Builder2 self() {
return this;
}
}
public static abstract class Builder> extends VariableDS.Builder {
public AxisType axisType;
protected String positive;
protected String boundaryRef;
protected boolean isContiguous = true;
private boolean built;
protected abstract T self();
public T setAxisType(AxisType axisType) {
this.axisType = axisType;
return self();
}
public T setPositive(String positive) {
this.positive = positive;
return self();
}
public T setBoundary(String boundaryRef) {
this.boundaryRef = boundaryRef;
return self();
}
public T setIsContiguous(boolean isContiguous) {
this.isContiguous = isContiguous;
return self();
}
@Override
public T copyFrom(VariableDS.Builder> vds) {
super.copyFrom(vds);
return self();
}
public CoordinateAxis build(Group parentGroup) {
if (built)
throw new IllegalStateException("already built");
built = true;
return new CoordinateAxis(this, parentGroup);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy