【问题标题】:Conversion from string to generic type从字符串到泛型类型的转换
【发布时间】:2014-07-05 18:43:39
【问题描述】:

我需要将字符串转换为变量值。我找到了仅适用于 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
        }
        ...
    }
}

【问题讨论】:

  • 如果您不知道 T 是什么,则无法转换为它。你可以施放,但很容易失败。糟糕的设计,我说。
  • 什么是“特定类型的泛型变量”?
  • 这没有意义,如果参数是字符串类型,怎么可能不是String
  • 我的意思是我用 instanceof 识别的 T 类型
  • @SpeedEX505 我不会撒谎,我是通过反复试验找到的。这对我来说也很模糊。但我不介意肮脏的把戏,只要他们听我的吩咐xD

标签: java generics casting


【解决方案1】:

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

多态性解决方案

基本通用(和抽象)属性

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&lt;T&gt; 作为参数的解决方案

要详细说明您的方法,这会起作用 - 但它很丑。

丑陋,不太方便。不知道你为什么想要它,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.");
        }
    }
}

别看我,我不会用那个。我有一些常识。

【讨论】:

  • 我以前有这个代码,但我试图避免它。我想我会在更高级别解析字符串,然后调用类型为 T 的方法 setValue(T input)。但是谢谢。
  • 您必须将 T 的类作为参数传递给 Property 的构造函数。然后,您可以将此类与一些已知类型进行比较并在其上进行分支 - 但它很丑陋。多态是可以使用的,而不是避免的。
  • 嗯,确实有一个用例。想象一个简单的文本配置文件、XML、webservice 结果(随便)给你一个 映射,你实际上需要一个通用方法,比如说&lt;V&gt; V GetValueByKey(string key),它为你提供适当的键值,但具有所需的类型- 而不是字符串。
【解决方案2】:

我发现了一种使用 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
}

【讨论】:

    猜你喜欢
    • 2011-06-29
    • 2010-09-05
    • 2013-05-15
    • 2014-06-30
    • 2013-09-30
    • 2021-10-25
    • 2019-09-10
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多