Java实现储存对象并按对象某属性排序的几种方法示例
作者:.长卿 发布时间:2022-05-04 18:05:57
本文实例讲述了Java实现储存对象并按对象某属性排序的几种方法。分享给大家供大家参考,具体如下:
在编程的时候,经常会出现对某一种类的对象们按照某属性进行自定义的排序,比如:学生对象按照age大小排序。
有一种方法就是把age单独提出来排好序,然后按照ages数组的顺序把students重存一次。但是这样太繁琐了,有没有更好的方法呢?
有滴~
第一种,可以实现边添加边排序,需要用到TreeSet。
第二种,用数组存放对象们,但是不需单独取出某属性排列好再重存,而是在原数组上用比较器重新排一次序。需要用到Arrays.sort(arr,comparator)。
第三种,用集合类中的list的子类存放对象们,然后排序。需要用到Collections.sort(list,comparator)。
以下分别讨论:
一、TreeSet
创建:
序号 | 构造函数的说明 |
---|---|
1 | TreeSet () 此构造函数构造空树集,将在根据其元素的自然顺序按升序排序。 |
2 | TreeSet (集合 c) 此构造函数生成树的集合,它包含的元素的集合 c。 |
3 | TreeSet (比较器 comp) 此构造函数构造一个空树集,将根据给定的比较器进行排序。 |
增:
boolean | add(E e) 将指定的元素添加到这套,如果它已不存在。 |
boolean | addAll(Collection<? extends E> c) 在加入这一组指定的集合中添加的所有元素。 |
删:
boolean | remove(Object o) 从这一组中移除指定的元素,如果它存在。 |
void | clear() 从这一组中移除所有元素。 |
查:
Comparator<? super E> | comparator() 返回用于排序在这集,或空元素,如果这套使用自然排序其元素的比较。 |
boolean | contains(Object o) 如果此集合包含指定的元素,则返回true 。 |
boolean | isEmpty() 如果此集不包含任何元素,则返回true 。 |
Iterator<E> | iterator() 返回迭代器中这套以升序排序的元素。 |
| |
int | size() 在这套 (其基数) 中返回的元素的数目。 |
|
遍历:通过迭代器遍历。
Iterator it=treeset.iterator();
while(it.hasNext()){
//操作当前结点。
}
代码实现:
TreeSet是一个有序集合,TreeSet中的元素将按照升序排列。
TreeSet存储对象的时候, 可以排序, 其中Integer有默认排序方法, String有默认排序方法,,而自定义的类对象存储的时候则没有顺序,需要自定义排序算法。
如果想把自定义类的对象存入TreeSet进行排序,或者对int,String对象想定义自己的排序方法,有以下两种方法:
排序的第一种方式:
让元素自身具备比较性。让元素实现Comparable接口,覆盖compareTo方法,在方法内定义比较算法, 根据大小关系, 返回正数负数或零。在使用TreeSet存储对象的时候, add()方法内部就会自动调用compareTo()方法进行比较, 根据比较结果使用二叉树形式进行存储。
排序的第二种方式:
自定义比较器。 定义一个类实现Comparator接口,覆盖compare方法。将该Comparator接口子类对象传递给TreeSet集合构造函数。
第一种:类定义时实现Comparable接口,定义自身的比较算法。
此处以Person类为例,把人名和年龄作为对象属性,存放进treeset中,按照年龄的升/降序保存。
public class TreeSetTest {
public static void main(String[] args) {
TreeSet people=new TreeSet();
people.add(new Person("小明", 20));
people.add(new Person("小张", 30));
people.add(new Person("小刘", 18));
people.add(new Person("小林", 17));
people.add(new Person("小刘", 35));
Iterator it=people.iterator();
while(it.hasNext()){
System.out.println(it.next());
}
}
}
class Person implements Comparable{//定义类时,实现比较接口
String name;
int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String toString(){
return "姓名:"+name+",年龄:"+age;
}
/*
compareTo(Object o):参数是从根节点开始依次传进来的结点,直到确定合适的位置用来安插新节点。
方法返回三个值,分别对应三种动作:返回1,则继续递进,把新结点与下一层的结点进行比较;
返回0,则该属性值不足以决定两结点位置区别,再定义其他属性的比较算法来进一步比较两对象;
返回-1,则新结点优先级大于当前层,往上一层比较;
*/
public int compareTo(Object o) {
Person curr=(Person) o;
//int result = this.age<curr.age?1:(this.age==curr.age?0:-1);//降序排列:新插入结点与当前比较层结点的属性比较,小则返回1,继续往下一层比较
int result = this.age>curr.age?1:(this.age==curr.age?0:-1);//升序排列:新插入结点与当前比较层结点的属性比较,大则返回1,继续往下一层比较
if(result==0){
result=this.name.compareTo(curr.name);//age相同时,则以姓名的字母序来排列。前面说过,int、string类型是有默认排列算法的,所以此处直接用
}
return result;
}
}
第二种:定义Comparator接口的实现类(比较器),在类中定义对象的比较算法。在创建Treeset时把比较器对象传进去。
public class TreeSetTest {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
TreeSet people=new TreeSet(new MyComparator());//把比较器对象作为TreeSet的构造函数参数
people.add(new Person("小明", 20));
people.add(new Person("小张", 30));
people.add(new Person("小刘", 18));
people.add(new Person("小林", 17));
people.add(new Person("小刘", 35));
Iterator it=people.iterator();//用迭代器遍历treeset
while(it.hasNext()){
System.out.println(it.next());
}
}
}
class Person {
String name;
int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String toString(){
return "姓名:"+name+",年龄:"+age;
}
}
class MyComparator implements Comparator{//实现Comparator接口,自定义比较器,实现compare方法定义比较算法
/*
compare(o1,o2):参数o1是待插入的结点,o2是从树根节点逐层遍历下来的结点,用于当前比较。
方法返回三个值,分别对应三种动作:返回1,则继续递进,把新结点与下一层的结点进行比较;
返回0,则该属性值不足以决定两结点位置区别,再定义其他属性的比较算法来进一步比较两对象;
返回-1,则新结点优先级大于当前层,往上一层比较;
*/
public int compare(Object o1, Object o2) {
Person p1=(Person)o1;
Person p2=(Person)o2;
int result=p1.age<p2.age?1:(p1.age==p2.age?0:-1);//降序排列
//int result=p1.age<p2.age?1:(p1.age==p2.age?0:-1);//升序排列
if(result==0){
result=p1.name.compareTo(p2.name);
}
return result;
}
}
二、用数组存放对象,用比较器改变sort()排序方法。
数组本身有默认的排序方法,针对int、string等基本类型有默认的sort()方法。而针对类对象的排序,可以给sort()方法传进一个比较器对象,赋予其排序的算法。
public class ArraysTest {
public static void main(String[] args) {
Person[] people=new Person[5];
people[0]=(new Person("小明", 20));
people[1]=(new Person("小张", 30));
people[2]=(new Person("小刘", 18));
people[3]=(new Person("小林", 17));
people[4]=(new Person("小刘", 35));
Arrays.sort(people,new MyCompare());//传进来一个比较器对象,使数组按比较器定义的规则来排序
for(Person i:people){
System.out.println(i);
}
}
}
class Person {
String name;
int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String toString(){
return "姓名:"+name+",年龄:"+age;
}
}
class MyCompare implements Comparator<Person>{//定义比较器
/*
重写比较方法compare(o1,o2):
方法传进来两个对象,用两个对象的某属性进行对比,返回一个int。
int>0,则o1排在o2后面;
int<0,则o1排在o2前面;
int=0,则维持原相对位置,即原来存放时o1、o2的前后地址顺序。
*/
public int compare(Person o1, Person o2) {
int result;
if(o1.age>o2.age){
result=1;
}
else if(o1.age<o2.age){
result=-1;
}
else{
result=0;
}
return result;
}
}
第三种:用list的子类:Vector、ArrayList存放对象们,调用Collections.sort(list,comparator)方法进行排序。
public class CollectionsTest {
public static void main(String[] args) {
Vector<Person> people=new Vector<>();//用向量保存对象
//ArrayList<Person> people=new ArrayList<>()://用ArrayList保存对象。
people.add(new Person("小明", 20));
people.add(new Person("小张", 30));
people.add(new Person("小刘", 18));
people.add(new Person("小林", 17));
people.add(new Person("小刘", 35));
Collections.sort(people,new MyComparator());//调用方法进行排序
Iterator it=people.iterator();
while(it.hasNext()){
System.out.println(it.next());
}
}
}
class Person {
String name;
int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String toString(){
return "姓名:"+name+",年龄:"+age;
}
}
class MyComparator implements Comparator<Person>{//实现Comparator接口,自定义比较器,实现compare方法定义比较算法
/*
compare(o1,o2):方法传进两个对象,根据某属性进行比较,返回一个int值。
int>0,o1排在o2后;
int<0,o1排在o2前;
int=0,维持原来存放时的相对位置。
*/
public int compare(Person p1, Person p2) {
int result=p1.age<p2.age?1:(p1.age==p2.age?0:-1);//降序排列
//int result=p1.age<p2.age?1:(p1.age==p2.age?0:-1);//升序排列
if(result==0){
result=p1.name.compareTo(p2.name);
}
return result;
}
}
希望本文所述对大家java程序设计有所帮助。
来源:https://www.cnblogs.com/ygj0930/p/5950971.html


猜你喜欢
- 本文实例讲述了Android编程自定义菜单实现方法。分享给大家供大家参考,具体如下:在android开发的过程中系统自带的菜单往往满足不了开
- 背景 我们知道在.NET Framework中存在四种常用的定时器,他们分别是:1 两个是通用的多线程定时器:Syste
- 本文实例为大家分享了Swing拆分窗格控件JSplitPane的使用方法,供大家参考,具体内容如下package day1115;impor
- 类必须先定义才能使用。类是创建对象的模板,创建对象也叫类的实例化。下面通过一个简单的例子来理解Java中类的定义:public class
- 引言layout: postcategories: Javatitle: 一文带你了解 Spring 的@Enablexxx 注解tagli
- 问题(1)synchronized的特性?(2)synchronized的实现原理?(3)synchronized是否可重入?(4)sync
- 前言poi的解析方式是dom解析,把结果一次都读入内存操作,这样的操作平时是不会有问题的,但是并发量上来的时候就会出现OOM,EasyExc
- 我们很容易能想到,可以用递归来实现汉诺塔游戏。因为要将n(n>1)个盘子从“源”柱子移到“目标”柱子,我们要先把n-1个盘子从“源”柱
- 本文针对java * 进行知识点整理,具体内容如下一. JAVA的 * (比较官方说法) 代理模式是常用的java设计模式,他的特征是代
- 前言各位小伙伴大家好,我是A哥。IDEA上个较大版本的发布,要追溯到4月份了:时隔近4个月,北京时间2020-07-28深夜,Intelli
- 上一篇文章Android进程间通信(IPC)机制Binder简要介绍和学习计划简要介绍了Android系统进程间通
- 一、前言在spring中,定义rabbitMq的消费者可以相当方便,只需要在消息处理类或者类方法加上@RabbitListener注解,指定
- 代码如下: package swt_jface.demo9; import org.eclipse.swt.SWT; import org.
- Java%(取模运算)Java的取模运算1.实现算法public static double ramainder(double divide
- 给大家看个计算题,看看大家的算术能力。0.1 +0.1 +0.1 - 0.3 等于几?大家可能会说这么简单的问题,是不是看不起我?肯定等于0
- 一、先看结果1.1创造营2020撑腰榜前三甲创造营2020撑腰榜前三名分别是 希林娜依·高、陈卓璇 、郑乃馨>>>df1[
- 本文实例讲述了Java实现合并两个有序序列算法。分享给大家供大家参考,具体如下:问题描述输入:序列A<a0,a1,a2,...aq,a
- 代码很简单,如下所示: package swt_jface.demo1; import org.eclipse.swt.SWT; impor
- 前言相信每位Android开发者都用过Toast,都知道是弹出消息的。类似于js里面的alert,C#里面的MesageBox。当然andr
- 本文实例为大家分享了QT实现简单计算器功能的具体代码,供大家参考,具体内容如下效果图:新建工程,创建类MainWindow,基类是QMain