Получите Последнюю пятницу месяца на Java
Вопрос
Я работаю над проектом, где требуется, чтобы дата была рассчитана как последняя пятница данного месяца.Я думаю, у меня есть решение, которое использует только стандартную Java, но мне было интересно, знает ли кто-нибудь что-нибудь более краткое или эффективное.Ниже приведено то, что я тестировал в этом году:
for (int month = 0; month < 13; month++) {
GregorianCalendar d = new GregorianCalendar();
d.set(d.MONTH, month);
System.out.println("Last Week of Month in " + d.getDisplayName(d.MONTH, Calendar.LONG, Locale.ENGLISH) + ": " + d.getLeastMaximum(d.WEEK_OF_MONTH));
d.set(d.DAY_OF_WEEK, d.FRIDAY);
d.set(d.WEEK_OF_MONTH, d.getActualMaximum(d.WEEK_OF_MONTH));
while (d.get(d.MONTH) > month || d.get(d.MONTH) < month) {
d.add(d.WEEK_OF_MONTH, -1);
}
Date dt = d.getTime();
System.out.println("Last Friday of Last Week in " + d.getDisplayName(d.MONTH, Calendar.LONG, Locale.ENGLISH) + ": " + dt.toString());
}
Решение
Основанный на отмеченные 23-е предложение:
public Date getLastFriday( int month, int year ) {
Calendar cal = Calendar.getInstance();
cal.set( year, month + 1, 1 );
cal.add( Calendar.DAY_OF_MONTH, -( cal.get( Calendar.DAY_OF_WEEK ) % 7 + 1 ) );
return cal.getTime();
}
Другие советы
Позвольте Calendar.class сотворить это волшебство за вас ;)
pCal.set(GregorianCalendar.DAY_OF_WEEK,Calendar.FRIDAY);
pCal.set(GregorianCalendar.DAY_OF_WEEK_IN_MONTH, -1);
Вам никогда не нужно зацикливаться, чтобы выяснить это.Чтобы определить дату "последней пятницы" для этого месяца, начните с первого числа следующего месяца.Вычтите соответствующее количество дней в зависимости от того, на какой (числовой) день недели приходится первый день месяца.Вот ваша "последняя пятница". Я почти уверен, что это можно свести к длинноватому однострочнику, но я не Java-разработчик.Так что я оставлю это кому-нибудь другому.
java.время
Используя java.время библиотеку, встроенную в Java 8 и более поздние версии, вы можете использовать TemporalAdjusters.lastInMonth
:
val now = LocalDate.now()
val lastInMonth = now.with(TemporalAdjusters.lastInMonth(DayOfWeek.FRIDAY))
Вы можете выбрать любой день из предложенных DayOfWeek
перечисление.
Если вам нужно добавить информацию о времени, вы можете использовать любую доступную информацию LocalDate
Для LocalDateTime
преобразование, подобное
lastFriday.atStartOfDay() // e.g. 2015-11-27T00:00
Я бы использовал такую библиотеку, как Джодатим.У него очень полезный API, и он использует обычные номера месяцев.И самое главное, это потокобезопасность.
Я думаю, что у вас может быть решение с помощью (но, возможно, не самое короткое, но, безусловно, более читаемое):
DateTime now = new DateTime();
DateTime dt = now.dayOfMonth().withMaximumValue().withDayOfWeek(DateTimeConstants.FRIDAY);
if (dt.getMonthOfYear() != now.getMonthOfYear()) {
dt = dt.minusDays(7);
}
System.out.println(dt);
Вам нужно знать две вещи - количество дней в месяце и день недели, на который приходится первое число месяца.
Если первый день месяца приходится на
- Воскресенье, затем последняя пятница - всегда 27-е число.
- Понедельник, затем последняя пятница - всегда 26-е число.
- Вторник, затем последняя пятница - всегда 25-е число.
- Среда, тогда последняя пятница - 24-е, если в месяце нет 31 дня, тогда это 31-е
- Четверг, тогда последняя пятница - 23-е, если в месяце нет 30 дней или больше, тогда это 30-е число.
- Пятница, тогда последняя пятница - 22-е, если в месяце нет 29 дней или больше, тогда это 29-е.
- Суббота, затем последняя пятница - всегда 28-е число.
Есть только три особых случая.Один оператор switch и три оператора if (или троичные операторы, если вы хотите, чтобы в каждом случае была одна строка ...)
Изложите это на бумаге.Не нужны никакие специальные библиотеки, функции, юлианские преобразования и т.д. (ну, за исключением того, чтобы узнать день недели, на который приходится 1-е число, и, возможно, количество дней в этом месяце...)
-Адам
код для Алгоритм Адама Дэвиса
public static int getLastFriday(int month, int year)
{
Calendar cal = Calendar.getInstance();
cal.set(year, month, 1, 0, 0, 0); // set to first day of the month
cal.set(Calendar.MILLISECOND, 0);
int firstDay = cal.get(Calendar.DAY_OF_WEEK);
int daysOfMonth = cal.getMaximum(Calendar.DAY_OF_MONTH);
switch (firstDay)
{
case Calendar.SUNDAY :
return 27;
case Calendar.MONDAY :
return 26;
case Calendar.TUESDAY :
return 25;
case Calendar.WEDNESDAY :
if (daysOfMonth == 31) return 31;
return 24;
case Calendar.THURSDAY :
if (daysOfMonth >= 30) return 30;
return 23;
case Calendar.FRIDAY :
if (daysOfMonth >= 29) return 29;
return 22;
case Calendar.SATURDAY :
return 28;
}
throw new RuntimeException("what day of the month?");
}}
Хотя я согласен со scubabbl, вот версия без внутреннего while.
int year = 2008;
for (int m = Calendar.JANUARY; m <= Calendar.DECEMBER; m++) {
Calendar cal = new GregorianCalendar(year, m, 1);
cal.set(Calendar.DAY_OF_MONTH, cal.getActualMaximum(Calendar.DAY_OF_MONTH));
int diff = Calendar.FRIDAY - cal.get(Calendar.DAY_OF_WEEK);
if (diff > 0) {
diff -= 7;
}
cal.add(Calendar.DAY_OF_MONTH, diff);
System.out.println(cal.getTime());
}
Немного легче читается, подход "грубой силы":
public int getLastFriday(int month, int year) {
Calendar cal = Calendar.getInstance();
cal.set(year, month, 1, 0, 0, 0); // set to first day of the month
cal.set(Calendar.MILLISECOND, 0);
int friday = -1;
while (cal.get(Calendar.MONTH) == month) {
if (cal.get(Calendar.DAY_OF_WEEK) == Calendar.FRIDAY) { // is it a friday?
friday = cal.get(Calendar.DAY_OF_MONTH);
cal.add(Calendar.DAY_OF_MONTH, 7); // skip 7 days
} else {
cal.add(Calendar.DAY_OF_MONTH, 1); // skip 1 day
}
}
return friday;
}
вот как получить последнюю пятницу или любой другой день недели месяца:
Calendar thisMonth = Calendar.getInstance();
dayOfWeek = Calendar.FRIDAY; // or whatever
thisMonth.set(Calendar.WEEK_OF_MONTH, thisMonth.getActualMaximum(Calendar.WEEK_OF_MONTH);;
thisMonth.set(Calendar.DAY_OF_WEEK, dayOfWeek);
int lastDay = thisMonth.get(Calendar.DAY_OF_MONTH); // this should be it.
Приведенная ниже программа рассчитана на последнюю пятницу каждого месяца.его можно использовать для получения последнего числа любого дня недели в любом месяце.Переменная offset=0
означает текущий месяц (системную дату), смещение = 1 означает следующий месяц и так далее.Тот Самый getLastFridayofMonth(int offset)
метод вернет дату последней пятницы.
import java.text.SimpleDateFormat;
import java.util.Calendar;
public class LastFriday {
public static Calendar getLastFriday(Calendar cal,int offset){
int dayofweek;//1-Sunday,2-Monday so on....
cal.set(Calendar.MONTH,cal.get(Calendar.MONTH)+offset);
cal.set(Calendar.DAY_OF_MONTH, cal.getActualMaximum(Calendar.DAY_OF_MONTH)); //set calendar to last day of month
dayofweek=cal.get(Calendar.DAY_OF_WEEK); //get the day of the week for last day of month set above,1-sunday,2-monday etc
if(dayofweek<Calendar.FRIDAY) //Calendar.FRIDAY will return integer value =5
cal.set(Calendar.DAY_OF_MONTH, cal.get(Calendar.DAY_OF_MONTH)-7+Calendar.FRIDAY-dayofweek);
else
cal.set(Calendar.DAY_OF_MONTH, cal.get(Calendar.DAY_OF_MONTH)+Calendar.FRIDAY-dayofweek);
return cal;
}
public static String getLastFridayofMonth(int offset) { //offset=0 mean current month
final String DATE_FORMAT_NOW = "dd-MMM-yyyy";
Calendar cal = Calendar.getInstance();
SimpleDateFormat sdf = new SimpleDateFormat(DATE_FORMAT_NOW);
cal=getLastFriday(cal,offset);
return sdf.format(cal.getTime());
}
public static void main(String[] args) {
System.out.println(getLastFridayofMonth(0)); //0 = current month
System.out.println(getLastFridayofMonth(1));//1=next month
System.out.println(getLastFridayofMonth(2));//2=month after next month
}
}
В Java 8 мы можем сделать это просто как:
LocalDate lastFridayOfMonth = LocalDate
.now()
.with(lastDayOfMonth())
.with(previous(DayOfWeek.FRIDAY));
Это выглядит как вполне приемлемое решение.Если это сработает, используйте это.Это минимальный код, и нет причин оптимизировать его без крайней необходимости.
public static int lastSundayDate()
{
Calendar cal = getCalendarInstance();
cal.setTime(new Date(getUTCTimeMillis()));
cal.set( Calendar.DAY_OF_MONTH , 25 );
return (25 + 8 - (cal.get(Calendar.DAY_OF_WEEK) != Calendar.SUNDAY ? cal.get(Calendar.DAY_OF_WEEK) : 8));
}
Надеюсь, это поможет..
public static void getSundaysInThisMonth(int monthNumber, int yearNumber){
//int year =2009;
//int dayOfWeek = Calendar.SUNDAY;
// instantiate Calender and set to first Sunday of 2009
Calendar cal = new GregorianCalendar();
cal.set(Calendar.MONTH, monthNumber-1);
cal.set(Calendar.YEAR, yearNumber);
cal.set(Calendar.DATE, 1);
int dayOfWeek = cal.get(Calendar.DAY_OF_WEEK);
int dateOfWeek = cal.get(Calendar.DATE);
while (dayOfWeek != Calendar.SUNDAY) {
cal.set(Calendar.DATE, ++dateOfWeek);
dayOfWeek = cal.get(Calendar.DAY_OF_WEEK);
}
cal.set(Calendar.DATE, dateOfWeek);
int i = 1;
while (cal.get(Calendar.YEAR) == yearNumber && cal.get(Calendar.MONTH)==monthNumber-1)
{
System.out.println("Sunday " + " " + i + ": " + cal.get(Calendar.DAY_OF_MONTH));
cal.add(Calendar.DAY_OF_MONTH, 7);
i++;
}
}
public static void main(String args[]){
getSundaysInThisMonth(1,2009);
}
public static Calendar getNthDow(int month, int year, int dayOfWeek, int n) {
Calendar cal = Calendar.getInstance();
cal.set(year, month, 1);
cal.set(Calendar.DAY_OF_WEEK, dayOfWeek);
cal.set(Calendar.DAY_OF_WEEK_IN_MONTH, n);
return (cal.get(Calendar.MONTH) == month) && (cal.get(Calendar.YEAR) == year) ? cal : null;
}