Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(2) 自己封装的Calendar接口
本章主要是收藏一些常用的类和接口,包括:万年历(农历、阳历节日、阴历节日)、自定义的Calendar接口。
万年历
源码如下(ChineseCalendar.java):
package com.via.mce.monthcalendar.utils; import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.HashMap; /**
* 农历日历。<br>
* 将农历从1901年到2100年之间各年、月的大小以及历年节气保存,然后基于这些数据进行计算。<br>
* <br>
* 新增了几个用于农历的常量属性字段,可以使用get()方法获取日历对应的值;<br>
* 农历年、月、日还可以使用set()/add()/roll()方法设置,其他农历属性自动计算;<br>
* 另外,还提供了getChinese(int field)方法用于获得农历的中文文字(仅适用于农历属性和星期)。<br>
* <ul>
* <li>CHINESE_YEAR - 农历年</li>
* <li>CHINESE_MONTH - 农历月</li>
* <li>CHINESE_DATE - 农历日</li>
* <li>CHINESE_SECTIONAL_TERM - 当月的节气</li>
* <li>CHINESE_PRINCIPLE_TERM - 当月的中气</li>
* <li>CHINESE_HEAVENLY_STEM - 农历年的天干</li>
* <li>CHINESE_EARTHLY_BRANCH - 农历年的地支</li>
* <li>CHINESE_ZODIAC - 农历年的属相</li>
* <li>CHINESE_TERM_OR_DATE - 如果当天存在一个节气则指示节气,否则如果当天是初一则指示农历月,否则指示农历日</li>
* </ul>
* 注意:<br>
* 由于Calendar类的设定,公历月份从0起始。所有方法都遵循了这一约定。<br>
* 但所有的农历属性从1起始。即使是在Calendar提供的方法中,农历月也是从1起始的,并以负数表示闰月。<br>
* clear()方法在某些情况下会导致农历和公历日期不对应或是不能达到预期的重置效果,应尽量避免使用。<br>
* 使用getSimpleDateString()获得公历日期字符串时,公历月已经修正;<br>
* 使用getSimpleChineseDateString()获得农历日期字符串时,农历闰月以*表示。<br>
*
* @version 0.12 2011-9-5 <br>
* <blockquote>修复一个当使用农历正月日期初始化日历时陷入死循环的问题。</blockquote>
* @version 0.11 2009-12-27 <br>
* <blockquote>修复了获取中文农历时未计算农历日期的问题;<br>
* 加入一个字段CHINESE_TERM_OR_DATE用于模仿台历的显示方式:如果当天有节气则指示节气,如果是初一指示农历月,
* 否则指示农历日。</blockquote>
* @version 0.10 2009-12-22
*/
public final class ChineseCalendar extends GregorianCalendar {
private static final long serialVersionUID = 8L; /** 农历年 */
public static final int CHINESE_YEAR = 801;
/** 农历月 */
public static final int CHINESE_MONTH = 802;
/** 农历日 */
public static final int CHINESE_DATE = 803;
/** 当月的节气对应的公历日(前一个节气) */
public static final int CHINESE_SECTIONAL_TERM = 804;
/** 当月的中气对应的公历日(后一个节气) */
public static final int CHINESE_PRINCIPLE_TERM = 805;
/** 天干 */
public static final int CHINESE_HEAVENLY_STEM = 806;
/** 地支 */
public static final int CHINESE_EARTHLY_BRANCH = 807;
/** 农历年的属相(生肖) */
public static final int CHINESE_ZODIAC = 808;
/** 节气或者农历日 */
public static final int CHINESE_TERM_OR_DATE = 888; // add by skywang
/** 农历节日 */
public static final int LUNAR_FESTIVAL = 809;
/** 阳历节日 */
public static final int SOLAR_FESTIVAL = 810;
/** 节气 */
public static final int CHINESE_TERM = 811;
/** 月或者农历日 */
public static final int CHINESE_MONTH_OR_DATE = 812;
/** 节日 或 节气 或 农历日 */
public static final int FESTIVAL_OR_TERM_OR_DATE = 813; private int chineseYear;
private int chineseMonth; // 1起始,负数表示闰月
private int chineseDate;
private int sectionalTerm; // 当月节气的公历日
private int principleTerm; // 当月中气的公历日 private boolean areChineseFieldsComputed; // 农历日期是否已经经过计算确认
private boolean areSolarTermsComputed; // 节气是否已经经过计算确认
private boolean lastSetChinese; // 最后设置的是不是农历属性 /** 使用当前时间构造一个实例。 */
public ChineseCalendar() {
super();
} /** 使用指定时间构造一个实例。 */
public ChineseCalendar(Date d) {
super.setTime(d);
} /** 使用指定时间构造一个实例。 */
public ChineseCalendar(Calendar c) {
this(c.getTime());
} /** 使用指定公历日期构造一个实例。 */
public ChineseCalendar(int y, int m, int d) {
super(y, m, d);
} /**
* 使用指定日期构造一个实例。
*
* @param isChinese
* 是否为农历日期
* @param y
* @param m
* @param d
*/
public ChineseCalendar(boolean isChinese, int y, int m, int d) {
if (isChinese) {
set(CHINESE_YEAR, y);
set(CHINESE_MONTH, m);
set(CHINESE_DATE, d);
} else {
set(y, m, d);
}
} public void set(int field, int value) {
computeIfNeed(field); if (isChineseField(field)) {
// 农历属性
switch (field) {
case CHINESE_YEAR:
chineseYear = value;
break;
case CHINESE_MONTH:
chineseMonth = value;
break;
case CHINESE_DATE:
chineseDate = value;
break;
default:
throw new IllegalArgumentException("不支持的field设置:" + field);
}
lastSetChinese = true;
} else {
// 非农历属性
super.set(field, value);
lastSetChinese = false;
}
areFieldsSet = false;
areChineseFieldsComputed = false;
areSolarTermsComputed = false;
} public int get(int field) {
computeIfNeed(field); if (!isChineseField(field)) {
return super.get(field);
} switch (field) {
case CHINESE_YEAR:
return chineseYear;
case CHINESE_MONTH:
return chineseMonth;
case CHINESE_DATE:
return chineseDate;
case CHINESE_SECTIONAL_TERM:
return sectionalTerm;
case CHINESE_PRINCIPLE_TERM:
return principleTerm;
case CHINESE_HEAVENLY_STEM:
return (chineseYear - 4) % 10 + 1;
case CHINESE_EARTHLY_BRANCH:
case CHINESE_ZODIAC:
return (chineseYear - 4) % 12 + 1;
case CHINESE_MONTH_OR_DATE:
if (get(CHINESE_DATE) == 1) {
return CHINESE_MONTH;
} else {
return CHINESE_DATE;
}
case CHINESE_TERM_OR_DATE:
int option;
if (get(Calendar.DATE) == get(CHINESE_SECTIONAL_TERM)) {
option = CHINESE_SECTIONAL_TERM;
} else if (get(Calendar.DATE) == get(CHINESE_PRINCIPLE_TERM)) {
option = CHINESE_PRINCIPLE_TERM;
} else if (get(CHINESE_DATE) == 1) {
option = CHINESE_MONTH;
} else {
option = CHINESE_DATE;
}
return option;
default:
throw new IllegalArgumentException("不支持的field获取:" + field);
}
} public void add(int field, int amount) {
computeIfNeed(field); if (!isChineseField(field)) {
super.add(field, amount);
lastSetChinese = false;
areChineseFieldsComputed = false;
areSolarTermsComputed = false;
return;
} switch (field) {
case CHINESE_YEAR:
chineseYear += amount;
break;
case CHINESE_MONTH:
for (int i = 0; i < amount; i++) {
chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
if (chineseMonth == 1) {
chineseYear++;
}
}
break;
case CHINESE_DATE:
int maxDate = daysInChineseMonth(chineseYear, chineseMonth);
for (int i = 0; i < amount; i++) {
chineseDate++;
if (chineseDate > maxDate) {
chineseDate = 1;
chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
if (chineseMonth == 1) {
chineseYear++;
}
maxDate = daysInChineseMonth(chineseYear, chineseMonth);
}
}
default:
throw new IllegalArgumentException("不支持的field:" + field);
} lastSetChinese = true;
areFieldsSet = false;
areChineseFieldsComputed = false;
areSolarTermsComputed = false;
} public void roll(int field, int amount) {
computeIfNeed(field); if (!isChineseField(field)) {
super.roll(field, amount);
lastSetChinese = false;
areChineseFieldsComputed = false;
areSolarTermsComputed = false;
return;
} switch (field) {
case CHINESE_YEAR:
chineseYear += amount;
break;
case CHINESE_MONTH:
for (int i = 0; i < amount; i++) {
chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
}
break;
case CHINESE_DATE:
int maxDate = daysInChineseMonth(chineseYear, chineseMonth);
for (int i = 0; i < amount; i++) {
chineseDate++;
if (chineseDate > maxDate) {
chineseDate = 1;
}
}
default:
throw new IllegalArgumentException("不支持的field:" + field);
} lastSetChinese = true;
areFieldsSet = false;
areChineseFieldsComputed = false;
areSolarTermsComputed = false;
} /**
* 获得属性的中文,可以使用的属性字段为DAY_OF_WEEK以及所有农历属性字段。
*
* @param field
* @return
*/
public String getChinese(int field) {
computeIfNeed(field); switch (field) {
case CHINESE_YEAR:
return getChinese(CHINESE_HEAVENLY_STEM)
+ getChinese(CHINESE_EARTHLY_BRANCH) + "年";
case CHINESE_MONTH:
if (chineseMonth > 0)
return chineseMonthNames[chineseMonth] + "月";
else
return "闰" + chineseMonthNames[-chineseMonth] + "月";
case CHINESE_DATE:
return chineseDateNames[chineseDate];
case CHINESE_SECTIONAL_TERM:
return sectionalTermNames[get(Calendar.MONTH)];
case CHINESE_PRINCIPLE_TERM:
return principleTermNames[get(Calendar.MONTH)];
case CHINESE_HEAVENLY_STEM:
return stemNames[get(field)];
case CHINESE_EARTHLY_BRANCH:
return branchNames[get(field)];
case CHINESE_ZODIAC:
return animalNames[get(field)];
case Calendar.DAY_OF_WEEK:
return chineseWeekNames[get(field)];
case CHINESE_TERM_OR_DATE:
return getChinese(get(CHINESE_TERM_OR_DATE));
case LUNAR_FESTIVAL:
return getLunarFestival();
case SOLAR_FESTIVAL:
return getSolarFestival();
case FESTIVAL_OR_TERM_OR_DATE:
return getFestivalOrTermOrDate();
// TODO CHECK
case CHINESE_MONTH_OR_DATE:
return getChinese(get(CHINESE_MONTH_OR_DATE));
case CHINESE_TERM:
return getChineseTerm();
default:
throw new IllegalArgumentException("不支持的field中文获取:" + field);
}
} public String getSimpleGregorianDateString() {
return new StringBuffer().append(get(YEAR)).append("-")
.append(get(MONTH) + 1).append("-").append(get(DATE))
.toString();
} public String getSimpleChineseDateString() {
return new StringBuffer()
.append(get(CHINESE_YEAR))
.append("-")
.append(get(CHINESE_MONTH) > 0 ? "" + get(CHINESE_MONTH) : "*"
+ (-get(CHINESE_MONTH))).append("-")
.append(get(CHINESE_DATE)).toString();
} public String getChineseDateString() {
return new StringBuffer().append(getChinese(CHINESE_YEAR))
.append(getChinese(CHINESE_MONTH))
.append(getChinese(CHINESE_DATE)).toString();
} public String toString() {
StringBuffer buf = new StringBuffer();
buf.append(getSimpleGregorianDateString()).append(" | ")
.append(getChinese(DAY_OF_WEEK)).append(" | [农历]")
.append(getChineseDateString()).append(" ")
.append(getChinese(CHINESE_ZODIAC)).append("年 ")
.append(get(CHINESE_SECTIONAL_TERM)).append("日")
.append(getChinese(CHINESE_SECTIONAL_TERM)).append(" ")
.append(get(CHINESE_PRINCIPLE_TERM)).append("日")
.append(getChinese(CHINESE_PRINCIPLE_TERM));
return buf.toString();
} /**
* 判断是不是农历属性
*
* @param field
* @return
*/
private boolean isChineseField(int field) {
switch (field) {
case CHINESE_YEAR:
case CHINESE_MONTH:
case CHINESE_DATE:
case CHINESE_SECTIONAL_TERM:
case CHINESE_PRINCIPLE_TERM:
case CHINESE_HEAVENLY_STEM:
case CHINESE_EARTHLY_BRANCH:
case CHINESE_ZODIAC:
case CHINESE_TERM_OR_DATE:
case CHINESE_MONTH_OR_DATE:
return true;
default:
return false;
}
} /**
* 判断是不是与节气有关的属性
*
* @param field
* @return
*/
private boolean isChineseTermsField(int field) {
switch (field) {
case CHINESE_SECTIONAL_TERM:
case CHINESE_PRINCIPLE_TERM:
case CHINESE_TERM_OR_DATE:
return true;
default:
return false;
}
} /**
* 如果上一次设置的与这次将要设置或获取的属性不是同一类(农历/公历),<br>
* 例如上一次设置的是农历而现在要设置或获取公历,<br>
* 则需要先根据之前设置的农历日期计算出公历日期。
*
* @param field
*/
private void computeIfNeed(int field) {
if (isChineseField(field)) {
if (!lastSetChinese && !areChineseFieldsComputed) {
super.complete();
computeChineseFields();
areFieldsSet = true;
areChineseFieldsComputed = true;
areSolarTermsComputed = false;
}
if (isChineseTermsField(field) && !areSolarTermsComputed) {
computeSolarTerms();
areSolarTermsComputed = true;
}
} else {
if (lastSetChinese && !areFieldsSet) {
computeGregorianFields();
super.complete();
areFieldsSet = true;
areChineseFieldsComputed = true;
areSolarTermsComputed = false;
}
}
} /**
* 使用农历日期计算出公历日期
*/
private void computeGregorianFields() {
int y = chineseYear;
int m = chineseMonth;
int d = chineseDate;
areChineseFieldsComputed = true;
areFieldsSet = true;
lastSetChinese = false; // 调整日期范围
if (y < 1900)
y = 1899;
else if (y > 2100)
y = 2101; if (m < -12)
m = -12;
else if (m > 12)
m = 12; if (d < 1)
d = 1;
else if (d > 30)
d = 30; int dateint = y * 10000 + Math.abs(m) * 100 + d;
if (dateint < 19001111) { // 太小
set(1901, Calendar.JANUARY, 1);
super.complete();
} else if (dateint > 21001201) { // 太大
set(2100, Calendar.DECEMBER, 31);
super.complete();
} else {
if (Math.abs(m) > 12) {
m = 12;
}
int days = ChineseCalendar.daysInChineseMonth(y, m);
if (days == 0) {
m = -m;
days = ChineseCalendar.daysInChineseMonth(y, m);
}
if (d > days) {
d = days;
}
set(y, Math.abs(m) - 1, d);
computeChineseFields(); int amount = 0;
while (chineseYear != y || chineseMonth != m) {
amount += daysInChineseMonth(chineseYear, chineseMonth);
chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
if (chineseMonth == 1) {
chineseYear++;
}
}
amount += d - chineseDate; super.add(Calendar.DATE, amount);
}
computeChineseFields();
} /**
* 使用公历日期计算出农历日期
*/
private void computeChineseFields() {
int gregorianYear = internalGet(Calendar.YEAR);
int gregorianMonth = internalGet(Calendar.MONTH) + 1;
int gregorianDate = internalGet(Calendar.DATE); if (gregorianYear < 1901 || gregorianYear > 2100) {
return;
} int startYear, startMonth, startDate;
if (gregorianYear < 2000) {
startYear = baseYear;
startMonth = baseMonth;
startDate = baseDate;
chineseYear = baseChineseYear;
chineseMonth = baseChineseMonth;
chineseDate = baseChineseDate;
} else {
// 第二个对应日,用以提高计算效率
// 公历 2000 年 1 月 1 日,对应农历 4697(1999) 年 11 月 25 日
startYear = baseYear + 99;
startMonth = 1;
startDate = 1;
chineseYear = baseChineseYear + 99;
chineseMonth = 11;
chineseDate = 25;
} int daysDiff = 0; // 年
for (int i = startYear; i < gregorianYear; i++) {
if (isGregorianLeapYear(i)) {
daysDiff += 366; // leap year
} else {
daysDiff += 365;
}
} // 月
for (int i = startMonth; i < gregorianMonth; i++) {
daysDiff += daysInGregorianMonth(gregorianYear, i - 1);
} // 日
daysDiff += gregorianDate - startDate; chineseDate += daysDiff; int lastDate = daysInChineseMonth(chineseYear, chineseMonth);
while (chineseDate > lastDate) {
chineseDate -= lastDate;
chineseMonth = nextChineseMonth(chineseYear, chineseMonth);
if (chineseMonth == 1) {
chineseYear++;
}
lastDate = daysInChineseMonth(chineseYear, chineseMonth);
} } /**
* 计算节气
*/
private void computeSolarTerms() {
int gregorianYear = internalGet(Calendar.YEAR);
int gregorianMonth = internalGet(Calendar.MONTH); if (gregorianYear < 1901 || gregorianYear > 2100) {
return;
}
sectionalTerm = sectionalTerm(gregorianYear, gregorianMonth);
principleTerm = principleTerm(gregorianYear, gregorianMonth);
} /* 接下来是静态方法~ */
/**
* 是否为公历闰年
*
* @param year
* @return
*/
public static boolean isGregorianLeapYear(int year) {
boolean isLeap = false;
if (year % 4 == 0) {
isLeap = true;
}
if (year % 100 == 0) {
isLeap = false;
}
if (year % 400 == 0) {
isLeap = true;
}
return isLeap;
} /**
* 计算公历年的当月天数,公历月从0起始!
*
* @param y
* @param m
* @return
*/
public static int daysInGregorianMonth(int y, int m) {
int d = daysInGregorianMonth[m];
if (m == Calendar.FEBRUARY && isGregorianLeapYear(y)) {
d++; // 公历闰年二月多一天
}
return d;
} /**
* 计算公历年当月的节气,公历月从0起始!
*
* @param y
* @param m
* @return
*/
public static int sectionalTerm(int y, int m) {
m++;
if (y < 1901 || y > 2100) {
return 0;
}
int index = 0;
int ry = y - baseYear + 1;
while (ry >= sectionalTermYear[m - 1][index]) {
index++;
}
int term = sectionalTermMap[m - 1][4 * index + ry % 4];
if ((ry == 121) && (m == 4)) {
term = 5;
}
if ((ry == 132) && (m == 4)) {
term = 5;
}
if ((ry == 194) && (m == 6)) {
term = 6;
}
return term;
} /**
* 计算公历年当月的中气,公历月从0起始!
*
* @param y
* @param m
* @return
*/
public static int principleTerm(int y, int m) {
m++;
if (y < 1901 || y > 2100) {
return 0;
}
int index = 0;
int ry = y - baseYear + 1;
while (ry >= principleTermYear[m - 1][index]) {
index++;
}
int term = principleTermMap[m - 1][4 * index + ry % 4];
if ((ry == 171) && (m == 3)) {
term = 21;
}
if ((ry == 181) && (m == 5)) {
term = 21;
}
return term;
} /**
* 计算农历年的天数
*
* @param y
* @param m
* @return
*/
public static int daysInChineseMonth(int y, int m) {
// 注意:闰月 m < 0
int index = y - baseChineseYear + baseIndex;
int v = 0;
int l = 0;
int d = 30;
if (1 <= m && m <= 8) {
v = chineseMonths[2 * index];
l = m - 1;
if (((v >> l) & 0x01) == 1) {
d = 29;
}
} else if (9 <= m && m <= 12) {
v = chineseMonths[2 * index + 1];
l = m - 9;
if (((v >> l) & 0x01) == 1) {
d = 29;
}
} else {
v = chineseMonths[2 * index + 1];
v = (v >> 4) & 0x0F;
if (v != Math.abs(m)) {
d = 0;
} else {
d = 29;
for (int i = 0; i < bigLeapMonthYears.length; i++) {
if (bigLeapMonthYears[i] == index) {
d = 30;
break;
}
}
}
}
return d;
} /**
* 计算农历的下个月
*
* @param y
* @param m
* @return
*/
public static int nextChineseMonth(int y, int m) {
int n = Math.abs(m) + 1;
if (m > 0) {
int index = y - baseChineseYear + baseIndex;
int v = chineseMonths[2 * index + 1];
v = (v >> 4) & 0x0F;
if (v == m) {
n = -m;
}
}
if (n == 13) {
n = 1;
}
return n;
} /* 日历第一天的日期 */
private static final int baseYear = 1901;
private static final int baseMonth = 1;
private static final int baseDate = 1;
private static final int baseIndex = 0;
private static final int baseChineseYear = 1900;
private static final int baseChineseMonth = 11;
private static final int baseChineseDate = 11; /* 中文字符串 */
private static final String[] chineseWeekNames = { "", "星期日", "星期一", "星期二",
"星期三", "星期四", "星期五", "星期六" };
private static final String[] chineseMonthNames = { "", "正", "二", "三", "四",
"五", "六", "七", "八", "九", "十", "十一", "十二" };
private static final String[] chineseDateNames = { "", "初一", "初二", "初三",
"初四", "初五", "初六", "初七", "初八", "初九", "初十", "十一", "十二", "十三", "十四",
"十五", "十六", "十七", "十八", "十九", "二十", "廿一", "廿二", "廿三", "廿四", "廿五",
"廿六", "廿七", "廿八", "廿九", "三十" };
private static final String[] principleTermNames = { "大寒", "雨水", "春分",
"谷雨", "夏满", "夏至", "大暑", "处暑", "秋分", "霜降", "小雪", "冬至" };
private static final String[] sectionalTermNames = { "小寒", "立春", "惊蛰",
"清明", "立夏", "芒种", "小暑", "立秋", "白露", "寒露", "立冬", "大雪" };
private static final String[] stemNames = { "", "甲", "乙", "丙", "丁", "戊",
"己", "庚", "辛", "壬", "癸" };
private static final String[] branchNames = { "", "子", "丑", "寅", "卯", "辰",
"巳", "午", "未", "申", "酉", "戌", "亥" };
private static final String[] animalNames = { "", "鼠", "牛", "虎", "兔", "龙",
"蛇", "马", "羊", "猴", "鸡", "狗", "猪" }; /* 接下来是数据压缩表~ */
private static final int[] bigLeapMonthYears = { 6, 14, 19, 25, 33, 36, 38,
41, 44, 52, 55, 79, 117, 136, 147, 150, 155, 158, 185, 193 };
private static final char[][] sectionalTermMap = {
{ 7, 6, 6, 6, 6, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 5, 5,
5, 5, 5, 4, 5, 5 },
{ 5, 4, 5, 5, 5, 4, 4, 5, 5, 4, 4, 4, 4, 4, 4, 4, 4, 3, 4, 4, 4, 3,
3, 4, 4, 3, 3, 3 },
{ 6, 6, 6, 7, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 6, 5, 5,
5, 5, 4, 5, 5, 5, 5 },
{ 5, 5, 6, 6, 5, 5, 5, 6, 5, 5, 5, 5, 4, 5, 5, 5, 4, 4, 5, 5, 4, 4,
4, 5, 4, 4, 4, 4, 5 },
{ 6, 6, 6, 7, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5, 5, 6, 5, 5,
5, 5, 4, 5, 5, 5, 5 },
{ 6, 6, 7, 7, 6, 6, 6, 7, 6, 6, 6, 6, 5, 6, 6, 6, 5, 5, 6, 6, 5, 5,
5, 6, 5, 5, 5, 5, 4, 5, 5, 5, 5 },
{ 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7, 7, 7, 6, 6,
7, 7, 6, 6, 6, 7, 7 },
{ 8, 8, 8, 9, 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7,
7, 7, 6, 7, 7, 7, 6, 6, 7, 7, 7 },
{ 8, 8, 8, 9, 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7,
7, 7, 6, 7, 7, 7, 7 },
{ 9, 9, 9, 9, 8, 9, 9, 9, 8, 8, 9, 9, 8, 8, 8, 9, 8, 8, 8, 8, 7, 8,
8, 8, 7, 7, 8, 8, 8 },
{ 8, 8, 8, 8, 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7,
7, 7, 6, 6, 7, 7, 7 },
{ 7, 8, 8, 8, 7, 7, 8, 8, 7, 7, 7, 8, 7, 7, 7, 7, 6, 7, 7, 7, 6, 6,
7, 7, 6, 6, 6, 7, 7 } };
private static final char[][] sectionalTermYear = {
{ 13, 49, 85, 117, 149, 185, 201, 250, 250 },
{ 13, 45, 81, 117, 149, 185, 201, 250, 250 },
{ 13, 48, 84, 112, 148, 184, 200, 201, 250 },
{ 13, 45, 76, 108, 140, 172, 200, 201, 250 },
{ 13, 44, 72, 104, 132, 168, 200, 201, 250 },
{ 5, 33, 68, 96, 124, 152, 188, 200, 201 },
{ 29, 57, 85, 120, 148, 176, 200, 201, 250 },
{ 13, 48, 76, 104, 132, 168, 196, 200, 201 },
{ 25, 60, 88, 120, 148, 184, 200, 201, 250 },
{ 16, 44, 76, 108, 144, 172, 200, 201, 250 },
{ 28, 60, 92, 124, 160, 192, 200, 201, 250 },
{ 17, 53, 85, 124, 156, 188, 200, 201, 250 } };
private static final char[][] principleTermMap = {
{ 21, 21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20,
20, 20, 20, 20, 20, 19, 20, 20, 20, 19, 19, 20 },
{ 20, 19, 19, 20, 20, 19, 19, 19, 19, 19, 19, 19, 19, 18, 19, 19,
19, 18, 18, 19, 19, 18, 18, 18, 18, 18, 18, 18 },
{ 21, 21, 21, 22, 21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21,
20, 20, 20, 21, 20, 20, 20, 20, 19, 20, 20, 20, 20 },
{ 20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20, 21, 20, 20, 20, 20,
19, 20, 20, 20, 19, 19, 20, 20, 19, 19, 19, 20, 20 },
{ 21, 22, 22, 22, 21, 21, 22, 22, 21, 21, 21, 22, 21, 21, 21, 21,
20, 21, 21, 21, 20, 20, 21, 21, 20, 20, 20, 21, 21 },
{ 22, 22, 22, 22, 21, 22, 22, 22, 21, 21, 22, 22, 21, 21, 21, 22,
21, 21, 21, 21, 20, 21, 21, 21, 20, 20, 21, 21, 21 },
{ 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23, 22, 23, 23, 23,
22, 22, 23, 23, 22, 22, 22, 23, 22, 22, 22, 22, 23 },
{ 23, 24, 24, 24, 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23,
22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23, 23 },
{ 23, 24, 24, 24, 23, 23, 24, 24, 23, 23, 23, 24, 23, 23, 23, 23,
22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23, 23 },
{ 24, 24, 24, 24, 23, 24, 24, 24, 23, 23, 24, 24, 23, 23, 23, 24,
23, 23, 23, 23, 22, 23, 23, 23, 22, 22, 23, 23, 23 },
{ 23, 23, 23, 23, 22, 23, 23, 23, 22, 22, 23, 23, 22, 22, 22, 23,
22, 22, 22, 22, 21, 22, 22, 22, 21, 21, 22, 22, 22 },
{ 22, 22, 23, 23, 22, 22, 22, 23, 22, 22, 22, 22, 21, 22, 22, 22,
21, 21, 22, 22, 21, 21, 21, 22, 21, 21, 21, 21, 22 } };
private static final char[][] principleTermYear = {
{ 13, 45, 81, 113, 149, 185, 201 },
{ 21, 57, 93, 125, 161, 193, 201 },
{ 21, 56, 88, 120, 152, 188, 200, 201 },
{ 21, 49, 81, 116, 144, 176, 200, 201 },
{ 17, 49, 77, 112, 140, 168, 200, 201 },
{ 28, 60, 88, 116, 148, 180, 200, 201 },
{ 25, 53, 84, 112, 144, 172, 200, 201 },
{ 29, 57, 89, 120, 148, 180, 200, 201 },
{ 17, 45, 73, 108, 140, 168, 200, 201 },
{ 28, 60, 92, 124, 160, 192, 200, 201 },
{ 16, 44, 80, 112, 148, 180, 200, 201 },
{ 17, 53, 88, 120, 156, 188, 200, 201 } }; private static final char[] daysInGregorianMonth = { 31, 28, 31, 30, 31,
30, 31, 31, 30, 31, 30, 31 };
private static final char[] chineseMonths = { 0x00, 0x04, 0xad, 0x08, 0x5a,
0x01, 0xd5, 0x54, 0xb4, 0x09, 0x64, 0x05, 0x59, 0x45, 0x95, 0x0a,
0xa6, 0x04, 0x55, 0x24, 0xad, 0x08, 0x5a, 0x62, 0xda, 0x04, 0xb4,
0x05, 0xb4, 0x55, 0x52, 0x0d, 0x94, 0x0a, 0x4a, 0x2a, 0x56, 0x02,
0x6d, 0x71, 0x6d, 0x01, 0xda, 0x02, 0xd2, 0x52, 0xa9, 0x05, 0x49,
0x0d, 0x2a, 0x45, 0x2b, 0x09, 0x56, 0x01, 0xb5, 0x20, 0x6d, 0x01,
0x59, 0x69, 0xd4, 0x0a, 0xa8, 0x05, 0xa9, 0x56, 0xa5, 0x04, 0x2b,
0x09, 0x9e, 0x38, 0xb6, 0x08, 0xec, 0x74, 0x6c, 0x05, 0xd4, 0x0a,
0xe4, 0x6a, 0x52, 0x05, 0x95, 0x0a, 0x5a, 0x42, 0x5b, 0x04, 0xb6,
0x04, 0xb4, 0x22, 0x6a, 0x05, 0x52, 0x75, 0xc9, 0x0a, 0x52, 0x05,
0x35, 0x55, 0x4d, 0x0a, 0x5a, 0x02, 0x5d, 0x31, 0xb5, 0x02, 0x6a,
0x8a, 0x68, 0x05, 0xa9, 0x0a, 0x8a, 0x6a, 0x2a, 0x05, 0x2d, 0x09,
0xaa, 0x48, 0x5a, 0x01, 0xb5, 0x09, 0xb0, 0x39, 0x64, 0x05, 0x25,
0x75, 0x95, 0x0a, 0x96, 0x04, 0x4d, 0x54, 0xad, 0x04, 0xda, 0x04,
0xd4, 0x44, 0xb4, 0x05, 0x54, 0x85, 0x52, 0x0d, 0x92, 0x0a, 0x56,
0x6a, 0x56, 0x02, 0x6d, 0x02, 0x6a, 0x41, 0xda, 0x02, 0xb2, 0xa1,
0xa9, 0x05, 0x49, 0x0d, 0x0a, 0x6d, 0x2a, 0x09, 0x56, 0x01, 0xad,
0x50, 0x6d, 0x01, 0xd9, 0x02, 0xd1, 0x3a, 0xa8, 0x05, 0x29, 0x85,
0xa5, 0x0c, 0x2a, 0x09, 0x96, 0x54, 0xb6, 0x08, 0x6c, 0x09, 0x64,
0x45, 0xd4, 0x0a, 0xa4, 0x05, 0x51, 0x25, 0x95, 0x0a, 0x2a, 0x72,
0x5b, 0x04, 0xb6, 0x04, 0xac, 0x52, 0x6a, 0x05, 0xd2, 0x0a, 0xa2,
0x4a, 0x4a, 0x05, 0x55, 0x94, 0x2d, 0x0a, 0x5a, 0x02, 0x75, 0x61,
0xb5, 0x02, 0x6a, 0x03, 0x61, 0x45, 0xa9, 0x0a, 0x4a, 0x05, 0x25,
0x25, 0x2d, 0x09, 0x9a, 0x68, 0xda, 0x08, 0xb4, 0x09, 0xa8, 0x59,
0x54, 0x03, 0xa5, 0x0a, 0x91, 0x3a, 0x96, 0x04, 0xad, 0xb0, 0xad,
0x04, 0xda, 0x04, 0xf4, 0x62, 0xb4, 0x05, 0x54, 0x0b, 0x44, 0x5d,
0x52, 0x0a, 0x95, 0x04, 0x55, 0x22, 0x6d, 0x02, 0x5a, 0x71, 0xda,
0x02, 0xaa, 0x05, 0xb2, 0x55, 0x49, 0x0b, 0x4a, 0x0a, 0x2d, 0x39,
0x36, 0x01, 0x6d, 0x80, 0x6d, 0x01, 0xd9, 0x02, 0xe9, 0x6a, 0xa8,
0x05, 0x29, 0x0b, 0x9a, 0x4c, 0xaa, 0x08, 0xb6, 0x08, 0xb4, 0x38,
0x6c, 0x09, 0x54, 0x75, 0xd4, 0x0a, 0xa4, 0x05, 0x45, 0x55, 0x95,
0x0a, 0x9a, 0x04, 0x55, 0x44, 0xb5, 0x04, 0x6a, 0x82, 0x6a, 0x05,
0xd2, 0x0a, 0x92, 0x6a, 0x4a, 0x05, 0x55, 0x0a, 0x2a, 0x4a, 0x5a,
0x02, 0xb5, 0x02, 0xb2, 0x31, 0x69, 0x03, 0x31, 0x73, 0xa9, 0x0a,
0x4a, 0x05, 0x2d, 0x55, 0x2d, 0x09, 0x5a, 0x01, 0xd5, 0x48, 0xb4,
0x09, 0x68, 0x89, 0x54, 0x0b, 0xa4, 0x0a, 0xa5, 0x6a, 0x95, 0x04,
0xad, 0x08, 0x6a, 0x44, 0xda, 0x04, 0x74, 0x05, 0xb0, 0x25, 0x54,
0x03 }; private String getChineseTerm() {
if (get(Calendar.DATE) == get(CHINESE_SECTIONAL_TERM)) {
return sectionalTermNames[get(Calendar.MONTH)];
} else if (get(Calendar.DATE) == get(CHINESE_PRINCIPLE_TERM)) {
return principleTermNames[get(Calendar.MONTH)];
} else
return null;
}
// add by skywang
private String getLunarFestival() {
int day = get(CHINESE_DATE);
int month = get(CHINESE_MONTH);
String sToday = day < 10 ? "0" + day:"" + day;
String sMonth = month<10 ? "0" +(month):""+(month); return lFestival.get(sMonth+sToday);
}
private String getSolarFestival() {
int day = get(Calendar.DATE);
int month = get(Calendar.MONTH);
String sToday = day < 10 ? "0" + day:"" + day;
String sMonth = month<10 ? "0" +(month+1):""+(month+1); return sFestival.get(sMonth+sToday);
}
private String getFestivalOrTermOrDate() {
String ret;
if ((ret = getSolarFestival()) != null)
return ret;
if ((ret = getLunarFestival()) != null)
return ret;
return getChinese(get(CHINESE_TERM_OR_DATE));
} //公历节日
private static HashMap<String,String> sFestival =new HashMap<String,String>();
// 农历介入
private static HashMap<String,String> lFestival =new HashMap<String,String>();
static {
sFestival.put("0101","元旦");
sFestival.put("0214","情人节");
sFestival.put("0308","妇女节");
sFestival.put("0312","植树节");
sFestival.put("0401","愚人节");
sFestival.put("0501","劳动节");
sFestival.put("0504","青年节");
sFestival.put("0601","儿童节");
sFestival.put("0701","建党节");
sFestival.put("0801","建军节");
sFestival.put("0910","教师节");
sFestival.put("1001","国庆节");
sFestival.put("1031","万圣节");
// sFestival.put("1112","孙中山诞辰");
sFestival.put("1225","圣诞节"); lFestival.put("0101","春节");
// lFestival.put("0102","大年初二");
// lFestival.put("0103","大年初三");
lFestival.put("0115","元宵节");
lFestival.put("0505","端午节");
lFestival.put("0707","七夕");
lFestival.put("0815","中秋节");
lFestival.put("0909","重阳节");
lFestival.put("1208","腊八节");
// lFestival.put("1299","除夕");
}
}
自定义的Calendar接口
这些接口在写日历程序时可能会用到。
源代码如下(CalendarSelfDefineTest.java):
import java.util.Calendar; /**
* 根据Calendar的API封装的一些常用函数
*
* @author skywang
* @email kuiwu-wang@163.com
*/
public class CalendarSelfDefineTest { public static void main(String[] args) {
Calendar cal = Calendar.getInstance(); // 设置日期为“2013-09-18”
cal.set(2013, Calendar.SEPTEMBER, 18); // 获取“年”
System.out.printf("year: %s\n", getYear(cal) );
// 获取“月”
System.out.printf("month: %s\n", getMonth(cal) );
// 获取“上月”
System.out.printf("previcou month: %s\n", getLastMonth(cal) );
// 获取“下月”
System.out.printf("next month: %s\n", getNextMonth(cal) );
// 获取“日”
System.out.printf("day: %s\n", getDay(cal) );
// 获取Cal对应星期几
System.out.printf("weekday: %s\n", getWeekDay(cal) );
// 本月天数
System.out.printf("Current Month days: %s\n", getMonthDays(cal) );
// 上月天数
System.out.printf("Previcous Month days: %s\n", getPrevMonthDays(cal) );
// 下月天数
System.out.printf("Next Month days: %s\n", getNextMonthDays(cal) );
// 获取当月第一天的星期几
System.out.printf("First day' weekday : %s\n", getFirstDayWeekday(cal) );
// 获取当前月最后一天的星期几
System.out.printf("Last day' weekday : %s\n", getLastDayWeekday(cal) );
// 获取上月最后一天的星期几
System.out.printf("PrevMonth Last day' weekday: %s\n", getLastDayWeekdayOfPrevMonth(cal) );
// 获取下月第一天的星期几
System.out.printf("NextMonth First day' weekday: %s\n", getFirstDayWeekdayOfNextMonth(cal) );
} /**
* 获取“年”
*
* @return 例如,2013-09-18,则返回2013
*/
public static int getYear(Calendar cal) {
return cal.get(Calendar.YEAR);
} /**
* 获取“月”
*
* @return 返回值可以为以下值:
* JANUARY, FEBRUARY, MARCH, APRIL, MAY, JUNE, JULY, AUGUST, SEPTEMBER, OCTOBER, NOVEMBER, DECEMBER, UNDECIMBER。
* 其中第一个月是 JANUARY,它为 0。
*
* 例如,2013-09-18,则返回8
*/
public static int getMonth(Calendar cal) {
return cal.get(Calendar.MONTH);
} /**
* 获取“上一个月”
*
* @return 返回值可以为以下值:
* JANUARY, FEBRUARY, MARCH, APRIL, MAY, JUNE, JULY, AUGUST, SEPTEMBER, OCTOBER, NOVEMBER, DECEMBER, UNDECIMBER。
* 其中第一个月是 JANUARY,它为 0。
*
* 例如,2012-01-12的上一个月是“11”(即DECEMBER)。
*/
public static int getLastMonth(Calendar cal) {
return (cal.get(Calendar.MONTH) + 11) % 12;
} /**
* 获取“下一个月”
*
* @return 返回值可以为以下值:
* JANUARY, FEBRUARY, MARCH, APRIL, MAY, JUNE, JULY, AUGUST, SEPTEMBER, OCTOBER, NOVEMBER, DECEMBER, UNDECIMBER。
* 其中第一个月是 JANUARY,它为 0。
*
* 例如,2013-12-12的下一个月是“1”(即DECEMBER)。
*/
public static int getNextMonth(Calendar cal) {
return (cal.get(Calendar.MONTH) + 13) % 12;
} /**
* 获取“日”
*
* @return 例如,2013-09-18,则返回18
*
*/
public static int getDay(Calendar cal) {
return cal.get(Calendar.DATE);
} /**
* 获取“本月的天数”
*
* @return 例如,2013-09-18,则返回30
*
*/
public static int getMonthDays(Calendar cal) {
return cal.getActualMaximum(Calendar.DATE);
} /**
* 获取“上一个月的天数”
*
* @return 例如,2013-01-18,则返回31 (因为2012-12有31天)
*
*/
public static int getPrevMonthDays(Calendar cal) {
Calendar tmpCal = (Calendar)cal.clone(); // 克隆cal。后面对tmpCal操作,就不会改变cal
tmpCal.add(Calendar.MONTH, -1); // 设为“上一个月”
return tmpCal.getActualMaximum(Calendar.DATE);
} /**
* 获取“下一个月的天数”
*
* @return 例如,2013-12-18,则返回31 (因为2014-01有31天)
*
*/
public static int getNextMonthDays(Calendar cal) {
Calendar tmpCal = (Calendar)cal.clone(); // 克隆cal。后面对tmpCal操作,就不会改变cal
tmpCal.add(Calendar.MONTH, 1); // 设为“下一个月”
return tmpCal.getActualMaximum(Calendar.DATE);
} /**
* 获取Cal对应星期几
*
* @return 返回“星期几”,可以为以下值:
* SUNDAY、MONDAY、TUESDAY、WEDNESDAY、THURSDAY、FRIDAY 和 SATURDAY。
* SUNDAY为1,MONDAY为2,依次类推。
* 例如,2013-09-18(星期三),则返回4
*/
public static int getWeekDay(Calendar cal) {
return cal.get(Calendar.DAY_OF_WEEK);
} /**
* 获取当月第一天对应星期几
*
* @return SUNDAY为1,MONDAY为2,依次类推。
*/
public static int getFirstDayWeekday(Calendar cal) { Calendar tmpCal = (Calendar)cal.clone(); // 克隆cal。后面对tmpCal操作,就不会改变cal
tmpCal.set(Calendar.DATE, 1); // 把日期设置为当月第一天
return tmpCal.get(Calendar.DAY_OF_WEEK);
} /**
* 获取当前月最后一天对应星期几
*
* @return SUNDAY为1,MONDAY为2,依次类推。
*/
public static int getLastDayWeekday(Calendar cal) {
Calendar tmpCal = (Calendar)cal.clone(); // 克隆cal。后面对tmpCal操作,就不会改变cal
tmpCal.set(Calendar.DATE, 1); // 把日期设置为当月第一天
tmpCal.roll(Calendar.DATE, -1); // 把日期设置为当月最后一天
return tmpCal.get(Calendar.DAY_OF_WEEK);
} /**
* 获取上月最后一天的星期几
*
* @return SUNDAY为1,MONDAY为2,依次类推。
*/
public static int getLastDayWeekdayOfPrevMonth(Calendar cal) { Calendar tmpCal = (Calendar)cal.clone(); // 克隆cal。后面对tmpCal操作,就不会改变cal
tmpCal.set(Calendar.DATE, 1); // 把日期设置为当月第一天
tmpCal.add(Calendar.DATE, -1); // 把日期设置为上一个月最后一天
return tmpCal.get(Calendar.DAY_OF_WEEK);
} /**
* 获取下月第一天的星期偏移
*
* @return SUNDAY为1,MONDAY为2,依次类推。
*/
public static int getFirstDayWeekdayOfNextMonth(Calendar cal) { Calendar tmpCal = (Calendar)cal.clone(); // 克隆cal。后面对tmpCal操作,就不会改变cal
tmpCal.add(Calendar.MONTH, 1); // 设为“下一个月”
tmpCal.set(Calendar.DATE, 1); // 设为“第一天”
return tmpCal.get(Calendar.DAY_OF_WEEK);
}
}
更多内容
Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(1) Calendar
Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(2) 自己封装的Calendar接口
Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(3) Date
Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(4) DateFormat
Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(5) SimpleDateFormat
Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(6) Locale
Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(7) TimeZone
Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(2) 自己封装的Calendar接口的更多相关文章
- Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(1) Calendar
Java 操作日期/时间,往往会涉及到Calendar,Date,DateFormat这些类. 最近决定把这些内容系统的整理一下,这样以后使用的时候,会更得心应手.本章的内容是主要讲解“Java时间框 ...
- Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(3) Date
本章主要介绍Date类,并通过示例学习如何使用它.最后,讲解一下UTC.GMT和时区的关系. Date 介绍 Date 定义 public class Date implements java.io. ...
- Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(4) DateFormat
本章主要介绍DateFormat. DateFormat 介绍 DateFormat 的作用是 格式化并解析“日期/时间”.实际上,它是Date的格式化工具,它能帮助我们格式化Date,进而将Date ...
- Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(5) SimpleDateFormat
本章介绍SimpleDateFormat. SimpleDateFormat 介绍 SimpleDateFormat 是一个格式化Date 以及 解析日期字符串 的工具.它的最常用途是,能够按照指定的 ...
- Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(6) Locale
本章介绍Locale. 1 Locale 介绍 Locale 表示地区.每一个Locale对象都代表了一个特定的地理.政治和文化地区. 在操作 Date, Calendar等表示日期/时间的对象时,经 ...
- Java Calendar,Date,DateFormat,TimeZone,Locale等时间相关内容的认知和使用(7) TimeZone
本章介绍TimeZone. TimeZone 简介 TimeZone 表示时区偏移量,也可以计算夏令时.在操作 Date, Calendar等表示日期/时间的对象时,经常会用到TimeZone:因为不 ...
- java之Date(日期)、Date格式化、Calendar(日历)
参考http://how2j.cn/k/date/date-date/346.html Date(日期) Date类 注意:是java.util.Date; 而非 java.sql.Date,此类是给 ...
- Java中Date和Calender类的使用方法
查看文章 Java中Date和Calender类的使用方法 2009-10-04 20:49 Date和Calendar是Java类库里提供对时间进行处理的类,由于日期在商业逻辑的应用中占据着 ...
- Java中在java.sql.Date的系统时间上加上30天并写入oracle
在java.sql.Date的系统时间上加上30天,并写入oracle 思路:通过 Calendar.getInstance() 获得对象,然后 add() 方法添加 时间,再通过 new java. ...
随机推荐
- .NetCore使用Swagger进行单版本或多版本控制处理
前面已经介绍过了Swagger的基础使用了 下面继续分别详细说明下 不添加版本控制以及添加版本控制的使用情况,其实也基本一致,对看起来可能更加容易理解 第一步 导入nuget包 nuget导入Swas ...
- JS两种事件的触发方式
一.入侵式触发方式 <input type="button" id="one" onclick="事件" /> 二.非入侵式触发 ...
- js 判断移动端是否安装应用
var u = navigator.userAgent; var isAndroid = u.indexOf('Android') > -1 || u.indexOf('Linux') > ...
- 神仙数据结构——FHQ_Treap
$FHQ\_Treap$是平衡树的一种,它不仅支持几乎所有的平衡树的操作,而且实现特别简单,总共只有两个操作.这里来简单介绍一下. 基本操作 $FHQ\_Treap$和$Treap$一样是需要用随机值 ...
- 给定一种 pattern(模式) 和一个字符串 str ,判断 str 是否遵循相同的模式。 这里的遵循指完全匹配,例如, pattern 里的每个字母和字符串 str 中的每个非空单词之间存在着双向连接的对应模式。
这个是LeetCode上的一道题目.本机上运行时正确的,但是LeetCode上显示是错误的,所以没有办法了只能记录在博客上了. 我的想法是先把pattern和str都转化成数组.例如"abb ...
- 005.iSCSI客户端配置示例-Windows
一 环境 Linux作为iSCSI服务端,Windows2008R2作为iSCSI客户端 二 服务端配置过程 2.1 客户端配置 在Linux上参照之前的配置建立三个LUN卷作为共享盘,最终配置如下: ...
- WIN10下使用Anaconda配置opencv、tensorflow、pygame并在pycharm中运用
昨天想运行一段机器学习的代码,在win10系统下配置了一天的python环境,真的是头疼,准备写篇博客来帮助后面需要配置环境的兄弟. 1.下载Anaconda 根据昨天的经历,发现Anaconda真的 ...
- .NET工作准备--02基础知识
(已过时) 框架基础,语法基础,字符串&集合&流,常见类和接口; 02.net基础(重点) -第一部分 框架基础 1.基础概念 CTS(Common Type System),CLS( ...
- Spring框架学习01——使用IDEA开发Spring程序
1.创建项目 点击“Create New Project”,新建项目 选择Maven项目 项目配置 使用本地安装的Maven 一直点击Next,最后点击完成当控制台中出现“BUILD SUCCESS” ...
- Java持久化之 -- 傲娇的NIO
NIO: Jdk 1.4+ New IO 面向通道和缓冲区 所在包:java.nio 执行流程: 数据总数由通道写入到buffer , 或者是从buffer写入通道 完全替换IO(面向流 单向的) ...