使用 Java 过滤日期

Filter date with Java

我正在尝试使用 Java 进行筛选,我无法使用 "Select ..." 来查找结果。

问题是我的对象总是加入我的数组。

我的数据是 ->

Registro 1-> Fecha Inicio "2018-09-01" , Fecha Fin -> "2018-09-30"
Registro 2-> Fecha inicio "2018-10-01" , Fecha Fin -> "2018-10-05"
Registro 3-> Fecha inicio "2017-12-31" , Fecha Fin -> "2018-11-30"
Registro 4-> Fecha inicio "2018-12-01" , Fecha Fin -> "2019-10-01"
Registro 5-> Fecha inicio "2018-12-30" , Fecha Fin -> "2018-12-31"
Registro 6-> Fecha inicio "2018-11-30" , Fecha Fin -> "2018-12-01"

用户插入 -> "2018-12-01" hasta el "9999-31-12" 过滤器。

寄存器 4 应该插入到数组中,因为我的用户插入“2018-12-01”并且我的寄存器具有相同的日期开始。

寄存器 5º 应该插入到数组中,因为我的用户插入“2018-12-01" and finish year->9999 那么我的寄存器是如何介于两者之间的,它应该插入到数组中。

Register 6º -> should be insert to array because the date End 因为它的结束日期对应于用户的开始日期和结束日期之间的时间段,即用户想要在“2018- 12-01”至“9999-31-12”,注册结束日期为“2018-12-01”

public List<Article> filterResult(String paramSelect, String dateStart, String dateEnd) {
    List<Article> list = Collections.emptyList();

    try {
        // Sino me ponen fecha de fin, la establezco al máximo.
        if(dateStart != null && dateEnd == null) {
            dateEnd ="9999-31-12";
            list = (List<Article>) this.pgRepository.findAll();
            list =  this.getStart(list, dateStart, dateEnd);
        }
     } catch (Exception e) {
         LOGGER.error(e.getMessage());
     }

     return list;
}

private List<Article> getStart(List<Article> list, String dateStart, String dateEnd) throws ParseException {
    DateFomrat df = new SimpleDateFormat("yyyy-MM-dd"); 
    Date userDate = df.parse(dateStart);
    Date userEnd = df.parse(dateEnd);

    List<Article> filter = new ArrayList<Article>();

    for(Article param : list) {
        Date paramStart = df.parse(param.getStartdatevalidity());
        Date paramEnd = df.parse(param.getEnddatevalidity());

        if(paramStart.after(userDate) || paramEnd.before(userDate) && paramStart.after(userEnd) ||  paramEnd.before(userEnd)) {
            filter.add(param);
        }
    }

    return filter;
}

我有两个问题-> 1º 我的对象总是在我的数组中被看守。 2º 不知道我的逻辑对不对...

谢谢,对不起我的英语。

编辑 2 - 删除了此处提出的问题以进一步说明 Ole V.V 的回答中的概念,我认为它是明确有用的答案。

我根据提供的详细信息提出的建议 -

  1. 使用boolean java.util.Date.after(Date when)

    //Change suggestion in original code if(paramStart.after(userDate) || paramEnd.before(userDate) paramEnd.before(userEnd)) { filter.add(param); }

  2. 使用int java.util.Date.compareTo(java.util.Date anotherDate)

我想你想要这个:

    if (! (paramEnd.before(userDate) || paramStart.after(userEnd))) {
        filter.add(param);
    }

是的,就这么简单。你可能会说,我把条件颠倒过来了。您不想要有效期在 userDate 之前结束的文章,也不想要有效期在 userEnd 之后开始的文章。所有其他文章与用户间隔有重叠,因此包括在内。

也就是说,您使用的 Date class 早已过时并且存在许多设计问题。 SimpleDateFormat 以麻烦出名。此外 Date 尽管它的名字并不代表日期,而是代表时间点。我建议您使用 java.time 中的 LocalDate,现代 Java 日期和时间 API,来表示日期,两者都在您的 Article class 中在您的搜索和过滤代码中。作为一个额外的好处,LocalDate 解析你的 yyyy-MM-dd 日期格式而不需要任何明确的格式化程序,因为这种格式是标准的(它也被称为 ISO 8601)。例如:

public class Article {

    LocalDate startDateValidity;
    LocalDate endDateValidity;

    // Note that this constructor accepts string arguments for convenience
    public Article(String startValidity, String endValidity) {
        this.startDateValidity = LocalDate.parse(startValidity);
        this.endDateValidity = LocalDate.parse(endValidity);
    }

    // getters etc.

}

In filterResult parse dateStart and dateEnd into LocalDate in the same say as in the constructor above.如果缺少 dateEnd,请设置 userEnd tp LocalDate.MAX。当所有日期都是 LocalDate 时,您的 if 条件没有太大变化;方法名前有一个is

    if (! (paramEnd.isBefore(userDate) || paramStart.isAfter(userEnd))) {
        filter.add(param);
    }

编辑:让我们测试一下。

使用您问题中 6 次注册的日期:

    List<Article> list = Arrays.asList(
            new Article("2018-09-01", "2018-09-30"),
            new Article("2018-10-01", "2018-10-05"),
            new Article("2017-12-31", "2018-11-30"),
            new Article("2018-12-01", "2019-10-01"),
            new Article("2018-12-30", "2018-12-31"),
            new Article("2018-11-30", "2018-12-01"));

    LocalDate userDate = LocalDate.parse("2018-12-01");
    LocalDate userEnd = LocalDate.MAX;

    for(Article param : list) {
        LocalDate paramStart = param.getStartDateValidity();
        LocalDate paramEnd = param.getEndDateValidity();

        if (! (paramEnd.isBefore(userDate) || paramStart.isAfter(userEnd))) {
            System.out.println("Included " + param);
        }
    }

此代码段打印了以下输出:

Included Article [startDateValidity=2018-12-01, endDateValidity=2019-10-01]
Included Article [startDateValidity=2018-12-30, endDateValidity=2018-12-31]
Included Article [startDateValidity=2018-11-30, endDateValidity=2018-12-01]

您会认出第 4、5 和 6 号注册的日期,也就是您说过要插入过滤器数组的日期。

你的代码出了什么问题?

I have two problems -> 1º My object always is guarded in my array. 2º I don't know if my logic is correct...

你是对的,你的逻辑不正确。让我们看看你的 if 条件:

if(paramStart.after(userDate)
        || paramEnd.before(userDate) && paramStart.after(userEnd)
        || paramEnd.before(userEnd)) …

&& 的优先级高于 ||,因此它被解释为好像 paramEnd.before(userDate) && paramStart.after(userEnd) 周围有括号。但是,条件的中间部分实际上永远不会为真,但是如果第一部分或最后一部分为真,则将包含一篇文章。只有 5 号注册满足条件的第一部分,因为 after 表示“严格遵循”。但是,所有 6 个注册都满足最后一部分 paramEnd.before(userDate) && paramStart.after(userEnd)。在各部分之间使用 || 这足以将所有 6 篇文章都包含在筛选结果中。

链接

我提议完全不使用从 StringDate 的日期解析。如果用户以相同的格式存储和输入数据 yyyy-mm-dd,那么要比较两个日期,只需比较两个字符串就足够了。

P.S. 实际上,为了比较大量相同格式的日期,我们对字符串使用 RadixSort

这样的话,你的getStart()方法就变得很简单了:

public static List<Article> getStart(List<Article> list, String dateStart, String dateEnd) {
    final Predicate<String> isInRange = date -> date.compareTo(dateStart) >= 0 && date.compareTo(dateEnd) <= 0;
    final Predicate<Article> isIntersect = article -> isInRange.test(article.getStartdatevalidity()) || isInRange.test(article.getEnddatevalidity());
    return list.stream().filter(isIntersect).collect(Collectors.toList());
}