Как вызвать шаблонную Java-функцию из Scala
У меня есть библиотека Java с некоторыми общими контейнерами:
public interface IColumnTable<T extends IColumn<?, ?>>
{
}
public interface IColumn<D extends IColumnValues<?>, M extends IMetaData>
{
}
public interface IColumnValues<E>
{
}
public interface IMetaData
{
}
и фабричный метод, чтобы получить конкретные экземпляры из них
public interface StorageFactory
{
IColumnTable<? extends IColumn<? extends IColumnValues<?>, ? extends IMetaData>> read(String tableName) throws IOException;
}
И у меня также есть служебный метод, который печатает мои подстановочные таблицы (путем приведения и преобразования значений)
public class TableConverterUtil
{
public static <T, V> IColumnTable<IColumn<IColumnValues<T>, IMetaData>> getPureTypedTable(
IColumnTable<? extends IColumn<? extends IColumnValues<V>, ? extends IMetaData>> tableRaw,
Class<T> type,
Optional<Function<V, Optional<T>>> converter
)
}
(Обратите внимание на использование V вместо? В качестве параметра для IColumnValues)
В Java я могу вызвать следующее, чтобы получить таблицу со значениями Double:
try {
IColumnTable<IColumn<IColumnValuesExact<Double>, IMetaData>> myDoubleTable = TableConverterUtil.getPureTypedTable(
StorageManagerUtil.getDefault().get("default").read("myTableName"),
Double.class,
Optional.empty()
);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (StorageManagerInstantiationException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Однако в Scala мне кажется, что я не могу заставить работать тот же вызов:
TableConverterUtil.getPureTypedTable(
StorageManagerUtil.getDefault.get("default").read("myTableName"),
classOf[Double],
Optional.empty()
)
Результаты в:
[file and line number]: no type parameters for method emp
ty: ()java.util.Optional[T] exist so that it can be applied to arguments ()
[error] --- because ---
[error] undetermined type
[error] Optional.empty()
[error] ^
[error] [file and line number]: type mismatch;
[error] found : java.util.Optional[T]
[error] required: java.util.Optional[java.util.function.Function[?,java.util.Op
tional[?]]]
[error] Optional.empty()
[error]
^
Я также попробовал это с нулем вместо Optional.empty(), что приводит к:
[error] [file & line no]: no type parameters for method get
PureTypedTable: (x$1: com.wwa.data.interfaces.IColumnTable[_ <: com.wwa.data.int
erfaces.IColumn[_ <: com.wwa.data.interfaces.IColumnValues[V], _ <: com.wwa.data
.interfaces.IMetaData]], x$2: Class[T], x$3: java.util.Optional[java.util.functi
on.Function[V,java.util.Optional[T]]])com.wwa.data.interfaces.IColumnTable[com.w
wa.data.interfaces.IColumn[com.wwa.data.interfaces.IColumnValuesExact[T],com.wwa
.data.interfaces.IMetaData]] exist so that it can be applied to arguments (com.w
wa.data.interfaces.IColumnTable[?0], Class[Double], Null)
[error] --- because ---
[error] argument expression's type is not compatible with formal parameter type;
[error] found : com.wwa.data.interfaces.IColumnTable[?0(in method doWork)] wh
ere type ?0(in method doWork) <: com.wwa.data.interfaces.IColumn[_ <: com.wwa.da
ta.interfaces.IColumnValues[_], _ <: com.wwa.data.interfaces.IMetaData]
[error] required: com.wwa.data.interfaces.IColumnTable[_ <: com.wwa.data.interf
aces.IColumn[_ <: com.wwa.data.interfaces.IColumnValues[?V], _ <: com.wwa.data.i
nterfaces.IMetaData]]
[error] TableConverterUtil.getPureTypedTable(
[error] ^
[error] [file & line No.]: type mismatch;
[error] found : com.wwa.data.interfaces.IColumnTable[?0(in method doWork)] wh
ere type ?0(in method doWork) <: com.wwa.data.interfaces.IColumn[_ <: com.wwa.da
ta.interfaces.IColumnValues[_], _ <: com.wwa.data.interfaces.IMetaData]
[error] required: com.wwa.data.interfaces.IColumnTable[_ <: com.wwa.data.interf
aces.IColumn[_ <: com.wwa.data.interfaces.IColumnValues[V], _ <: com.wwa.data.in
terfaces.IMetaData]]
[error] StorageManagerUtil.getDefault.get("default").read("myTableName"),
[error]
^
Я также попробовал вспомогательный метод, чтобы перехватить подстановочный знак, данный IColumnValues, чтобы я мог фактически передать функцию. Но не повезло, кажется, что всякий раз, когда я пытаюсь связать один из внутренних _ с параметрами именованного типа, компилятор выдает поместиться.
Я могу немного изменить библиотеку Java, но думаю, что мне чего-то не хватает, поскольку Scala и Java должны быть полностью совместимы.
Есть ли способ вызвать getPureTypedTable из Scala без изменения Java? (с функцией преобразователя, не равной NULL или необязательной переменной)
PS: извините за длинную почту, поздно в пятницу, и я смотрю на это некоторое время сейчас
1 ответ
Думаю, я нашел ответ в этой ветке группы scala https://groups.google.com/forum/
Он включает в себя сопоставление с образцом для именования типа, но также выдает предупреждение компилятора о непроверенных типах, которые, я думаю, можно игнорировать...
def callWildcardTableFunction(table: IColumnTable[_ <: IColumn[_ <: IColumnValues[_], _ <: IMetaData]])
: IColumnTable[IColumn[IColumnValues[Double], IMetaData]] = table match
{
case boundTable: IColumnTable[IColumn[IColumnValues[valueType], m]] @unchecked =>
TableConverterUtil.getPureTypedTable[Double, valueType](boundTable, classOf[Double], null)
}
Если у кого-то есть лучший способ / это на самом деле не безопасно, пожалуйста, скажите!