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

src.com.ibm.as400.access.AS400Time Maven / Gradle / Ivy

There is a newer version: 20.0.8
Show newest version
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename:  AS400Time.java
//
// The source code contained herein is licensed under the IBM Public License
// Version 1.0, which has been approved by the Open Source Initiative.
// Copyright (C) 2010-2010 International Business Machines Corporation and
// others.  All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////

package com.ibm.as400.access;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Hashtable;
import java.util.TimeZone;

/**
 Provides a converter between a {@link java.sql.Time java.sql.Time} object and an IBM i time 
 value such as "23:59:59" or "11:59 PM".
 In the IBM i programming reference, this type is referred to as the "Time Data Type", 
 or DDS data type T.
 

An IBM i time value simply indicates an hour/minute/second within some (unspecified) 24-hour period, and does not indicate a contextual day, month, year, or time zone. Internally, this class interprets all date- and time-related strings as relative to the server's time zone.

Suggestion: To avoid confusion and unexpected results when crossing time zones:
Whenever creating or interpreting instances of {@link java.sql.Date java.sql.Date}, {@link java.sql.Time java.sql.Time}, or {@link java.sql.Timestamp java.sql.Timestamp}, always assume that the reference time zone for the object is the server's time zone, and avoid using any deprecated methods. If it is necessary to convert date/time values between the server's time zone and other time zones, use methods of Calendar. Rather than using toString() to display the value of a date/time object, use DateFormat.format() after specifying the server timezone. For example: import java.text.SimpleDateFormat; java.sql.Time time1; // value to be generated by AS400Time SimpleDateFormat formatter = new SimpleDateFormat("HH:mm:ss"); // Set the formatter's time zone to the server timezone. formatter.setTimeZone(as400.getTimeZone()); ... System.out.println("Time value: " + formatter.format(time1)); @see AS400Timestamp @see AS400Date **/ public class AS400Time extends AS400AbstractTime { static final long serialVersionUID = 4L; private java.sql.Time defaultValue_; private static Hashtable formatsMap_; /** The minimum value representable by this date type. This value represents the time 00:00:00. **/ public static final java.sql.Time MIN_VALUE = new java.sql.Time(0L); private static final long MILLISECONDS_IN_A_DAY = 24*60*60*1000; // milliseconds in 24 hours /** The maximum value representable by this date type. This value represents the time 23:59:59.999. **/ public static final java.sql.Time MAX_VALUE = new java.sql.Time(MILLISECONDS_IN_A_DAY - 1); /** Format HMS: hh:mm:ss
Default separator: ':' **/ public static final int FORMAT_HMS = 100; // valid separators: { : . , & } /** Format ISO: hh.mm.ss
Default separator: '.' **/ public static final int FORMAT_ISO = 101; // valid separators: { . } /** Format USA: hh:mm AM or hh:mm PM
Default separator: ':'
Note: Unlike the other formats, this format has a granularity of minutes rather than seconds. **/ public static final int FORMAT_USA = 102; // valid separators: { : } /** Format EUR: hh.mm.ss
Default separator: '.' **/ public static final int FORMAT_EUR = 103; // valid separators: { . } /** Format JIS: hh:mm:ss
Default separator: ':' **/ public static final int FORMAT_JIS = 104; // valid separators: { : } // Note to maintenance programmer: Update these values when adding new formats. private static final int FORMAT_RANGE_MINIMUM = FORMAT_HMS; private static final int FORMAT_RANGE_MAXIMUM = FORMAT_JIS; /** Constructs an AS400Time object. This uses the GMT time zone. The timezone is used when converting from a string to a "Time" object. The time object will use the timezone of the current JVM. Format {@link #FORMAT_ISO FORMAT_ISO} and separator '.' are used. **/ public AS400Time() { // Note: The default internal format for IBM i time variables is *ISO. this(FORMAT_ISO); } /** Constructs an AS400Time object. Format {@link #FORMAT_ISO FORMAT_ISO} and separator '.' are used. * @param timeZone **/ public AS400Time(TimeZone timeZone) { // Note: The default internal format for IBM i time variables is *ISO. this(timeZone, FORMAT_ISO); } /** Constructs an AS400Time object. The specified format's default separator is used. The default GMT timezone is used @param format The format for this object. For a list of valid values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. **/ public AS400Time(int format) { super(); setFormat(format, defaultSeparatorFor(format)); } /** Constructs an AS400Time object. The specified format's default separator is used. * @param timeZone @param format The format for this object. For a list of valid values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. **/ public AS400Time(TimeZone timeZone, int format) { super(timeZone); setFormat(format, defaultSeparatorFor(format)); } /** Constructs an AS400Time object. The specified format's default separator is used. The default GMT time zone will be used. @param format The format for this object. Valid values are:

  • {@link #FORMAT_HMS FORMAT_HMS}
  • {@link #FORMAT_ISO FORMAT_ISO}
  • {@link #FORMAT_USA FORMAT_USA}
  • {@link #FORMAT_EUR FORMAT_EUR}
  • {@link #FORMAT_JIS FORMAT_JIS}
@param separator The separator character. Valid values are:
  • ' ' (blank)
  • ':' (colon)
  • '.' (period)
  • ',' (comma)
  • '&' (ampersand)
  • (null)
A null value indicates "no separator". Refer to the IBM i programming reference to determine which separator characters are valid with each format. **/ public AS400Time(int format, Character separator) { super(); setFormat(format, separator); } /** Constructs an AS400Time object. The specified format's default separator is used. * @param timeZone @param format The format for this object. Valid values are:
  • {@link #FORMAT_HMS FORMAT_HMS}
  • {@link #FORMAT_ISO FORMAT_ISO}
  • {@link #FORMAT_USA FORMAT_USA}
  • {@link #FORMAT_EUR FORMAT_EUR}
  • {@link #FORMAT_JIS FORMAT_JIS}
@param separator The separator character. Valid values are:
  • ' ' (blank)
  • ':' (colon)
  • '.' (period)
  • ',' (comma)
  • '&' (ampersand)
  • (null)
A null value indicates "no separator". Refer to the IBM i programming reference to determine which separator characters are valid with each format. **/ public AS400Time(TimeZone timeZone, int format, Character separator) { super(timeZone); setFormat(format, separator); } // Overrides non-public method of superclass, making it public. /** Gets the format of this AS400Time object. @return format The format for this object. For a list of possible values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. **/ public int getFormat() { return super.getFormat(); } /** Gets the separator character of this AS400Time object. @return separator The separator character. For a list of possible values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. If the format contains no separators, null is returned. @see #setFormat(int,Character) **/ public Character getSeparator() { return super.getSeparator(); } /** Sets the format of this AS400Time object. The specified format's default separator is used. @param format The format for this object. For a list of valid values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. **/ public void setFormat(int format) { super.setFormat(format, defaultSeparatorFor(format)); } // Method used by TimeFieldDescription. /** Sets the format of this AS400Time object. The specified format's default separator character is used. @param format The format for this object, expressed as a string. For a list of valid values, refer to {@link #toFormat(String) toFormat(String)}. **/ void setFormat(String format) { super.setFormat(toFormat(format)); } // Method used by TimeFieldDescription. /** Sets the separator character of this AS400Time object. @param separator The separator character. **/ void setSeparator(Character separator) { super.setSeparator(separator); } // Overrides non-public method of superclass, making it public. /** Sets the format of this AS400Time object. @param format The format for this object. For a list of valid values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. @param separator The separator character. For a list of valid values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. A null value indicates "no separator". Refer to the IBM i programming reference to determine which separator characters are valid with each format. **/ public void setFormat(int format, Character separator) { super.setFormat(format, separator); } /** Sets the format of this AS400Time object. @param format The format for this object. For a list of valid values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. @param separator The separator character. @deprecated Use {@link #setFormat(int,Character) setFormat(int,Character)} instead. **/ public void setFormat(int format, char separator) { super.setFormat(format, new Character(separator)); } private static Hashtable getFormatsMap() { if (formatsMap_ == null) { synchronized (AS400Time.class) { if (formatsMap_ == null) { formatsMap_ = new Hashtable(12); formatsMap_.put("HMS", new Integer(FORMAT_HMS)); formatsMap_.put("ISO", new Integer(FORMAT_ISO)); formatsMap_.put("USA", new Integer(FORMAT_USA)); formatsMap_.put("EUR", new Integer(FORMAT_EUR)); formatsMap_.put("JIS", new Integer(FORMAT_JIS)); } } } return formatsMap_; } /** Returns the integer format value that corresponds to specified format name. If null is specified, the default format (FORMAT_ISO) is returned. This method is provided for use by the PCML infrastructure. @param formatName The format name. Valid values are:
  • HMS
  • ISO
  • USA
  • EUR
  • JIS
@return the format value. For example, if formatName is "ISO", then {@link #FORMAT_ISO FORMAT_ISO} is returned. **/ public static int toFormat(String formatName) { if (formatName == null || formatName.length() == 0) { if (Trace.traceOn_) { Trace.log(Trace.DIAGNOSTIC, "AS400Time.toFormat("+formatName+"): Returning default time format."); } return FORMAT_ISO; } Integer formatInt = (Integer)getFormatsMap().get(formatName.trim().toUpperCase()); if (formatInt == null) { throw new ExtendedIllegalArgumentException("format ("+formatName+")", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } return formatInt.intValue(); } // Overrides method of superclass. /** Returns a Java object representing the default value of the data type. @return A {@link java.sql.Time java.sql.Time} object representing time 00:00:00 GMT (on January 1, 1970). **/ public Object getDefaultValue() { if (defaultValue_ == null) { defaultValue_ = new java.sql.Time(0L); // 00:00:00 GMT } return defaultValue_; } // Implements abstract method of superclass. /** Returns {@link AS400DataType#TYPE_TIME TYPE_TIME}. @return AS400DataType.TYPE_TIME. **/ public int getInstanceType() { return AS400DataType.TYPE_TIME; } // Implements abstract method of superclass. /** Returns the Java class that corresponds with this data type. @return java.sql.Time.class. **/ public Class getJavaType() { return java.sql.Time.class; } // Overrides method of superclass. This allows us to be more specific in the javadoc. /** Converts the specified Java object into IBM i format in the specified byte array. @param javaValue The object corresponding to the data type. It must be an instance of {@link java.sql.Time java.sql.Time}. The range of valid values is {@link #MIN_VALUE MIN_VALUE} through {@link #MAX_VALUE MAX_VALUE}. Year, month, day-of-month, and fractional seconds are disregarded. @param as400Value The array to receive the data type in IBM i format. There must be enough space to hold the IBM i value. @param offset The offset into the byte array for the start of the IBM i value. It must be greater than or equal to zero. @return Eight (8), the number of bytes in the IBM i representation of the data type. **/ public int toBytes(Object javaValue, byte[] as400Value, int offset) { return super.toBytes(javaValue, as400Value, offset); } // Implements abstract method of superclass. /** Converts the specified IBM i data type to a Java object. @param as400Value The array containing the data type in IBM i format. The entire data type must be represented. @param offset The offset into the byte array for the start of the IBM i value. It must be greater than or equal to zero. @return a {@link java.sql.Time java.sql.Time} object, representing the number of milliseconds into the day. The reference time zone for the object is the timezone specified on the constructor. **/ public Object toObject(byte[] as400Value, int offset) { if (as400Value == null) throw new NullPointerException("as400Value"); String timeString = getCharConverter().byteArrayToString(as400Value, offset, getLength()); // Parse the string, and create a java.sql.Time object. return parse(timeString); } // Implements abstract method of superclass. /** Converts the specified Java object into a String representation that is consistent with the format of this data type. @param javaValue The object corresponding to the data type. This must be an instance of {@link java.sql.Time java.sql.Time}, and must be within the range representable by this data type. Any timezone context is disregarded. @return A String representation of the specified value, formatted appropriately for this data type. **/ public String toString(Object javaValue) { if (javaValue == null) throw new NullPointerException("javaValue"); java.sql.Time timeObj; try { timeObj = (java.sql.Time)javaValue; } catch (ClassCastException e) { Trace.log(Trace.ERROR, "javaValue is of type " + javaValue.getClass().getName()); throw e; } return getDateFormatter().format(timeObj); } /** Converts a string representation of a time, to a Java object. @param source A time value expressed as a string in the format specified for this AS400Time object. @return A {@link java.sql.Time java.sql.Time} object representing the specified time. The reference timezone is the timezone specified for the object. **/ public java.sql.Time parse(String source) { if (source == null) throw new NullPointerException("source"); try { SimpleDateFormat formatter = getDateFormatter(); java.util.Date dateObj = formatter.parse(source); long milliseconds = dateObj.getTime(); java.sql.Time time = new java.sql.Time(milliseconds); // argument is "milliseconds into day" // Convert to the base time type. return time; } catch (Exception e) { // Assume that the exception is because we got bad input. Trace.log(Trace.ERROR, e.getMessage(), source); Trace.log(Trace.ERROR, "Time string is expected to be in format: " + patternFor(getFormat(), getSeparator())); throw new ExtendedIllegalArgumentException("source ("+source+")", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } } /** Converts the specified HMS representation of a time, to a Java object. This method is provided for use by the PCML infrastructure; in particular, when parsing 'init=' values for 'time' data elements. This method assumes that the reference timezone is GMT @param source A time value expressed as a string in format HH:mm:ss. @return A {@link java.sql.Time java.sql.Time} object representing the specified time. **/ public static java.sql.Time parseXsdString(String source) { return parseXsdString(source, AS400AbstractTime.TIMEZONE_GMT); } /** Converts the specified HMS representation of a time, to a Java object. This method is provided for use by the PCML infrastructure; in particular, when parsing 'init=' values for 'time' data elements. @param source A time value expressed as a string in format HH:mm:ss. @param timeZone The time zone used to interpret the value. @return A {@link java.sql.Time java.sql.Time} object representing the specified time. **/ public static java.sql.Time parseXsdString(String source, TimeZone timeZone) { if (source == null) throw new NullPointerException("source"); try { java.util.Date simpleDateObj = getTimeFormatterXSD(timeZone).parse(source); long milliseconds = simpleDateObj.getTime(); java.sql.Time returnTime = new java.sql.Time(milliseconds); return returnTime; } catch (ParseException e) { // Assume that the exception is because we got bad input. Trace.log(Trace.ERROR, e.getMessage(), source); Trace.log(Trace.ERROR, "Value is expected to be in standard XML Schema 'time' format: " + TIME_PATTERN_XSD); throw new ExtendedIllegalArgumentException("source ("+source+")", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } } /** Converts the specified Java object into a String representation that is consistent with the format of this data type. This method is provided for use by the PCML infrastructure. This version uses the GMT timezone. @param javaValue The object corresponding to the data type. This must be an instance of {@link java.sql.Time java.sql.Time}, and must be within the range specifiable by this data type. @return The time expressed as a string in format HH:mm:ss. **/ public static String toXsdString(Object javaValue) { return toXsdString(javaValue, TIMEZONE_GMT ); } /** Converts the specified Java object into a String representation that is consistent with the format of this data type. This method is provided for use by the PCML infrastructure. @param javaValue The object corresponding to the data type. This must be an instance of {@link java.sql.Time java.sql.Time}, and must be within the range specifiable by this data type. @param timeZone The time zone used to interpret the value. @return The time expressed as a string in format HH:mm:ss. **/ public static String toXsdString(Object javaValue, TimeZone timeZone) { if (javaValue == null) throw new NullPointerException("javaValue"); java.sql.Time timeObj; try { timeObj = (java.sql.Time)javaValue; } catch (ClassCastException e) { Trace.log(Trace.ERROR, "javaValue is of type " + javaValue.getClass().getName()); throw e; } return getTimeFormatterXSD(timeZone).format(timeObj); } // Implements abstract method of superclass. String patternFor(int format, Character separator) { String sep = ( separator == null ? "" : separator.toString()); switch (format) { case FORMAT_USA: String sep2 = ( sep.equals("") ? "" : " " ); return "hh" + sep + "mm" + sep2 + "a"; // hh:mm AM hh:mm PM default: return "HH" + sep + "mm" + sep + "ss"; } } // Implements abstract method of superclass. Character defaultSeparatorFor(int format) { if (!isValidFormat(format)) { throw new ExtendedIllegalArgumentException("format ("+format+")", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } switch (format) { case FORMAT_HMS: case FORMAT_USA: case FORMAT_JIS: return COLON; // ':' case FORMAT_ISO: case FORMAT_EUR: return PERIOD; // '.' default: // none of the above formats // Should never happen. throw new InternalErrorException(InternalErrorException.UNKNOWN, "Unrecognized format: " + format, null); } } // Implements abstract method of superclass. boolean isValidFormat(int format) { return validateFormat(format); } /** Validates the specified format value. This method is provided for use by the PCML infrastructure. @param format The format. For a list of valid values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. @return true if the format is valid; false otherwise. **/ public static boolean validateFormat(int format) { if (format < FORMAT_RANGE_MINIMUM || format > FORMAT_RANGE_MAXIMUM) return false; else return true; } /** Returns the number of bytes occupied on the IBM i system by a field of this type. This method is provided for use by the PCML infrastructure. @param format The format. This argument is ignored. For a list of valid values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. @param separator The separator character. This argument is ignored. For a list of valid values, refer to {@link #AS400Time(int,Character) AS400Time(int,Character)}. @return the number of bytes occupied. **/ public static int getByteLength(int format, Character separator) { if (separator == null) return 6; // field size (without separators) is 6 bytes else return 8; // field size (with separators) is 8 bytes } // Implements abstract method of superclass. int lengthFor(int format) { return getByteLength(format, getSeparator()); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy