Java中具有多个条件的动态排序字段

时间:2019-05-07 20:31:39

标签: java sorting dynamic criteria

我需要对动态对象列表和动态搜索条件列表进行排序。所以我有一个接受

的方法
sort(List<SortCriterion> sortCriteria, List<T> searchList){

}

现在,如果我有一个看起来像这样的TradeOperator对象列表。我可以在sort方法中发送其他对象,例如CarOperator。

 public class TradeOperator implements SearchResult{

      private String name;
      private String address;
      private String status;
      private Long   regNum;
      private Double price;
}

SortCriterion代码如下所示。

 public class SortCriterion{

       protected String field;   
       protected String direction;
    }

,该字段可以是TradeOperator.name,方向= ASC或DESC。现在,我可以通过具有多个字段和方向的SortCriterion列表。

这就是我所做的。我在MAP sortValues中收集了所有ASC字段和所有DESC字段,其中key = ASC和DESC,并且Values是该字段的列表。然后以ASC排序,然后以相反的顺序进行排序。

sort(List<SortCriterion> sortCriteria, List<T> searchList){
   Map<String, Set<String>> sortValues = new HashMap<>();
    populateSortedValues(sortCriteria, "ASC", sortValues);//group fields on ASC or DESC as key = ASC and Value = List of Fields
    populateSortedValues(sortCriteria, "DESC", sortValues);

    if (null != sortValues.get(Constants.ASC_ORDER))
    {
        for (String ascendingOrder : 
         sortValues.get(Constants.ASC_ORDER))
        {
        Collections.sort(list, new SearchResultComparator(ascendingOrder));
        }
    }

    if (null != sortValues.get(Constants.DESC_ORDER))
    {
        for (String descOrder : sortValues.get(Constants.DESC_ORDER))
        {
        Collections.sort(list, new 
               SearchResultComparator(descOrder).reversed());
        }
    }
}

这是SearchResultComparator类。

public class SearchResultComparator implements Comparator<Object> {

private String getter;


public SearchResultComparator(String field) {
    this.getter = "get" + field.substring(0, 1).toUpperCase() + field.substring(1);

}

@SuppressWarnings("unchecked")
public int compare(Object o1, Object o2) {
    try {
        if (o1 != null && o2 != null) {
            o1 = o1.getClass().getMethod(getter, new Class[0]).invoke(o1, new Object[0]);
            o2 = o2.getClass().getMethod(getter, new Class[0]).invoke(o2, new Object[0]);
            if(isDouble(o1.toString()) && isDouble(o2.toString())){
                Double d1 =  Double.parseDouble(o1.toString());
                Double d2 =  Double.parseDouble(o2.toString());
                return (d1 == null) ? -1 : ((d2 == null) ? 1 : ((Comparable<Double>) d1).compareTo(d2));
            }
        }
    } catch (Exception e) {
        throw new SystemException("Cannot compare " + o1 + " with " + o2 + " on " + getter, e);
    }

        return (o1 == null) ? -1 : ((o2 == null) ? 1 : ((Comparable<Object>) o1).compareTo(o2));


}

public boolean isDouble(String value)
{
try
{
    Double.parseDouble(value);
    return true;
}
catch (NumberFormatException e)
{
    return false;
}
}

}

它以一种方式排序。如果我在ASC中对一个字段进行排序,则在DESC中另一个字段将丢失ASC排序。请帮忙。我是Java的新手。

1 个答案:

答案 0 :(得分:0)

您不需要SearchResultComparator或将ASC和DESC分开。使用此技术对任何订单进行链式排序。

sort(List<SortCriterion> sortCriteria, List<T> list){
    Collections.sort(list, new Comparator<T>()
            {
            public int compare(T one, T two)
                {
            CompareToBuilder compToBuild = new CompareToBuilder();
            sortCriteria.stream().forEachOrdered(sc->{
                String fieldName = sc.getField();
                String direction = sc.getDirection();
                String fv1 = getFieldValue(fieldName, one);
                String fv2 = getFieldValue(fieldName, two);
                if(direction.equals("ASC")){            
                    compToBuild.append(fv1,fv2);                
                }
                if(direction.equals("DESC")){               
                    compToBuild.append(fv2,fv1);

                }
            });
            return compToBuild.toComparison();

                }

            });
}

使用反射获取字段值

 private String getFieldValue(String fieldName, T object)
    {
    Field field;
    try
    {
        field = object.getClass().getDeclaredField(fieldName);
    }
    catch (NoSuchFieldException e)
    {
        logger.error(e);
        return null;// or ""
    }
    field.setAccessible(true);
    try
    {
        return (String) field.get(object));
    }
    catch (IllegalAccessException e)
    {
        logger.error(e);
        return null;// or ""
    }