如何编写比较器来在Java中泛化两种类型的对象?

时间:2017-06-03 01:56:03

标签: java generics comparator

我使用以下方法调用来返回对象列表。

final List<LinkModelSpi> documentLinks = this.documentLinksModelSpi.getDocumentLinks(); 

接下来,我使用Comparator对此列表进行排序。但List包含两种类型的对象。一些功能上它返回 DocumentLinkModelImpl 以及它返回的一些函数操作 LinkModelImpl .So,Comparator函数传递了Class Cast异常当 LinkModelImple 更改为 DocumentLinkModelImpl 时。这两个对象父类型也都是deference。

// Add This method for Sorting Document Link View Same Order. Now The application display same links order
// in catalogue administration edit drug window
Collections.sort(documentLinks,
    (Comparator<? super LinkModelSpi>) new Comparator<DocumentLinkModelImpl>() {
      @Override
      public int compare(DocumentLinkModelImpl o1, DocumentLinkModelImpl o2) {
        return o1.getResource().compareTo(o2.getResource());
      }

      /*@Override
      public int compare(LinkModelImpl o1,LinkModelImpl o2) {
        return o1.getResource().compareTo(o2.getResource());
      }*/

});

我需要一些专家的帮助来解决这个问题?

注意: 我新添加了上面的排序方法,但在我添加它之前,该方法对所有对象都执行得很好。

final List<LinkModelSpi> documentLinks = this.documentLinksModelSpi.getDocumentLinks();

for (final LinkModelSpi documentLinkModelSpi : documentLinks)
{
  //noinspection unchecked
  ((DtoPresentationModelSpi<?, MDTO>) documentLinkModelSpi).addPropertyChangeListener(this);
}

-------------- ERROR ----------------------

Presentation.impl.LinkModelImpl cannot be cast to Presentation.impl.DocumentLinkModelImpl
    at Client.catadmin.view.component.AbstractEditLinkDialog$1.compare(AbstractEditLinkDialog.java:171)
    at java.util.TimSort.countRunAndMakeAscending(TimSort.java:355)
    at java.util.TimSort.sort(TimSort.java:220)

2 个答案:

答案 0 :(得分:3)

让我更清楚我的评论意思。如果DocumentLinkModelImplLinkModelImpl都有方法getResource,那么很可能是超类型(无法看到它是由两者还是超类实现的接口)LinkModelSpi方法getResource也是如此。那么你可以写:

Collections.sort(documentLinks,
    new Comparator<LinkModelSpi>() {
      @Override
      public int compare(LinkModelSpi o1, LinkModelSpi o2) {
        return o1.getResource().compareTo(o2.getResource());
      }
});

即使情况并非如此,你也可以解决这个问题:

Collections.sort(documentLinks,
    new Comparator<LinkModelSpi>() {
      @Override
      public int compare(LinkModelSpi o1, LinkModelSpi o2) {
        String resource1, resource2; // String or whatever type the getResource method returns
        if (o1 instanceof DocumentLinkModelImpl) {
            resource1 = ((DocumentLinkModelImpl)o1).getResource();
        } else if (o1 instanceof LinkModelImpl) {
            resource1 = ((LinkModelImpl)o1).getResource();
        } else {
            throw new IllegalArgumentException();
        }
        if (o2 instanceof DocumentLinkModelImpl) {
            resource2 = ((DocumentLinkModelImpl)o2).getResource();
        } else if (o2 instanceof LinkModelImpl) {
            resource2 = ((LinkModelImpl)o2).getResource();
        } else {
            throw new IllegalArgumentException();
        }
        return resource1.compareTo(resource2);
      }
});

(当然你会将instanceof检查纳入自己的方法来清理代码)

答案 1 :(得分:0)

我解决了上面添加此代码片段的麻烦。但是,如果我们知道任何一个,我需要知道一些更高级的解决方案。

if (Checks.checkNotNullAndNonEmpty(documentLinks)) {
  if (this.documentLinksModelSpi.getDocumentLinks().iterator().next() instanceof DocumentLinkModelSpi) {

  } else {

  }
}