一、前言

Java的比较器是用来对List集合进行排序用的,分为内部比较器和外部比较器两类

内部比较器:被排序的类要 implements Comparable 类,并实现compareTo方法。

外部比较器:需要实现一个implements  Comparator的比较器,实现compare方法,并在sort方法中将该比较器当参数传入。

(具体实现,网络上有很多资料,此处不再赘述)

二、发现问题

外部比较器虽然相比内部比较器 实现了解耦,代码侵入小,但是两者都不能摆脱同一个麻烦的问题:

那就是对每个待排序的对象,均需单独实现一个比较器类。如果项目中有很多类都需要排序,那就需要重复劳动,写很多个比较器类了。

本文的目的,即通过泛型和反射机制的应用,来设计实现一个通用的外部比较器。该比较器对任何待排序的对象均能适用,减少低级的重复劳动。

三、实现思路

四、实现

话不多说,直接贴代码:

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.lang.reflect.Method;
import java.text.Collator;
import java.util.Comparator;

/**
 * @Description:
指定需根据某字段排序的方法名,进行排序,返回类型自动判断,目前支持Integer String
 * @Author: wangzhen3
 * @CreationTime: 2018/5/29 15:02
 * @ModifiedBy:
 
*/
public class CompareByFunctionName<T> implements Comparator<T> {
    private static
final Logger logger = LoggerFactory.getLogger(CompareByFunctionName.class);
    /**
     *
需比较字段的方法名 如 getStatus
     */
   
private String functionName = "";
    /**
     *
是否倒序
    
*/
   
private Boolean invert = false;

public CompareByFunctionName(String functionName){
        this.functionName = functionName;
        this.invert = false;
    }

public CompareByFunctionName(String functionName, Boolean invert){
        this.functionName = functionName;
        this.invert = invert;
    }
    @Override
    public int compare(T o1, T o2) {
        Object ret1;
        Object ret2;
        Method method;
        try {
            method =  o1.getClass().getMethod(functionName);
            ret1 =
method.invoke(o1);
            ret2 =
method.invoke(o2);
           if(ret1 instanceof Integer){
               return !invert? (Integer)ret1 - (Integer)ret2 :
                       (Integer)ret2 -
(Integer)ret1;
           }else if(ret1 instanceof String){
               return !invert? Collator.getInstance(java.util.Locale.CHINA).compare(ret1,ret2) :
                       Collator.getInstance(java.util.Locale.CHINA).compare(ret2,ret1);
           }else{
               //直接toString 比较
               return !invert? Collator.getInstance(java.util.Locale.CHINA).compare(ret1.toString(),ret2.toString())
:
                       Collator.getInstance(java.util.Locale.CHINA).compare(ret2.toString(),ret1.toString());
           }

}catch (Exception e){
            logger.error(e.getMessage());
        }
        logger.error("比较失败,o1={},o2={}",o1.toString(),o2.toString());
        return 0;
    }
}

关键步骤说明:

1)java反射机制中的Field 和 Method 方法,网上有很多博客介绍,此处不赘述。

2)functionName 必须为public方法,不然无访问权限。

或许你会有疑问,为什么此处反射是使用Method获得数据值 而不直接用字段Field来获取数据值,原因是字段一般被声明为private,所以通过Field一般无访问权限,无法取出数据,而Method ,比如status 字段的getStatus方法,通常是public ,所以能取出数据。

3)当然,functionName 是可以换成 字段名称fieldName 的,只需要新增转换逻辑,根据java的驼峰式命名约定,把fieldName 转换为functionName 即可

4)Collator.getInstance(java.util.Locale.CHINA)
.compare()

为了实现中文字符串排序功能

五、实际应用

项目中使用示例:

根据字段status对List<MonitorAlarmData>  进行倒序排列,使status=1的排在开头,status=-1的排在末尾

public class MonitorAlarmData {

    private String id; //problem id

    private Integer status;//当前状态 -1 已忽略; 0 OK; 1 problem;

    public String getId() {
        return id;
    }     public void setId(String id) {
        this.id = id;
    }     public Integer getStatus() {
        return status;
    }     public void setStatus(Integer status) {
        this.status = status;
    }
   }
List<MonitorAlarmData> monitorAlarmDataList = new ArrayList<>();
//省略代码,往monitorAlarmDataList 中添加数据
//排序
monitorAlarmDataList.sort(new CompareByFunctionName("getStatus",true));

java-基于泛型和反射机制的通用比较器实现的更多相关文章

  1. java的泛型与反射机制

    什么是泛型? 泛型,即“参数化类型”.顾名思义,就是将类型由原来的具体的类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使用/调用时传入具体的类型(类型实参) ...

  2. Java基础系列 - 泛型和反射机制

    package com.test5; import java.lang.reflect.Field; import java.lang.reflect.Method; /** * Java泛型和反射机 ...

  3. JAVA动态性之一一反射机制reflection

    package com.bjsxt.reflection.test.bean; public class User { private int id; private int age; private ...

  4. Java 核心类库之反射机制

    1:什么是反射机制? 2:反射机制它可以做什么呢? 3:反射机制对应的API又是什么? 1):通过反射机制来获取一个对象的全限定名称(完整包名),和类名: 2):实例化Class对象 3):获取对象的 ...

  5. Java基于注解和反射导入导出Excel

    代码地址如下:http://www.demodashi.com/demo/11995.html 1. 构建项目 使用Spring Boot快速构建一个Web工程,并导入与操作Excel相关的POI包以 ...

  6. 利用JAVA反射机制设计通用的DAO

    利用JAVA反射机制设计一个通用的DAO 反射机制 反射机制指的是程序在运行时能够获取自身的信息.在java中,只要给定类的名字,    那么就可以通过反射机制来获得类的所有信息. 反射机制创建类对象 ...

  7. 关于Java中泛型、反射和注解的扫盲篇

    泛型 泛型概念   泛型是在JDK1.5之后引入的,旨在让我们写出更加通用化,更加灵活的代码.通用化的手段在于让数据类型变得参数化,定义泛型时,对应的数据类型是不确定的,泛型方法被调用时,会指定具体类 ...

  8. java的RTTI和反射机制

    RTTI,即Run-Time Type Identification,运行时类型识别.RTTI能在运行时就能够自动识别每个编译时已知的类型. 很多时候需要进行向上转型,比如Base类派生出Derive ...

  9. java.lang.Class<T> -- 反射机制

    1反射机制是什么 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为jav ...

随机推荐

  1. #ifndef <标识> #define <标识> #endif

    头件的中的#ifndef,这是一个很关键的东西.比如你有两个C文件,这两个C文件都include了同一个头文件.而编译时,这两个C文件要一同编译成一个可运行文件,于是问题来了,大量的声明冲突. 还是把 ...

  2. DC 兼容的DC

    DC是 "Device Content" , MS VC++ 的 MFC图形设备接口 的 设备描述表.它是MFC的主要对象之一.通过CDC类进行各种绘图操作,例如选笔,选色,选涂色 ...

  3. JAVA基础--异常10

    一.Object类简介 1.Object类简介 Object,是Java一切对象的最高父类:如果一个类没有显示继承一个类,他的父类就是Object: 它描述的是Java世界里所有对象最最共性的信息,它 ...

  4. CoreBluetooth Central模式 Swift版

    也是醉了,CB这个API到现在也没有Swift的文档.最新的文档还是3年前还是4年前的OC版的,被雷的外焦里嫩的.自己一点一点写成Swift还各种报错,最坑的是这些错误压根找不到解决方案.索性自己做个 ...

  5. 当打开一个.h或.cpp文件时, Solution Explorer就自动展开文件所在的目录

    当打开一个.h或.cpp文件时,  Solution Explorer就自动展开文件所在的目录: 如果不想展开: Tools           -> Options           -&g ...

  6. 优化 SQL 查询:如何写出高性能SQL语句

    1. 首先要搞明白什么叫执行计划? 执行计划是数据库根据SQL语句和相关表的统计信息作出的一个查询方案,这个方案是由查询优化器自动分析产生的,比如一条SQL语句如果用来从一个 10万条记录的表中查1条 ...

  7. java中多线程的两种创建方式

    一丶继承Thread类实现多线程 第一步:继承Thread类第二步:重写run()方法第三步:创建继承了Thread类的对象 , 调用start()方法启动. //线程创建方式一 : /* 第一步:继 ...

  8. nginx部署h5项目

    1. nginx部署h5项目 此为windows部署,liunx也类似的 1.1. 前言 部署h5项目还是很简单的,不过对小白来讲一开始可能也是一脸懵逼,这个简单教程针对的是从未部署过前后端分离前端项 ...

  9. Ubuntu 设置文件默认打开的应用

    右键单击该文件,然后点击属性,打开属性面板 然后进入open with的选项,选择应用后,点击 set as default

  10. FZu Problem 2236 第十四个目标 (线段树 + dp)

    题目链接: FZu  Problem 2236 第十四个目标 题目描述: 给出一个n个数的序列,问这个序列内严格递增序列有多少个?不要求连续 解题思路: 又遇到了用线段树来优化dp的题目,线段树节点里 ...