java-基于泛型和反射机制的通用比较器实现
一、前言
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-基于泛型和反射机制的通用比较器实现的更多相关文章
- java的泛型与反射机制
什么是泛型? 泛型,即“参数化类型”.顾名思义,就是将类型由原来的具体的类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使用/调用时传入具体的类型(类型实参) ...
- Java基础系列 - 泛型和反射机制
package com.test5; import java.lang.reflect.Field; import java.lang.reflect.Method; /** * Java泛型和反射机 ...
- JAVA动态性之一一反射机制reflection
package com.bjsxt.reflection.test.bean; public class User { private int id; private int age; private ...
- Java 核心类库之反射机制
1:什么是反射机制? 2:反射机制它可以做什么呢? 3:反射机制对应的API又是什么? 1):通过反射机制来获取一个对象的全限定名称(完整包名),和类名: 2):实例化Class对象 3):获取对象的 ...
- Java基于注解和反射导入导出Excel
代码地址如下:http://www.demodashi.com/demo/11995.html 1. 构建项目 使用Spring Boot快速构建一个Web工程,并导入与操作Excel相关的POI包以 ...
- 利用JAVA反射机制设计通用的DAO
利用JAVA反射机制设计一个通用的DAO 反射机制 反射机制指的是程序在运行时能够获取自身的信息.在java中,只要给定类的名字, 那么就可以通过反射机制来获得类的所有信息. 反射机制创建类对象 ...
- 关于Java中泛型、反射和注解的扫盲篇
泛型 泛型概念 泛型是在JDK1.5之后引入的,旨在让我们写出更加通用化,更加灵活的代码.通用化的手段在于让数据类型变得参数化,定义泛型时,对应的数据类型是不确定的,泛型方法被调用时,会指定具体类 ...
- java的RTTI和反射机制
RTTI,即Run-Time Type Identification,运行时类型识别.RTTI能在运行时就能够自动识别每个编译时已知的类型. 很多时候需要进行向上转型,比如Base类派生出Derive ...
- java.lang.Class<T> -- 反射机制
1反射机制是什么 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为jav ...
随机推荐
- BZOJ_1916_[Usaco2010 Open]冲浪_分层图+拓扑排序+DP
BZOJ_1916_[Usaco2010 Open]冲浪_分层图+拓扑排序+DP Description 受到秘鲁的马丘比丘的新式水上乐园的启发,Farmer John决定也为奶牛们建 一个水上乐园. ...
- [51nod 1129] 字符串最大值(kmp)
传送门 题目大意 求一个字符串的前 缀出现次数乘以长度的最大值. 题解 暴力枚举每一个前缀求出现次数再乘以常数取最大 这样做会T几个点 看了老师的做法是任意前缀出现的次数,它的next也会出现这些次数 ...
- CS231n 2016 通关 第二章-KNN
课程内容全纪录: 1.讲解图像分类的难点 1.光照强度 2.主体变形 3.主体与背景咬合 4.主体与背景相接近 5.同类别间存在区别 2.KNN 1.最近邻算法 2.Knn 3.hyperpara ...
- UI控件初始化问题:initWithFrame和initWithCoder、aweakFromNib的执行
在iOS学习和程序开发过程中,我们经常会遇到一些自定义UI控件或控制器在初始化时出现问题,尤其在大家刚开始接触时,几种初始化方法的作用以及调用的时机往往容易混淆,这也跟我们对iOS程序设计中,类的创建 ...
- 在idea中创建maven父子工程,子工程无法导入父工程依赖的问题
创建maven父子工程时遇到一个问题,当子工程的名称前缀和父工程的名称一样时,子工程会出现一系列的问题.比如我的父工程名称是microservicecloud,子工程名称是microservicecl ...
- E20170417-sl
recursive adj.递归的 strategy n.战略
- MySQL 错误码对照
1005:创建表失败 1006:创建数据库失败 1007:数据库已存在,创建数据库失败 1008:数据库不存在,删除数据库失败 1009:不能删除数据库文件导致删除数据库失败 1010:不能删除数据目 ...
- 键值编码 KVC
http://www.cnblogs.com/dyf520/p/3805297.html 1,什么是Key-Value Coding? Key-Value Coding是一种间接访问对象属性的机制,使 ...
- 编译安装Apache:出现错误configure: error: mod_deflate
在进行编译安装Apache时,出现如下错误 checking whether to enable mod_deflate... configure: error: mod_deflate has be ...
- UIImageView 使图片圆形的方法
UIImageView 圆形的两种方法 1.cornerRadius (tableView,collectionView尽量避免使用,影响性能) //想要圆角 cornerRadius必须是 imag ...