Calendar的set方法陷阱,导致日历应用无法正确显示2月份日历

正确写法,加上clear()

package model;

import java.util.Calendar;

public class MyCalendarLogic {
    //カレンダーインスタンスを生成するメソッド(int...は可変長引数)
    public MyCalendar createMyCalendar(int... args) {
        //マイカレンダーインスタンス生成
        MyCalendar mc = new MyCalendar();
        //現在日時でカレンダーインスタンス生成
        Calendar cal = Calendar.getInstance();

        //2つの引数が来ていたら
        if (args.length == 2) {
            cal.clear();//不加上显示不了2月份日历
            //最初の引数で年を設定
            System.out.println("\033[34;4m" +"args[0] = " + args[0]+ "\033[0m");
            cal.set(Calendar.YEAR, args[0]);
            //次の引数で月を設定
            System.out.println("\033[34;4m" +"args[1] = " + args[1]+ "\033[0m");
            cal.set(Calendar.MONTH, args[1]-1);
            //cal.add(Calendar.MONTH, args[1]-1);也可以
        }

        System.out.println("\033[35;4m"+"Calendar.YEAR = " + cal.get(Calendar.YEAR)+"\033[0m");
        System.out.println("\033[35;4m"+"Calendar.MONTH = " + cal.get(Calendar.MONTH)+"\033[0m");

        //マイカレンダーに年を設定
        mc.setYear(cal.get(Calendar.YEAR));
        //マイカレンダーの元号の設定
        if (mc.getYear() > 2018) {
            mc.setGengou("令和" + (mc.getYear() - 2018));
        } else if (mc.getYear() > 1988) {
            mc.setGengou("平成" + (mc.getYear() - 1988));
        } else if (mc.getYear() > 1925) {
            mc.setGengou("昭和" + (mc.getYear() - 1925));
        } else if (mc.getYear() > 1911) {
            mc.setGengou("大正" + (mc.getYear() - 1911));
        } else {
            mc.setGengou("" + mc.getYear());
        }
        //マイカレンダーに月の設定
        mc.setMonth(cal.get(Calendar.MONTH) + 1);
        System.out.println("**********************************");
        //その月の1日が何曜日かを調べる為に日付を1日にする
        cal.set(Calendar.DATE, 1);
        //カレンダーの最初の空白の数
        int before = cal.get(Calendar.DAY_OF_WEEK) - 1;
        System.out.println("before = " + before);
        //カレンダーの日付の数
        int daysCount = cal.getActualMaximum(Calendar.DAY_OF_MONTH);
        System.out.println("daysCount = " + daysCount);
        //その月の最後の日が何曜日かを調べるために日付を最終日にする
        cal.set(Calendar.DATE, daysCount);
        //最後の日後の空白の数
        int after = 7 - cal.get(Calendar.DAY_OF_WEEK);
        System.out.println("after = " + after);
        //すべての要素数
        int total = before + daysCount + after;
        //その要素数を幅7個の配列に入れていった場合何行になるか
        int rows = total / 7;
        System.out.println("rows = " + rows);
        //その行数で2次元配列を生成
        String[][] data = new String[rows][7];
        //今見ているカレンダーが今月かどうかを調べるために、この瞬間の日付情報をもつもう一つのインスタンス作成しておく
        Calendar now = Calendar.getInstance();
        for (int i = 0; i < rows; i++) {
            for (int j = 0; j < 7; j++) {
                if (i == 0 && j < before || i == rows - 1 && j >= (7 - after)) {
                    //カレンダーの前後に入る空白の部分は空文字
                    data[i][j] = "";
                } else {
                    //カウンター変数と実際の日付の変換
                    int date = i * 7 + j + 1 - before;
                    //配列に日付を入れる
                    data[i][j] = String.valueOf(date);
                    //今作業しているマイカレンダーが今月のカレンダーだったら今日の日付の先頭に*を付与する
                    if (now.get(Calendar.DATE) == date && now.get(Calendar.MONTH) + 1 == mc.getMonth() && now.get(Calendar.YEAR) == mc.getYear()) {
                        data[i][j] = "*" + data[i][j];
                    }
                }
            }
        }
        //作成した2次元配列をマイカレンダーにセットする。
        mc.setData(data);
        return mc;
    }
}

在这里插入图片描述
在这里插入图片描述
Test.java
date取29,30,31时会使Calendar.MONTH 的值不一致
1~28日Calendar.MONTH 的值一样

package controller;

import java.util.Calendar;

public class Test {
    public static void main(String[] args) {
        method1();
        System.out.println("");
        method2();
        System.out.println("");
        method3();
    }

    //真正的月份 = cal.get(Calendar.MONTH) + 1
    //测试日期位2021/12/30,cal.get(Calendar.MONTH)=11

    private static void method1() {
        Calendar cal = Calendar.getInstance();
        System.out.println(cal.get(Calendar.MONTH));
        cal.set(Calendar.YEAR, Calendar.DECEMBER, 30);
        cal.clear();//关键的一部,set()不是reset重置时间,而是在之前时间基础上add,2月份只有28或29天,不会30天,set直接跳过它了
        System.out.println("clear execute");
        System.out.println("Calendar.MONTH = " + cal.get(Calendar.MONTH));
        cal.set(Calendar.MONTH, 1);
        System.out.println("\033[35;4m" + "Calendar.MONTH = " + cal.get(Calendar.MONTH) + "\033[0m");
    }

    private static void method2() {
        Calendar cal = Calendar.getInstance();
        System.out.println(cal.get(Calendar.MONTH));
        cal.set(Calendar.YEAR, Calendar.DECEMBER, 30);
        System.out.println("Calendar.MONTH = " + cal.get(Calendar.MONTH));
        cal.set(Calendar.MONTH, 1);
        System.out.println("\033[35;4m" + "Calendar.MONTH = " + cal.get(Calendar.MONTH) + "\033[0m");
    }

    private static void method3() {
        Calendar cal = Calendar.getInstance();
        System.out.println(cal.get(Calendar.MONTH));
        cal.set(2021, 11, 30);//month要少减1,月份0~11
        System.out.println("Calendar.MONTH = " + cal.get(Calendar.MONTH));
        cal.set(Calendar.MONTH, 1);
        System.out.println("\033[35;4m" + "Calendar.MONTH = " + cal.get(Calendar.MONTH) + "\033[0m");
    }
}

在这里插入图片描述

Calendar c = Calendar.getInstance();
 
c.set(2014, Calendar.MARCH, 31);
c.add(Calendar.MONTH, 13);
System.out.println(c.getTime());// 2015-04-30

 
c.set(2014, Calendar.MARCH, 31);
c.set(Calendar.MONTH, c.get(Calendar.MONTH) + 13);
System.out.println(c.getTime());// 2015-05-01

 
c.set(2014, Calendar.MARCH, 31);
c.roll(Calendar.MONTH, 13);
System.out.println(c.getTime());//2014-04-30

ADD方法
以调整的单位为基点(本例中为月),较大的单位(年)会发生借位、进位。 较小的单位会往小调整。
本例中,2014-03-31,加上13个月,年份会进位为2015年。 4月31日是不存在的,所以往小调整为4月30日。
比较典型的运用场景是,日历的按月切换。
当前日期为2014-03-31,点击【下一月】按钮时,日期会变成2014-04-30.

SET方法
所有的单位都会往大调整。
本例中,2014-03-31,加上13个月,年份会进位为2015年。 4月31日是不存在的,所以往大调整为5月1日

ROLL方法
以调整的单位为基点(本例中为月),较大的单位(年)不会发生改变。 较小的单位会往小调整。
本例中,2014-03-31,加上13个月,年份依然为2014年。 4月31日是不存在的,所以往小调整为4月30日。
日会根据年、月来判断出日的取值范围,然后在1~31之间无限循环滚动,但并不会影响到年、月的值。

总结三点:
1、add() 有两条规则:
a)当被修改的字段超出它的取值范围时,那么比它大的字段会自动修正。
b)如果比它小的字段是不可变的/不在取值范围内(由 Calendar 的实现类决定),那么该小字段会修正到变化最小的值。
2、Roll() 的规则只有第二条
当被修改的字段超出它的取值范围时,那么比它大的字段不会被修正。比它小的字段会修正到变化最小的值。
3、Set()
比被修改的字段大的字段会根据字段是增大还是减小自动改变大小,比被修改字段小的字段如果是不可变的/不在取值范围内,会自动增大到变化最小的值。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值