java - java.util.ArrayList 不能转换为 java.lang.Comparable
问题描述
我有 UniversalComparator,它完成了与排序相关的所有任务,它使用反射 API 来识别方法名称和调用目标。
现在发生的事情是,我需要对 进行排序sites
,以前它是使用 排序的"name"
,现在发生的事情是,用户需要以每月、每季度、半和每年的频率上传文件。
要求
如果文档未以任何频率上传,则该数据以white
颜色表示,其余块以red
颜色表示。
现在有六个站点,第一个站点只有一个白块,第二个站点没有白块。第三个站点没有白块 四个有三个白块 第五个有三个白块 第六个有三个白块
所以计数是 1 0 0 3 3
现在我做了什么,我创建了一个整数的 ArrayList 并存储所有计数现在我需要代表这个计数对 SITE Block 的列表进行排序,所以它应该像
0 0 1 3 3
代码
package com.lear.common.utility;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Comparator;
public class UniversalComparator implements Comparator {
/**
* static final int ASCENDING
*/
public static final int ASCENDING = 1;
/**
* static final int DESCENDING
*/
public static final int DESCENDING = -1;
/*
* int for descAscIndicator
*/
private int descAscIndicator = 1;
/*
* String to store method Name
*/
private String methodName = "toString";
/**
* Constructor for UniversalComparator
*
* @param descAscIndicator
* int to store descAscIndicator.
*/
public UniversalComparator(int descAscIndicator) {
this.descAscIndicator = descAscIndicator;
}
/**
* Constructor for UniversalComparator
*
* @param methodName
* : name of method as criteria
* @param descAscIndicator
* : order of sorting
*/
public UniversalComparator(String methodName, int descAscIndicator) {
this(descAscIndicator);
this.methodName = methodName;
}
/**
* This Method compare Two Objects
*
* @param o1
* : An Instance of Object.
* @param o2
* : An Instance of Object.
* @return int
*/
public int compare(Object o1, Object o2) {
Object comp1 = null;
Object comp2 = null;
try {
Method o1_Method = (o1.getClass()).getMethod(methodName, null);
Method o2_Method = (o2.getClass()).getMethod(methodName, null);
comp1 = o1_Method.invoke(o1, null);
comp2 = o2_Method.invoke(o2, null);
} catch (NoSuchMethodException e) {
throw new RuntimeException("Method does not exist" + e.getMessage());
} catch (IllegalAccessException e) {
throw new RuntimeException("Illegal access" + e.getMessage());
} catch (InvocationTargetException e) {
throw new RuntimeException("InvocationTargetException"
+ e.getMessage());
}
Comparable c1 = (Comparable) comp1;
Comparable c2 = (Comparable) comp2;
return c1.compareTo(c2) * descAscIndicator;
}
/**
* Check for Equality obect
*
* @param obj
* : An Instance of Object.
* @return boolean return true if equal or false if not
*/
public boolean equals(Object obj) {
return this.equals(obj);
}
}
记分卡管理器.java
List<Integer> naSiteDataList = new ArrayList<Integer>();
public String getComparativeSiteAnalysis(Integer divId, String lang, int selectedYear) {
// PLENTY OF CODE HERE
int annualDataCount = site.getComparativeColorAnnual().equalsIgnoreCase("White") ? 1 : 0;
naSiteDataCount = monthlyDataCount + quaterlyDataCount + semiAnnualDataCount + annualDataCount;
naSiteDataList.add(naSiteDataCount);
naSiteCounter.add(naSiteDataCount);
site.setNaSiteCount(naSiteDataList);
site.setNaSiteCounter(naSiteCounter);
System.out.println("datacount is" + naSiteDataCount);
}
// THIS LINE
Collections.sort(sites, new UniversalComparator("getNaSiteCount", 1));
站点.java
public class Site{
// lot of properties
private List<Integer> naSiteCount;
public List<Integer> getNaSiteCount() {
return naSiteCount;
}
public void setNaSiteCount(List<Integer> naSiteCount) {
this.naSiteCount = naSiteCount;
}
}
解决方案
反射是一种非常糟糕的方法。您在此代码中绝对没有类型安全性。您无法保证以字符串形式提供的名称的方法确实存在。你不能保证它不需要任何参数。你不知道它可能抛出什么异常。
如果您使用的是 Java 8 plus,那么实现特定的比较器将是微不足道的:
Comparator<Object> comparator = Comparator.comparing(Object::toString);
但是 Java 8 特性的缺乏并不是使用反射的理由。
定义一个抽象类:
abstract class AbstractComparator<T, C extends Comparable<? super C>> implements Comparator<T> {
abstract C toComparable(T object);
@Override public int compare(T a, T b) {
return toComparable(a).compareTo(toComparable(b));
}
Comparator<T> reverse() {
return new Comparator<T>() {
@Override public int compare(T a, T b) {
return toComparable(b).compareTo(toComparable(a));
}
}
}
}
然后针对您的特定情况实施此操作:
Comparator<Object> comparator = new AbstractComparator<Object, String> {
@Override String toComparable(Object object) { return object.toString(); }
}.reverse();
如果您正在使用 Guava 或其他带有Function
-like 类的库,或者很乐意自己定义它,您当然可以采用组合方法,而不是使用继承:
Comparator<Object> comparator = new ConcreteComparator<>(new Function<Object, String>() {
@Override public String apply(Object object) {
return object.toString();
}
});
如果您不在比较器或您正在比较的事物周围使用原始类型,那么ClassCastException
s 现在将是不可能的。
推荐阅读
- sql - 如何计算超过 17 周的工作时间
- php - 使用 Homestead 和 Vagrant 设置 phpMyAdmin 的问题
- android-camera2 - 2秒延迟点击抓图按钮后抓图
- javascript - 防止 TypeError 的方法:值不可迭代
- android - 在 exoplayer Android 中播放加密的 .m3u8 流
- sql - 如何在 SQL 中为密码构造正则表达式?
- sas - 合并日期
- java - Android SQLite:如果没有给出唯一键,如何确保批量上传和删除成功时的一致性?
- python - 如何在 Python 中削减所需的无限深度 JSON 数据?
- django - Django Graphene Relay order_by (OrderingFilter)