一月分四周的JAVA实现方法
需求:给定任意一个月,如何按照中国周的习惯,把一个月分成四个时间段
(1)以自然周为划分依据
(2)不能跨月
(3)把首尾自然周,天数较少的合并到其最近的自然周里面
(4)最后结果应该是吧一个月分成四个时间段,并给出每段的起止日期
大概需求就如上所述,不废话,上代码:(以封装好在main()函数里,直接copy即可,记得导相应的包和新建实体类)
public class one_month_four_week { public static void main(String[] args) { int year = 2017; int month = 10;//改成你要的年月,运行即可查看结果 Calendar cal = Calendar.getInstance(); cal.set(Calendar.YEAR, year); cal.set(Calendar.MONTH, month - 1);// -1才正确 int days = cal.getActualMaximum(Calendar.DATE);// Calendar.DAY_OF_MONTH也对 System.out.println("首先,计算这个月有" + days + "天"); String[] weekDays = { "星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六" };// 国外一周起始 int[] chinaWeek = { 7, 1, 2, 3, 4, 5, 6 };// 对应成中国一周的第几天 SimpleDateFormat str_date_format = new SimpleDateFormat("yyyy-MM-dd"); List<EachDayAttr> monthDay = new ArrayList<>(); int week_of_month = 1; for (int i = 1; i <= days; i++) { EachDayAttr oneDay = new EachDayAttr(); oneDay.setWhich(i); // 计算这个月的每一天,对应星期几 String dayStr = year + "-" + month + "-" + i; Date daysDate = null; try { daysDate = str_date_format.parse(dayStr); } catch (ParseException e) { e.printStackTrace(); } cal.setTime(daysDate); int w1 = cal.get(Calendar.DAY_OF_WEEK) - 1; // 指示一个星期中的第几天 int which_day_chinaWeek = chinaWeek[w1]; System.out.println("该月的第"+ i +"天,该星期(外国习惯)的第"+ w1 +"天:"+ weekDays[w1] +">>对应中国周的第"+ which_day_chinaWeek +"天!"); //关键:该月每一天对应,中国周里面的第几周 if (which_day_chinaWeek == 1 && monthDay.size() > 0) {// 星期一,中国习惯里新的一周的开始 week_of_month++; oneDay.setWeek(week_of_month); } else { oneDay.setWeek(week_of_month); } monthDay.add(oneDay); } // 分组:按中国习惯分组 Map<Integer, List<EachDayAttr>> weekGroup = new TreeMap<>(); for (EachDayAttr oneDay : monthDay) { if (weekGroup.containsKey(oneDay.getWeek())) { weekGroup.get(oneDay.getWeek()).add(oneDay); } else { List<EachDayAttr> addGroup = new ArrayList<>(); addGroup.add(oneDay); weekGroup.put(oneDay.getWeek(), addGroup); } } // 再分组:合并成四周 if (weekGroup.size() == 4) { // weekGroup不做处理 } else if (weekGroup.size() == 5) { if (weekGroup.get(1).size() < weekGroup.get(5).size()) { weekGroup.get(2).addAll(weekGroup.get(1)); weekGroup.remove(1); } else { weekGroup.get(4).addAll(weekGroup.get(5)); weekGroup.remove(5); } } else if (weekGroup.size() == 6) { weekGroup.get(2).addAll(weekGroup.get(1)); weekGroup.get(5).addAll(weekGroup.get(6)); weekGroup.remove(1); weekGroup.remove(6); } // 吧weekGroup按天数排序,取最小为开始日期,最大为结束日期即可 for (Entry<Integer, List<EachDayAttr>> entry : weekGroup.entrySet()) { Collections.sort(entry.getValue(), new Comparator<EachDayAttr>() { public int compare(EachDayAttr day1, EachDayAttr day2) { return day1.getWhich() - day2.getWhich(); } }); } // 最后从分好组、排好序的weekGroup中取值封装 // weekGroup-->entry中健为整数,而值为list<?>集合 Map<Integer, String[]> rtResult = new HashMap<>(); int signWeek = 1; for (Entry<Integer, List<EachDayAttr>> entry : weekGroup.entrySet()) { String[] addResult = new String[2]; int firstDay = entry.getValue().get(0).getWhich(); int lastDay = entry.getValue().get(entry.getValue().size() - 1).getWhich(); addResult[0] = year + "-" + month + "-" + firstDay; addResult[1] = year + "-" + month + "-" + lastDay; rtResult.put(signWeek, addResult); signWeek++; } // 至此,输出看下结果 for (Entry<Integer, String[]> entry : rtResult.entrySet()) { System.out.println("----该月第"+ entry.getKey() +"周-----"); System.out.println("开始日期:" + entry.getValue()[0]); System.out.println("结束日期:" + entry.getValue()[1]); } } }
总结:上面方法。涉及知识点其实挺多,也是日常项目中遇到的比较多的,比如Calendar这个出来日期的类,还有分组排序、map的遍历等;
加上自定义的实体类,强迫症的小伙伴可以看下:
public class EachDayAttr { private int which; private int week; private int flag;
//get和set方法省略 }
最后。附上SQL大神直接用sql是怎么把一个月按中国周分的(这里一个月最多会有六周,为了更清楚,笔者有改动,但该sql精华不变)
select min(dt) 一周开始, max(dt) 一周结束, count(flag) 该周天数, ROW_NUMBER() over(order by 1) as 周数 from ( select level rn, trunc(to_date(201707,'yyyymm'), 'mm') + level - 1 dt, max(decode (to_char(trunc(to_date(201707,'yyyymm'), 'mm') + level - 1, 'd'), 2, level, 0)) over(order by level) flag from dual connect by level <= last_day(trunc(to_date(201707,'yyyymm'))) - trunc(to_date(201707,'yyyymm'), 'mm') + 1 ) group by flag order by min(rn)
这是在mybatis里面实现的,传入的参数是整形的年月,比如int yyyymm = 201707;