hssfdateutil.java

来自「Office格式转换代码」· Java 代码 · 共 285 行

JAVA
285
字号
/* ==================================================================== * The Apache Software License, Version 1.1 * * Copyright (c) 2003 The Apache Software Foundation.  All rights * reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright *    notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright *    notice, this list of conditions and the following disclaimer in *    the documentation and/or other materials provided with the *    distribution. * * 3. The end-user documentation included with the redistribution, *    if any, must include the following acknowledgment: *       "This product includes software developed by the *        Apache Software Foundation (http://www.apache.org/)." *    Alternately, this acknowledgment may appear in the software itself, *    if and wherever such third-party acknowledgments normally appear. * * 4. The names "Apache" and "Apache Software Foundation" and *    "Apache POI" must not be used to endorse or promote products *    derived from this software without prior written permission. For *    written permission, please contact apache@apache.org. * * 5. Products derived from this software may not be called "Apache", *    "Apache POI", nor may "Apache" appear in their name, without *    prior written permission of the Apache Software Foundation. * * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * ==================================================================== * * This software consists of voluntary contributions made by many * individuals on behalf of the Apache Software Foundation.  For more * information on the Apache Software Foundation, please see * <http://www.apache.org/>. *//* * DateUtil.java * * Created on January 19, 2002, 9:30 AM */package org.apache.poi.hssf.usermodel;import java.util.Calendar;import java.util.Date;import java.util.GregorianCalendar;/** * Contains methods for dealing with Excel dates. * * @author  Michael Harhen * @author  Glen Stampoultzis (glens at apache.org) * @author  Dan Sherman (dsherman at isisph.com) */public class HSSFDateUtil{    private HSSFDateUtil()    {    }    private static final int    BAD_DATE          =        -1;   // used to specify that date is invalid    private static final long   DAY_MILLISECONDS  = 24 * 60 * 60 * 1000;    private static final double CAL_1900_ABSOLUTE =        ( double ) absoluteDay(new GregorianCalendar(1900, Calendar        .JANUARY, 1)) - 2.0;    /**     * Given a Date, converts it into a double representing its internal Excel representation,     *   which is the number of days since 1/1/1900. Fractional days represent hours, minutes, and seconds.     *     * @return Excel representation of Date (-1 if error - test for error by checking for less than 0.1)     * @param  date the Date     */    public static double getExcelDate(Date date)    {        Calendar calStart = new GregorianCalendar();        calStart.setTime(            date);   // If date includes hours, minutes, and seconds, set them to 0        if (calStart.get(Calendar.YEAR) < 1900)        {            return BAD_DATE;        }        else        {            calStart = dayStart(calStart);            double fraction = (date.getTime() - calStart.getTime().getTime())                              / ( double ) DAY_MILLISECONDS;            return fraction + ( double ) absoluteDay(calStart)                   - CAL_1900_ABSOLUTE;        }    }    /**     * Given a excel date, converts it into a Date.     * Assumes 1900 date windowing.     *     * @param  date the Excel Date     *     * @return Java representation of a date (null if error)     * @see #getJavaDate(double,boolean)     */    public static Date getJavaDate(double date)    {        return getJavaDate(date,false);    }        /**     *  Given an Excel date with either 1900 or 1904 date windowing,     *  converts it to a java.util.Date.     *     *  @param date  The Excel date.     *  @param use1904windowing  true if date uses 1904 windowing,     *   or false if using 1900 date windowing.     *  @return Java representation of the date, or null if date is not a valid Excel date     */    public static Date getJavaDate(double date, boolean use1904windowing) {        if (isValidExcelDate(date)) {            int startYear = 1900;            int dayAdjust = -1; // Excel thinks 2/29/1900 is a valid date, which it isn't            int wholeDays = (int)Math.floor(date);            if (use1904windowing) {                startYear = 1904;                dayAdjust = 1; // 1904 date windowing uses 1/2/1904 as the first day            }            else if (wholeDays < 61) {                // Date is prior to 3/1/1900, so adjust because Excel thinks 2/29/1900 exists                // If Excel date == 2/29/1900, will become 3/1/1900 in Java representation                dayAdjust = 0;            }            GregorianCalendar calendar = new GregorianCalendar(startYear,0,                                                     wholeDays + dayAdjust);            int millisecondsInDay = (int)((date - Math.floor(date)) *                                           (double) DAY_MILLISECONDS + 0.5);            calendar.set(GregorianCalendar.MILLISECOND, millisecondsInDay);            return calendar.getTime();        }        else {            return null;        }    }    /**     * given a format ID this will check whether the format represents     * an internal date format or not.      */    public static boolean isInternalDateFormat(int format) {      boolean retval =false;            switch(format) {                // Internal Date Formats as described on page 427 in                // Microsoft Excel Dev's Kit...                case 0x0e:                case 0x0f:                case 0x10:                case 0x11:                case 0x12:                case 0x13:                case 0x14:                case 0x15:                case 0x16:                case 0x2d:                case 0x2e:                case 0x2f:                    retval = true;                    break;                                    default:                    retval = false;                    break;            }       return retval;    }    /**     *  Check if a cell contains a date     *  Since dates are stored internally in Excel as double values      *  we infer it is a date if it is formatted as such.      *  @see #isInternalDateFormat(int)     */    public static boolean isCellDateFormatted(HSSFCell cell) {        if (cell == null) return false;        boolean bDate = false;                double d = cell.getNumericCellValue();        if ( HSSFDateUtil.isValidExcelDate(d) ) {            HSSFCellStyle style = cell.getCellStyle();            int i = style.getDataFormat();            bDate = isInternalDateFormat(i);        }        return bDate;    }    /**     * Given a double, checks if it is a valid Excel date.     *     * @return true if valid     * @param  value the double value     */    public static boolean isValidExcelDate(double value)    {        return (value > -Double.MIN_VALUE);    }    /**     * Given a Calendar, return the number of days since 1600/12/31.     *     * @return days number of days since 1600/12/31     * @param  cal the Calendar     * @exception IllegalArgumentException if date is invalid     */    private static int absoluteDay(Calendar cal)    {        return cal.get(Calendar.DAY_OF_YEAR)               + daysInPriorYears(cal.get(Calendar.YEAR));    }    /**     * Return the number of days in prior years since 1601     *     * @return    days  number of days in years prior to yr.     * @param     yr    a year (1600 < yr < 4000)     * @exception IllegalArgumentException if year is outside of range.     */    private static int daysInPriorYears(int yr)    {        if (yr < 1601)        {            throw new IllegalArgumentException(                "'year' must be 1601 or greater");        }        int y    = yr - 1601;        int days = 365 * y      // days in prior years                   + y / 4      // plus julian leap days in prior years                   - y / 100    // minus prior century years                   + y / 400;   // plus years divisible by 400        return days;    }    // set HH:MM:SS fields of cal to 00:00:00:000    private static Calendar dayStart(final Calendar cal)    {        cal.get(Calendar            .HOUR_OF_DAY);   // force recalculation of internal fields        cal.set(Calendar.HOUR_OF_DAY, 0);        cal.set(Calendar.MINUTE, 0);        cal.set(Calendar.SECOND, 0);        cal.set(Calendar.MILLISECOND, 0);        cal.get(Calendar            .HOUR_OF_DAY);   // force recalculation of internal fields        return cal;    }    // ---------------------------------------------------------------------------------------------------------}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?