จะวนซ้ำช่วงของวันที่ใน Java ได้อย่างไร?


145

ในสคริปต์ของฉันฉันต้องดำเนินการชุดของช่วงวันที่ให้กำหนดวันที่เริ่มต้นและสิ้นสุด
กรุณาให้คำแนะนำฉันเพื่อให้บรรลุสิ่งนี้โดยใช้ Java

for ( currentDate = starDate; currentDate < endDate; currentDate++) {

}

ฉันรู้ว่าโค้ดข้างต้นนั้นเป็นไปไม่ได้ แต่ฉันทำเพื่อแสดงให้คุณเห็นสิ่งที่ฉันต้องการบรรลุ


วิธีทำความสะอาด Java 8 และ 9: stackoverflow.com/a/51942109/1216775
akhil_mittal

คำตอบ:


198

คุณสามารถทำสิ่งนี้โดยใช้time-API ของ Java 8สำหรับปัญหานี้โดยเฉพาะjava.time.LocalDate(หรือคลาสJoda Time ที่เทียบเท่าสำหรับ Java 7 และเก่ากว่า)

for (LocalDate date = startDate; date.isBefore(endDate); date = date.plusDays(1))
{
    ...
}

ฉันอยากจะแนะนำอย่างละเอียดให้ใช้java.time(หรือ Joda Time) ในตัวDate/ Calendarคลาส


2
หากต้องการขยายจุดเกี่ยวกับ Joda Time: การพยายามใช้สิ่งนี้อย่างถูกต้องด้วยตัวเองนั้นยากกว่าที่คิดเพราะมุมคดีรอบ ๆ เปลี่ยนไปและจากฤดูร้อน
Raedwald

+1 สำหรับ Joda ฉันหวังว่าสักวันมันจะไปถึงที่นั่นใน API มาตรฐาน
gyorgyabraham

4
@gyabraham: JSR-310 ดูดีมากสำหรับ Java 8
Jon Skeet

4
สามารถยืนยันรหัสเดียวกันนี้ได้โดยใช้ java.time.LocalDate ของ Java 8 แทน Joda
Molten Ice

3
โปรเจ็กต์ Joda-Time อยู่ในโหมดบำรุงรักษาและแนะนำให้ทำการย้ายไปยังคลาส java.time ดังที่ได้กล่าวไว้ในความคิดเห็นรหัสของผู้ตอบคำถามนี้ทำงานได้เหมือนใน java.time เพียงแค่เปลี่ยนimportข้อความของคุณ
Basil Bourque

148

JodaTime นั้นดีอย่างไรก็ตามเพื่อความสมบูรณ์และ / หรือหากคุณต้องการสิ่งอำนวยความสะดวกที่มีให้กับ API นี่คือแนวทางมาตรฐานของ API

เมื่อเริ่มต้นด้วยjava.util.Dateอินสแตนซ์ด้านล่าง:

SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd");
Date startDate = formatter.parse("2010-12-20");
Date endDate = formatter.parse("2010-12-26");

นี่เป็นjava.util.Calendarวิธีการดั้งเดิมในกรณีที่คุณไม่ได้ใช้ Java8:

Calendar start = Calendar.getInstance();
start.setTime(startDate);
Calendar end = Calendar.getInstance();
end.setTime(endDate);

for (Date date = start.getTime(); start.before(end); start.add(Calendar.DATE, 1), date = start.getTime()) {
    // Do your job here with `date`.
    System.out.println(date);
}

และนี่คือjava.time.LocalDateแนวทางของ Java8 โดยพื้นฐานแล้วเป็นวิธี JodaTime:

LocalDate start = startDate.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();
LocalDate end = endDate.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();

for (LocalDate date = start; date.isBefore(end); date = date.plusDays(1)) {
    // Do your job here with `date`.
    System.out.println(date);
}

หากคุณต้องการรวมวันที่สิ้นสุดให้ใช้!start.after(end)และ!date.isAfter(end)ตามลำดับ


75

สไตล์Java 8โดยใช้คลาสjava.time :

// Monday, February 29 is a leap day in 2016 (otherwise, February only has 28 days)
LocalDate start = LocalDate.parse("2016-02-28"),
          end   = LocalDate.parse("2016-03-02");

// 4 days between (end is inclusive in this example)
Stream.iterate(start, date -> date.plusDays(1))
        .limit(ChronoUnit.DAYS.between(start, end) + 1)
        .forEach(System.out::println);

เอาท์พุท:

2016-02-28
2016-02-29
2016-03-01
2016-03-02

ทางเลือก:

LocalDate next = start.minusDays(1);
while ((next = next.plusDays(1)).isBefore(end.plusDays(1))) {
    System.out.println(next);
}

Java 9เพิ่มวิธีdateUntil () :

start.datesUntil(end.plusDays(1)).forEach(System.out::println);

1
คุณสามารถใส่ค่าทวีคูณได้ไหม? ตัวอย่างเช่น: เฉพาะวันจันทร์หรือวันพฤหัสบดีหรือทั้งสองอย่าง
delive

26

นี่เป็นคำตอบเดียวที่ BalusC ให้ แต่สามารถอ่านได้มากกว่าโดยใช้ลูป while แทน a for loop:

Calendar start = Calendar.getInstance();
start.setTime(startDate);

Calendar end = Calendar.getInstance();
end.setTime(endDate);

while( !start.after(end)){
    Date targetDay = start.getTime();
    // Do Work Here

    start.add(Calendar.DATE, 1);
}

3
สิ่งนี้จะไม่ทำงานหากตรรกะเกี่ยวข้องกับคำสั่ง "ดำเนินการต่อ" ในขณะที่รุ่นสำหรับลูปของ BalusC ทำงานร่วมกับคำสั่ง Continue
Sanjiv Jivan

6

Apache Commons

    for (Date dateIter = fromDate; !dateIter.after(toDate); dateIter = DateUtils.addDays(dateIter, 1)) {
        // ...
    }

+1, IMHO นี่คือรหัสที่สะอาดที่สุดเมื่อคุณทำงานกับรหัสเก่า เพียงแค่นำเข้าแบบคงที่พิเศษaddDays(..)และจะสั้นลง
Priidu Neemre

4
private static void iterateBetweenDates(Date startDate, Date endDate) {
    Calendar startCalender = Calendar.getInstance();
    startCalender.setTime(startDate);
    Calendar endCalendar = Calendar.getInstance();
    endCalendar.setTime(endDate);

    for(; startCalender.compareTo(endCalendar)<=0;
          startCalender.add(Calendar.DATE, 1)) {
        // write your main logic here
    }

}

3
public static final void generateRange(final Date dateFrom, final Date dateTo)
{
    final Calendar current = Calendar.getInstance();
    current.setTime(dateFrom);

    while (!current.getTime().after(dateTo))
    {
        // TODO

        current.add(Calendar.DATE, 1);
    }
}

3

เราสามารถโยกย้ายตรรกะไปยังวิธีการต่าง ๆ ศัตรูJava 7, Java 8 และ Java 9 :

public static List<Date> getDatesRangeJava7(Date startDate, Date endDate) {
    List<Date> datesInRange = new ArrayList<>();
    Calendar startCalendar = new GregorianCalendar();
    startCalendar.setTime(startDate);
    Calendar endCalendar = new GregorianCalendar();
    endCalendar.setTime(endDate);
    while (startCalendar.before(endCalendar)) {
        Date result = startCalendar.getTime();
        datesInRange.add(result);
        startCalendar.add(Calendar.DATE, 1);
    }
    return datesInRange;
}

public static List<LocalDate> getDatesRangeJava8(LocalDate startDate, LocalDate endDate) {
    int numOfDays = (int) ChronoUnit.DAYS.between(startDate, endDate);
    return IntStream.range(0, numOfDays)
            .mapToObj(startDate::plusDays)
            .collect(Collectors.toList());
}

public static List<LocalDate> getDatesRangeJava9(LocalDate startDate, LocalDate endDate) {
    return startDate.datesUntil(endDate).collect(Collectors.toList());
}

จากนั้นเราสามารถเรียกใช้วิธีการเหล่านี้เป็น:

SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd");
Date startDate = formatter.parse("2010-12-20");
Date endDate = formatter.parse("2010-12-26");
List<Date> dateRangeList = getDatesRangeJava7(startDate, endDate);
System.out.println(dateRangeList);

LocalDate startLocalDate = startDate.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();
LocalDate endLocalDate = endDate.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();
List<LocalDate> dateRangeList8 = getDatesRangeJava8(startLocalDate, endLocalDate);
System.out.println(dateRangeList8);
List<LocalDate> dateRangeList9 = getDatesRangeJava8(startLocalDate, endLocalDate);
System.out.println(dateRangeList9);

ผลลัพธ์จะเป็น:

[จันทร์ 20 ธันวาคม 00:00:00 IST 2010, วันที่ 21 ธันวาคม 00:00:00 IST 2010, พุธ 22 ธันวาคม 00:00:00 IST 2010, พฤหัสบดี 23 ธันวาคม 00:00:00 IST 2010, ศุกร์ 24 ธันวาคม 00: 00:00 IST 2010, เสาร์ 25 ธันวาคม 00:00:00 IST 2010]

[2010-12-2010, 2010-12-21, 2010-12-22, 2010-12-23, 2010-12-24, 2010-12-25]

[2010-12-2010, 2010-12-21, 2010-12-22, 2010-12-23, 2010-12-24, 2010-12-25]


1
เรียนแย่มากDateและCalendarถูกแทนที่ด้วยคลาสjava.time เมื่อหลายปีก่อน โดยเฉพาะแทนที่ด้วยและInstant ZonedDateDate
Basil Bourque

1
ฉันชอบ Java 8 และ 9 วิธี สำหรับ Java 6 และ 7 ฉันขอแนะนำให้ใช้ไลบรารี ThreeTen Backportและเช่นเดียวกับใน Java 8 คุณแสดงให้เห็นอย่างชัดเจนว่าวิธีนี้ชัดเจนและเป็นมิตรกับโปรแกรมอย่างไร
Ole VV

2

นี่คือโค้ด Java 8 ฉันคิดว่ารหัสนี้จะแก้ปัญหาของคุณ Happy Coding

    LocalDate start = LocalDate.now();
    LocalDate end = LocalDate.of(2016, 9, 1);//JAVA 9 release date
    Long duration = start.until(end, ChronoUnit.DAYS);
    System.out.println(duration);
     // Do Any stuff Here there after
    IntStream.iterate(0, i -> i + 1)
             .limit(duration)
             .forEach((i) -> {});
     //old way of iteration
    for (int i = 0; i < duration; i++)
     System.out.print("" + i);// Do Any stuff Here

นี่คือวิธีที่ดีที่สุดและง่ายที่คุณสามารถติดตาม
jatin Goyal

1

ทำไมไม่ใช้ยุคและวนซ้ำอย่างง่ายดาย

long startDateEpoch = new java.text.SimpleDateFormat("dd/MM/yyyy").parse(startDate).getTime() / 1000;

    long endDateEpoch = new java.text.SimpleDateFormat("dd/MM/yyyy").parse(endDate).getTime() / 1000;


    long i;
    for(i=startDateEpoch ; i<=endDateEpoch; i+=86400){

        System.out.println(i);

    }

1

คุณสามารถเขียนคลาสที่ชอบ (ใช้อินเตอร์เฟสตัววนซ้ำ) และวนซ้ำไปเรื่อย ๆ

public class DateIterator implements Iterator<Date>, Iterable<Date>
{

 private Calendar end = Calendar.getInstance();
 private Calendar current = Calendar.getInstance();

 public DateIterator(Date start, Date end)
 {
     this.end.setTime(end);
     this.end.add(Calendar.DATE, -1);
     this.current.setTime(start);
     this.current.add(Calendar.DATE, -1);
 }

 @Override
 public boolean hasNext()
 {
     return !current.after(end);
 }

 @Override
 public Date next()
 {
     current.add(Calendar.DATE, 1);
     return current.getTime();
 }

 @Override
 public void remove()
 {
     throw new UnsupportedOperationException(
        "Cannot remove");
 }

 @Override
 public Iterator<Date> iterator()
 {
     return this;
 }
}

และใช้มันเช่น:

Iterator<Date> dateIterator = new DateIterator(startDate, endDate);
while(dateIterator.hasNext()){
      Date selectedDate = dateIterator .next();

}

1

คุณสามารถลองสิ่งนี้:

OffsetDateTime currentDateTime = OffsetDateTime.now();
for (OffsetDateTime date = currentDateTime; date.isAfter(currentDateTime.minusYears(YEARS)); date = date.minusWeeks(1))
{
    ...
}

0

วิธีนี้จะช่วยให้คุณเริ่มต้น 30 วันย้อนหลังและวนซ้ำจนถึงวันที่วันนี้ คุณสามารถเปลี่ยนช่วงวันที่และทิศทางได้อย่างง่ายดาย

private void iterateThroughDates() throws Exception {
    Calendar start = Calendar.getInstance();
    start.add(Calendar.DATE, -30);
    Calendar end = Calendar.getInstance();
    for (Calendar date = start; date.before(end); date.add(Calendar.DATE, 1))
        {
        System.out.println(date.getTime());
        }
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.