从字符串转换为泛型类型

时间:2014-07-05 18:43:39

标签: java generics casting

我需要将字符串转换为变量值。我找到了仅针对C#的解决方案。我需要用Java。

public class Property<T> {

    T value;

    public void setValue(String input){
        if(value instanceof String){
           value= input; // value is type of T not type of string (compilation error)
                         // incompatible types: String cannot be converted to T
        }
        if(value instanceof int){
           //parse string
        }
        if(value instanceof boolean){
           //parse string
        }
        ...
    }
}

2 个答案:

答案 0 :(得分:10)

这不是它的工作原理。但是,您可以使用多态,以获得有用的结果。

具有多态性的解决方案

基础通用(和抽象)属性

public abstract class Property<T> {
    T value;
    public abstract void setValue(String input);
}

字符串的属性

public class StringProperty extends Property<String> {
    @Override
    public void setValue(String input) {
        this.value = input;
    }
}

整数属性

public class IntegerProperty extends Property<Integer> {
    @Override
    public void setValue(String input) {
        this.value = Integer.valueOf(input);
    }
}

不确定您的实际目标是什么,但这种方法可能有效。

请注意,由于类型擦除,input instanceof T将失败。它不会起作用。


Class<T>为参数的解决方案

要详细说明您的方法,这样可行 - 但它很有用。

丑陋且不太方便。不知道你为什么要这样,tbh。

class Property<T> {

    public T value;
    private final Class<T> clazz;

    public Property(Class<T> clazz) {
        super();
        this.clazz = clazz;
    }       

    @SuppressWarnings("unchecked")
    public void setValue(String input) {
        if (clazz.isAssignableFrom(String.class)) {
            value = (T) input;
        } else if (clazz.isAssignableFrom(Integer.class)) {
            value = (T) Integer.valueOf(input);
        } else if (clazz.isAssignableFrom(Boolean.class)) {
            value = (T) Boolean.valueOf(input);
        } else if (clazz.isAssignableFrom(Double.class)) {
            value = (T) Double.valueOf(input);
        } else {
            throw new IllegalArgumentException("Bad type.");
        }
    }
}

像这样使用:

Property<String> ff = new Property<>(String.class);
ff.setValue("sdgf");

Property<Integer> sdg = new Property<>(Integer.class);
sdg.setValue("123");

System.out.println(ff.value);
System.out.println(sdg.value);

Reflection

的解决方案

显然,可以找出用于实例化属性的参数。

这个小小的神奇公式可以为您提供:

(Class<?>) getClass().getTypeParameters()[0].getBounds()[0]

我甚至不知道我是如何找到它的。好吧,我们走了:

class Property<T> {

    T value;    

    @SuppressWarnings("unchecked")
    public void setValue(String input)
    {
        // BEHOLD, MAGIC!
        Class<?> clazz = (Class<?>) getClass().getTypeParameters()[0].getBounds()[0];

        if (clazz.isAssignableFrom(String.class)) {
            value = (T) input;
        } else if (clazz.isAssignableFrom(Integer.class)) {
            value = (T) Integer.valueOf(input);
        } else if (clazz.isAssignableFrom(Boolean.class)) {
            value = (T) Boolean.valueOf(input);
        } else if (clazz.isAssignableFrom(Double.class)) {
            value = (T) Double.valueOf(input); 
        } else {
            throw new IllegalArgumentException("Bad type.");
        }
    }
}

不要看着我,我不会用它。我有一些常识。

答案 1 :(得分:0)

我通过使用 Spring 找到了一种新方法。

public static <T> T convertJsonToObject(String json, Class<T> valueType) {
    try {
        return json != null ? new ObjectMapper().readValue(json, valueType) : null;
    } catch (Exception e) {
        return null;
    }
}
private static void processValidate(Class<?> valueClass, String value) {
     Object parsedValue = Utils.convertJsonToObject(value, valueClass);
     // logic
}