5

我正在尝试编写一个方法,在该方法中我需要创建一个泛型类型 T 的临时变量 sum。但是,我收到错误“局部变量 sum 可能尚未初始化”。如何初始化泛型变量?我无法将其设置为 0 或 0.0,而且我无法在任何地方找到有关如何处理此问题的信息。这是我正在使用的代码部分:

public Matrix<T,A> multiply(Matrix<T,A> right) throws MatrixException
{
    Matrix<T,A> temp = new Matrix<T,A>(arithmetics, rowSize, columnSize);

    T sum, product;

    if (rowSize != right.columnSize)
        throw new MatrixException("Row size of first matrix must match column size " 
                + "of second matrix to multiply");

    setup(temp,rowSize,columnSize);

    for (int i = 0; i < rowSize; i++){
        for (int j = 0; j < right.columnSize; j++) {
            product = (arithmetics.multiply(matrix[i][j] , right.matrix[j][i]));
            sum = arithmetics.add(product, sum);
            temp.matrix[i][j] = sum;
        }
    }
    return temp;
}

我不确定这是否有助于澄清,但这是我的界面算术:

public interface Arithmetics<T> {

public T zero();
public T add( T a, T b );
public T subtract( T a, T b);
public T multiply (T a, T b);
public T parseString( String str );
public String toString( T a );

}

这是我的一个类,DoubleArithmetics,只是为了展示我是如何实现接口的:

public class DoubleArithmetics implements Arithmetics<Double> {

protected Double value;

public Double zero() 
{

    return new Double(0);
}

public Double add( Double a, Double b ) 
{

    return new Double(a.doubleValue()+b.doubleValue());
}

public Double subtract (Double a, Double b)
{
    return new Double(a.doubleValue()-b.doubleValue());
}

public Double multiply (Double a, Double b)
{
    return new Double(a.doubleValue()*b.doubleValue());
}

public Double parseString( String str )
{
    return Double.parseDouble(str);
}

public String toString( Double a )
{
    return a.toString();
}
}
4

2 回答 2

3

由于类型擦除,在 Java 中实例化泛型有点棘手。

我的方法是将两个项目传递给您的泛型类的构造函数:(1)特定于类型 T 的 java.lang.reflect.Constructor;(2) 一个 Object[] 数组,包含一个特定于类型 T 的默认值。

当您稍后要实例化和初始化类型 T 时,您需要调用 Constructor.newInstance(Object[])。在下面的代码中,MyGenericClass 类代表您的泛型类(看起来它在您的原始帖子中称为 Matrix)。

我从InstantiationException 中获得了 newInstance()的解决方案并在 Java 中创建泛型类型的实例?

public class MyGenericClass<T>
{
    Constructor _constructorForT;
    Object[] _initialValueForT;

    public MyGenericClass(Constructor constructorForT, 
                          Object[] initialValueForT)
    {
        _constructorForT = constructorForT;
        _initialValueForT = initialValueForT;
    }

    public void doSomething() 
    {
        T sum = initializeT(_constructorForT, _initialValueForT);

        System.out.printf("d = %f\n", sum);
    }

    @SuppressWarnings("unchecked")
    private T initializeT(Constructor constructor, Object[] args)
    {
        T result = null;

        try
        {
            result = (T) constructor.newInstance(args);
        }
        catch (java.lang.InstantiationException ex)
        {
        }
        catch (java.lang.IllegalAccessException ex)
        {
        }
        catch (java.lang.reflect.InvocationTargetException ex)
        {
        }

        return result;
    }

    public static void main(String argv[]) throws Exception
    {
        Constructor constructor = 
            Double.class.getConstructor(new Class[]{double.class});

        Object[] initialValue = new Object[] { new Double(42.0) };

        MyGenericClass<Double> myGenericClass = 
            new MyGenericClass<Double>(constructor, initialValue);

        myGenericClass.doSomething();

    }
}
于 2013-06-30T19:43:54.450 回答
3

只需使用zero界面上已有的方法进行初始化sum

T sum = arithmetics.zero();

对于非零初始化,您还可以添加采用longdouble值并T为它们返回的方法:

public interface Arithmetics<T> {

    public T zero();
    public T create(long l);
    public T create(double d);
    public T add( T a, T b );
    public T subtract( T a, T b);
    public T multiply (T a, T b);
    public T parseString( String str );
    public String toString( T a );
}

然后实现它们:

public Double create(long l) {
    return new Double(l);
}

public Double create(double d) {
    return new Double(d);
}

最后,使用它们:

T one = arithmetics.create(1);
于 2013-06-30T18:38:27.973 回答