298 lines
10 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2008, 2018, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package vm.share.options;
import java.lang.reflect.Array;
import java.util.HashMap;
import java.util.Map;
import nsk.share.TestBug;
/**
* A utility class used to parse arguments of various primitive types.
*/
public class PrimitiveParser
{
/**
* Checks if the parser can handle passed type ("primitive/wrapper" or one-dim array of those).
* Note that one-dim arrays of primitives/strings/wrappers are also supported.
* @param type the type to parse against
* @return true, if can parse.
*/
public static boolean canHandle(Class<?> type)
{
if(type.isArray())
{
Class<?> compType = type.getComponentType();
if(compType.isArray()) return false; // Cannot handle multidimensional arrays
return canHandle(compType);
}
type = convertPrimitiveTypeToWrapper(type);
return parsers.containsKey(type);
}
/**
* A simple helper method.
* @param type the type to check for
* @return the parser to use, null if there is none.
*/
private static PParser getParser(Class<?> type)
{
return parsers.get(convertPrimitiveTypeToWrapper(type));
}
/**
* The main API method of this class.
* @param param the parameter to parse
* @param type parameter type to parse against
* @return returns the object of a given type.
* @throws vm.share.options.PrimitiveParser.ParserException
*/
public static Object parse(String param, Class<?> type) throws ParserException
{
if(type.isArray())
{
Class<?> compType = type.getComponentType();
if(compType.isArray())
throw new ParserException("Cannot handle multidimensional arrays");
if(!canHandle(compType))
throw new ParserException("Unable to parse unknown array component type " + compType);
String[] params = param.split(",");
Object arr = Array.newInstance(compType, params.length);
for (int i = 0; i < params.length; i++)
{
String par = params[i].trim();
Array.set(arr, i, parse(par, compType));
}
return arr;
}
else
{
if(!canHandle(type))
throw new ParserException("Unable to parse unknown type " + type);
return getParser(type).parse(param);
}
}
// I'm not sure, if generics are of any use here...
static private abstract class PParser<T>
{
abstract T parse(String param) throws ParserException;
// Class<T> getClassKey()
// {
// return PrimitiveParser.(Class<T>) T.getClass();
// }
}
/**
* Converts primitive types to corresponding wrapper classes.
* We could register int.class, boolean.class etc in the hashtable instead.
* (Or Integer.TYPE, etc.)
* @param type to convert to wrapper
* @return wrapper class or type if it is not primitive
*/
public static Class<?> convertPrimitiveTypeToWrapper(Class<?> type)
{
if(!type.isPrimitive()) return type;
Object arr = Array.newInstance(type, 1);
Object v = Array.get(arr, 0);
return v.getClass();
}
//"kind of state" machine stuff
private static Map<Class<?>, PParser<?>> parsers;
static
{
parsers = new HashMap<Class<?>, PrimitiveParser.PParser<?>>(16);
parsers.put(Integer.class, new PParser<Integer>()
{
@Override Integer parse(String param) throws ParserException
{
if ( param.startsWith("0x") )
return Integer.parseInt(param.substring(2));
else
return Integer.valueOf(param);
}
});
parsers.put(Boolean.class, new PParser<Boolean>()
{
@Override Boolean parse(String param) throws ParserException
{
//special behavior for options
if(param == null) return true;
if(param.trim().length()==0) return true;
return Boolean.valueOf(param);
}
});
parsers.put(String.class, new PParser<String>()
{
@Override String parse(String param) throws ParserException
{
if(param == null) throw new ParserException(" Got null value string.");
return param;
}
});
parsers.put(Character.class, new PParser<Character>()
{
@Override Character parse(String param) throws ParserException
{
if(param.length()!=1)
throw new TestBug("Found Character type option of length != 1");
return Character.valueOf(param.charAt(0));
}
});
parsers.put(Byte.class, new PParser<Byte>()
{
@Override Byte parse(String param) throws ParserException
{
if ( param.startsWith("0x") )
return Byte.parseByte(param.substring(2));
else
return Byte.valueOf(param);
}
});
parsers.put(Short.class, new PParser<Short>()
{
@Override Short parse(String param) throws ParserException
{
if ( param.startsWith("0x") )
return Short.parseShort(param.substring(2));
else
return Short.valueOf(param);
}
});
parsers.put(Long.class, new PParser<Long>()
{
@Override Long parse(String param) throws ParserException
{
if ( param.startsWith("0x") )
return Long.parseLong(param.substring(2));
else
return Long.valueOf(param);
}
});
parsers.put(Float.class, new PParser<Float>()
{
@Override Float parse(String param) throws ParserException
{
return Float.valueOf(param);
}
});
parsers.put(Double.class, new PParser<Double>()
{
@Override Double parse(String param) throws ParserException
{
return Double.valueOf(param);
}
});
}
/* Discussion
* 1. It was proposed to use instead of the convertPrimitive the following
*
* private static Map<Class<?>, Class<?>> wrapperClasses = new HashMap<Class<?>, Class<?>>();
*
* so we could do if(type.isPrimitive())
type = wrapperClasses.get(type);
static {
wrapperClasses.put(boolean.class, Boolean.class);
wrapperClasses.put(short.class, Short.class);
wrapperClasses.put(int.class, Integer.class);
wrapperClasses.put(Long.Type, Long.class); // we can do it this way!
wrapperClasses.put(float.class, Float.class);
wrapperClasses.put(double.class, Double.class);
}
* The alternative is to register PParsers with corresponding Primitive type too.
*
* Also canHandle() could use
return wrapperClasses.keySet().contains(type) || wrapperClasses.entrySet().contains(type);
* 2. Parsing can be implemented via reflection
return type.getMethod("valueOf", new Class[]{String.class}).invoke(null, string);
* I don't like using reflection as it prevents optimisation,
* also now Strings and Characters are handled in a nice fashion.
*
* As for convertToPrimitive trick both ways are good,
* but current looks more generic though tricky
*/
//// some test, should it be commented out?
// public static void main(String[] args)
// {
// try
// {
// String str = "0";
// Object o = null;
// str = "0";
// o = parse(str, String.class);
// System.out.println("value:" + str + " type: " + o.getClass() + " value:#" + o + "#");
//
// str = "0";
// o = parse(str, int.class);
// System.out.println("value:" + str + " type: " + o.getClass() + " value:#" + o + "#");
// str = "0";
// o = parse(str, Integer.class);
// System.out.println("value:" + str + " type: " + o.getClass() + " value:#" + o + "#");
//
// str = "0,1,2";
// o = parse(str, int[].class);
// System.out.println("value:" + str + " type: " + o.getClass() + " value:#" + (int[]) o + "#");
// System.out.println("DATA:" + java.util.Arrays.toString((int[])o));
// // System.out.println("DATA:" + java.util.Arrays.deepToString( (int[]) o));
//
// str = "0";
// o = parse(str, byte.class);
// System.out.println("value:" + str + " type: " + o.getClass() + " value:#" + o + "#");
//
// str = "0";
// o = parse(str, HashMap.class);
// System.out.println("value:" + str + " type: " + o.getClass() + " value:#" + o + "#");
//// String str = "0"; Object o = parsePrimitiveString(help_option, type); System.out.println("value:" + str + " type: " + o.getClass() + " value:#" + o + "#");
// } catch (ParserException ex)
// {
// System.out.println("" +ex);
// }
//// String str = "0"; Object o = parsePrimitiveString(help_option, type); System.out.println("value:" + str + " type: " + o.getClass() + " value:#" + o + "#");
//
// }
}