decimaldv.java

来自「JAVA的一些源码 JAVA2 STANDARD EDITION DEVELO」· Java 代码 · 共 325 行

JAVA
325
字号
/* * The Apache Software License, Version 1.1 * * * Copyright (c) 2001, 2002 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 "Xerces" and "Apache Software Foundation" 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", *    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 and was * originally based on software copyright (c) 2001, International * Business Machines, Inc., http://www.apache.org.  For more * information on the Apache Software Foundation, please see * <http://www.apache.org/>. */package com.sun.org.apache.xerces.internal.impl.dv.xs;import com.sun.org.apache.xerces.internal.impl.dv.InvalidDatatypeValueException;import com.sun.org.apache.xerces.internal.impl.dv.ValidationContext;/** * Represent the schema type "decimal" * * @author Neeraj Bajaj, Sun Microsystems, inc. * @author Sandy Gao, IBM * * @version $Id: DecimalDV.java,v 1.9 2003/05/08 20:11:55 elena Exp $ */public class DecimalDV extends TypeValidator {    public final short getAllowedFacets(){        return ( XSSimpleTypeDecl.FACET_PATTERN | XSSimpleTypeDecl.FACET_WHITESPACE | XSSimpleTypeDecl.FACET_ENUMERATION |XSSimpleTypeDecl.FACET_MAXINCLUSIVE |XSSimpleTypeDecl.FACET_MININCLUSIVE | XSSimpleTypeDecl.FACET_MAXEXCLUSIVE  | XSSimpleTypeDecl.FACET_MINEXCLUSIVE | XSSimpleTypeDecl.FACET_TOTALDIGITS | XSSimpleTypeDecl.FACET_FRACTIONDIGITS);    }    public Object getActualValue(String content, ValidationContext context) throws InvalidDatatypeValueException {        try {            return new XDecimal(content);        } catch (NumberFormatException nfe) {            throw new InvalidDatatypeValueException("cvc-datatype-valid.1.2.1", new Object[]{content, "decimal"});        }    }    public final int compare(Object value1, Object value2){        return ((XDecimal)value1).compareTo((XDecimal)value2);    }    public final int getTotalDigits(Object value){        return ((XDecimal)value).totalDigits;    }    public final int getFractionDigits(Object value){        return ((XDecimal)value).fracDigits;    }        // Avoid using the heavy-weight java.math.BigDecimal    static class XDecimal {        // sign: 0 for vlaue 0; 1 for positive values; -1 for negative values        int sign = 1;        // total digits. >= 1        int totalDigits = 0;        // integer digits when sign != 0        int intDigits = 0;        // fraction digits when sign != 0        int fracDigits = 0;        // the string representing the integer part        String ivalue = "";        // the string representing the fraction part        String fvalue = "";        // whether the canonical form contains decimal point        boolean integer = false;                XDecimal(String content) throws NumberFormatException {            initD(content);        }        XDecimal(String content, boolean integer) throws NumberFormatException {            if (integer)                initI(content);            else                initD(content);        }        void initD(String content) throws NumberFormatException {            int len = content.length();            if (len == 0)                throw new NumberFormatException();                // these 4 variables are used to indicate where the integre/fraction            // parts start/end.            int intStart = 0, intEnd = 0, fracStart = 0, fracEnd = 0;                        // Deal with leading sign symbol if present            if (content.charAt(0) == '+') {                // skip '+', so intStart should be 1                intStart = 1;            }            else if (content.charAt(0) == '-') {                // keep '-', so intStart is stil 0                intStart = 1;                sign = -1;            }                // skip leading zeroes in integer part            int actualIntStart = intStart;            while (actualIntStart < len && content.charAt(actualIntStart) == '0') {                actualIntStart++;            }                // Find the ending position of the integer part            for (intEnd = actualIntStart;                 intEnd < len && TypeValidator.isDigit(content.charAt(intEnd));                 intEnd++);                // Not reached the end yet            if (intEnd < len) {                // the remaining part is not ".DDD", error                if (content.charAt(intEnd) != '.')                    throw new NumberFormatException();                    // fraction part starts after '.', and ends at the end of the input                fracStart = intEnd + 1;                fracEnd = len;            }                // no integer part, no fraction part, error.            if (intStart == intEnd && fracStart == fracEnd)                throw new NumberFormatException();                // ignore trailing zeroes in fraction part            while (fracEnd > fracStart && content.charAt(fracEnd-1) == '0') {                fracEnd--;            }                // check whether there is non-digit characters in the fraction part            for (int fracPos = fracStart; fracPos < fracEnd; fracPos++) {                if (!TypeValidator.isDigit(content.charAt(fracPos)))                    throw new NumberFormatException();            }                intDigits = intEnd - actualIntStart;            fracDigits = fracEnd - fracStart;            totalDigits = intDigits + fracDigits;                if (intDigits > 0) {                ivalue = content.substring(actualIntStart, intEnd);                if (fracDigits > 0)                    fvalue = content.substring(fracStart, fracEnd);            }            else {                if (fracDigits > 0) {                    fvalue = content.substring(fracStart, fracEnd);                }                else {                    // ".00", treat it as "0"                    sign = 0;                }            }        }        void initI(String content) throws NumberFormatException {            int len = content.length();            if (len == 0)                throw new NumberFormatException();                // these 2 variables are used to indicate where the integre start/end.            int intStart = 0, intEnd = 0;                // Deal with leading sign symbol if present            if (content.charAt(0) == '+') {                // skip '+', so intStart should be 1                intStart = 1;            }            else if (content.charAt(0) == '-') {                // keep '-', so intStart is stil 0                intStart = 1;                sign = -1;            }                // skip leading zeroes in integer part            int actualIntStart = intStart;            while (actualIntStart < len && content.charAt(actualIntStart) == '0') {                actualIntStart++;            }                // Find the ending position of the integer part            for (intEnd = actualIntStart;                 intEnd < len && TypeValidator.isDigit(content.charAt(intEnd));                 intEnd++);                // Not reached the end yet, error            if (intEnd < len)                throw new NumberFormatException();                // no integer part, error.            if (intStart == intEnd)                throw new NumberFormatException();                intDigits = intEnd - actualIntStart;            fracDigits = 0;            totalDigits = intDigits;                if (intDigits > 0) {                ivalue = content.substring(actualIntStart, intEnd);            }            else {                // "00", treat it as "0"                sign = 0;            }                        integer = true;        }        public boolean equals(Object val) {            if (val == this)                return true;                if (!(val instanceof XDecimal))                return false;            XDecimal oval = (XDecimal)val;                        if (sign != oval.sign)               return false;            if (sign == 0)                return true;                        return intDigits == oval.intDigits && fracDigits == oval.fracDigits &&                   ivalue.equals(oval.ivalue) && fvalue.equals(oval.fvalue);        }        public int compareTo(XDecimal val) {            if (sign != val.sign)                return sign > val.sign ? 1 : -1;            if (sign == 0)                return 0;            return sign * intComp(val);        }        private int intComp(XDecimal val) {            if (intDigits != val.intDigits)                return intDigits > val.intDigits ? 1 : -1;            int ret = ivalue.compareTo(val.ivalue);            if (ret != 0)                return ret > 0 ? 1 : -1;;            ret = fvalue.compareTo(val.fvalue);            return ret == 0 ? 0 : (ret > 0 ? 1 : -1);        }        private String canonical;        public synchronized String toString() {            if (canonical == null) {                makeCanonical();            }            return canonical;        }                private void makeCanonical() {            if (sign == 0) {                if (integer)                    canonical = "0";                else                    canonical = "0.0";                return;            }            if (integer && sign > 0) {                canonical = ivalue;                return;            }            // for -0.1, total digits is 1, so we need 3 extra spots            StringBuffer buffer = new StringBuffer(totalDigits+3);            if (sign == -1)                buffer.append('-');            if (intDigits != 0)                buffer.append(ivalue);            else                buffer.append('0');            if (!integer) {                buffer.append('.');                if (fracDigits != 0) {                    buffer.append(fvalue);                }                else {                    buffer.append('0');                }            }            canonical = buffer.toString();        }    }} // class DecimalDV

⌨️ 快捷键说明

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