冷面阎罗

低调做人&&高调做事
随笔 - 208, 文章 - 3, 评论 - 593, 引用 - 0
数据加载中……

对前几天作的java万年历,再弄修改,加上农历

     前几天在blog中,对网友的java万年历作修改,看到有的网友说能不能加上农历,后来在网上看到有人写过几个阳历转阴历的算法,我比较了一个发现,这个算法还算不错,只要有的计算机编程基础的人看明白应该是没有问题的,其实这个就和我们以前在c中,判断一天是周几的算法差不多,都是和某一个特定的时间作比较,算出差多少天,再根据月大月小瑞月这些规则,算出是农历的那年那月那日.
package clock;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;

public class Lunar {
    
private int year;
    
private int month;
    
private int day;
    
private boolean leap;
    
final static String chineseNumber[] = {"""""""""""""""""""""十一""十二"};
    
static SimpleDateFormat chineseDateFormat = new SimpleDateFormat("yyyy年MM月dd日");
    
final static long[] lunarInfo = new long[]
    
{0x04bd80x04ae00x0a5700x054d50x0d2600x0d9500x165540x056a00x09ad00x055d2,
     
0x04ae00x0a5b60x0a4d00x0d2500x1d2550x0b5400x0d6a00x0ada20x095b00x14977,
     
0x049700x0a4b00x0b4b50x06a500x06d400x1ab540x02b600x095700x052f20x04970,
     
0x065660x0d4a00x0ea500x06e950x05ad00x02b600x186e30x092e00x1c8d70x0c950,
     
0x0d4a00x1d8a60x0b5500x056a00x1a5b40x025d00x092d00x0d2b20x0a9500x0b557,
     
0x06ca00x0b5500x153550x04da00x0a5d00x145730x052d00x0a9a80x0e9500x06aa0,
     
0x0aea60x0ab500x04b600x0aae40x0a5700x052600x0f2630x0d9500x05b570x056a0,
     
0x096d00x04dd50x04ad00x0a4d00x0d4d40x0d2500x0d5580x0b5400x0b5a00x195a6,
     
0x095b00x049b00x0a9740x0a4b00x0b27a0x06a500x06d400x0af460x0ab600x09570,
     
0x04af50x049700x064b00x074a30x0ea500x06b580x055c00x0ab600x096d50x092e0,
     
0x0c9600x0d9540x0d4a00x0da500x075520x056a00x0abb70x025d00x092d00x0cab5,
     
0x0a9500x0b4a00x0baa40x0ad500x055d90x04ba00x0a5b00x151760x052b00x0a930,
     
0x079540x06aa00x0ad500x05b520x04b600x0a6e60x0a4e00x0d2600x0ea650x0d530,
     
0x05aa00x076a30x096d00x04bd70x04ad00x0a4d00x1d0b60x0d2500x0d5200x0dd45,
     
0x0b5a00x056d00x055b20x049b00x0a5770x0a4b00x0aa500x1b2550x06d200x0ada0}
;

    
//====== 传回农历 y年的总天数
    final private static int yearDays(int y) {
        
int i, sum = 348;
        
for (i = 0x8000; i > 0x8; i >>= 1{
            
if ((lunarInfo[y - 1900& i) != 0) sum += 1;
        }

        
return (sum + leapDays(y));
    }


    
//====== 传回农历 y年闰月的天数
    final private static int leapDays(int y) {
        
if (leapMonth(y) != 0{
            
if ((lunarInfo[y - 1900& 0x10000!= 0)
                
return 30;
            
else
                
return 29;
        }
 else
            
return 0;
    }


    
//====== 传回农历 y年闰哪个月 1-12 , 没闰传回 0
    final private static int leapMonth(int y) {
        
return (int) (lunarInfo[y - 1900& 0xf);
    }


    
//====== 传回农历 y年m月的总天数
    final private static int monthDays(int y, int m) {
        
if ((lunarInfo[y - 1900& (0x10000 >> m)) == 0)
            
return 29;
        
else
            
return 30;
    }


    
//====== 传回农历 y年的生肖
    final public String animalsYear() {
        
final String[] Animals = new String[]{""""""""""""""""""""""""};
        
return Animals[(year - 4% 12];
    }


    
//====== 传入 月日的offset 传回干支, 0=甲子
    final private static String cyclicalm(int num) {
        
final String[] Gan = new String[]{""""""""""""""""""""};
        
final String[] Zhi = new String[]{""""""""""""""""""""""""};
        
return (Gan[num % 10+ Zhi[num % 12]);
    }


    
//====== 传入 offset 传回干支, 0=甲子
    final public String cyclical() {
        
int num = year - 1900 + 36;
        
return (cyclicalm(num));
    }


    
/**
     * 传出y年m月d日对应的农历.
     * yearCyl3:农历年与1864的相差数              ?
     * monCyl4:从1900年1月31日以来,闰月数
     * dayCyl5:与1900年1月31日相差的天数,再加40      ?
     * 
@param cal 
     * 
@return 
     
*/

    
public Lunar(Calendar cal) {
        @SuppressWarnings(
"unused"int yearCyl, monCyl, dayCyl;
        
int leapMonth = 0;
        Date baseDate 
= null;
        
try {
            baseDate 
= chineseDateFormat.parse("1900年1月31日");
        }
 catch (ParseException e) {
            e.printStackTrace();  
//To change body of catch statement use Options | File Templates.
        }


        
//求出和1900年1月31日相差的天数
        int offset = (int) ((cal.getTime().getTime() - baseDate.getTime()) / 86400000L);
        dayCyl 
= offset + 40;
        monCyl 
= 14;

        
//用offset减去每农历年的天数
        
// 计算当天是农历第几天
        
//i最终结果是农历的年份
        
//offset是当年的第几天
        int iYear, daysOfYear = 0;
        
for (iYear = 1900; iYear < 2050 && offset > 0; iYear++{
            daysOfYear 
= yearDays(iYear);
            offset 
-= daysOfYear;
            monCyl 
+= 12;
        }

        
if (offset < 0{
            offset 
+= daysOfYear;
            iYear
--;
            monCyl 
-= 12;
        }

        
//农历年份
        year = iYear;

        yearCyl 
= iYear - 1864;
        leapMonth 
= leapMonth(iYear); //闰哪个月,1-12
        leap = false;

        
//用当年的天数offset,逐个减去每月(农历)的天数,求出当天是本月的第几天
        int iMonth, daysOfMonth = 0;
        
for (iMonth = 1; iMonth < 13 && offset > 0; iMonth++{
            
//闰月
            if (leapMonth > 0 && iMonth == (leapMonth + 1&& !leap) {
                
--iMonth;
                leap 
= true;
                daysOfMonth 
= leapDays(year);
            }
 else
                daysOfMonth 
= monthDays(year, iMonth);

            offset 
-= daysOfMonth;
            
//解除闰月
            if (leap && iMonth == (leapMonth + 1)) leap = false;
            
if (!leap) monCyl++;
        }

        
//offset为0时,并且刚才计算的月份是闰月,要校正
        if (offset == 0 && leapMonth > 0 && iMonth == leapMonth + 1{
            
if (leap) {
                leap 
= false;
            }
 else {
                leap 
= true;
                
--iMonth;
                
--monCyl;
            }

        }

        
//offset小于0时,也要校正
        if (offset < 0{
            offset 
+= daysOfMonth;
            
--iMonth;
            
--monCyl;
        }

        month 
= iMonth;
        day 
= offset + 1;
    }


    
public static String getChinaDayString(int day) {
        String chineseTen[] 
= {"""""廿"""};
        
int n = day % 10 == 0 ? 9 : day % 10 - 1;
        
if (day > 30)
            
return "";
        
if (day == 10)
            
return "初十";
        
else
            
return chineseTen[day / 10+ chineseNumber[n];
    }


    
public String toString() {
        
return year + "" + (leap ? "" : ""+ chineseNumber[month - 1+ "" + getChinaDayString(day);
    }


    
public static void main(String[] args) throws ParseException {
        Calendar today 
= Calendar.getInstance();
        today.setTime(chineseDateFormat.parse(
"2003年1月1日"));
        Lunar lunar 
= new Lunar(today);

        System.out.println(
"北京时间:" + chineseDateFormat.format(today.getTime()) + " 农历" + lunar);
    }

}
 
      在以前程序的label[i].setText(count +"");下设置label的浮动提示
            try {
                now.setTime(chineseDateFormat.parse(year_log
+""+(month_log+1)+""+count+""));
            }
 catch (ParseException e) {
                e.printStackTrace();
            }

            label[i].setToolTipText(
"农历"+new Lunar(now));
就能出现这种效果:
1.jpg

posted on 2007-01-04 13:08 冷面阎罗 阅读(8319) 评论(17)  编辑  收藏 所属分类: java

评论

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

农历仍然是“丙戌”年……
2007-01-04 14:01 | weide

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

支持一下, 农历算法! 顶!
2007-01-04 14:03 | BeanSoft

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

不错。。


ps:
现在怎么都对swt swing j2se界面编程感兴趣??
觉的有点奇怪~~~
2007-01-04 16:31 | gm

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

回楼上,没办法,底层 的framework就是这个
2007-01-04 16:51 | zeyuphoenix

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

是呀,没有 SWT, SWING, 哪里来的 IDE 啊? 国内的人大多写写网页就够了, 可是要做 IDE 你不学这些就没办法了. PS, 谁要是能用网页做个 IDE, 那就 NB 大了!
2007-01-09 10:03 | BeanSoft

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

这些,都是java的基础的基础,如果连基础都一知办解的,那又如何深入研究java呢?
翻开任何一本java的书,在基础部分都会有swing,swt这些.
所以,希望想搞java的人,这方面最好要学一些,这仅是个人意见
2007-01-09 14:00 | 冷面阎罗

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

int yearCyl, monCyl, dayCyl, hourCyl
是为了传入到 final private static String cyclicalm(int num)
里面分别求出年,月,日,时辰的天干地支
int offset = (int) ((cal.getTime().getTime() - baseDate.getTime()) / 86400000L);
dayCyl = offset + 40;
monCyl = 14;
下面增加了这句算时辰的步长
hourCyl = (int) ((cal.getTime().getTime() - baseDate.getTime() + 3300000L) / 7200000L) ;

toString修改成下面这样,呵呵,这可以输出年月日时的天干地支,也就是所谓的八字了,大伙看看正确不?呵呵,
public String toString() {
return year + "年" + (leap ? "闰" : "") + chineseNumber[month - 1] + "月" + getChinaDayString(day)
+" "+ cyclicalm(yearCyl)+"年"+ cyclicalm(monCyl)+"月"+ cyclicalm(dayCyl)+"日" + cyclicalm(hourCyl)+"时辰";
}
2007-10-12 10:41 | 御风而行

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

如何添加类似备忘录的功能呢
2008-06-24 00:08 | 飞过海

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

天干地支的月份是按照节气的
2009-11-28 11:16 | CU

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

楼主能不能留下QQ
我想要一个八字排版生成装置~
不知道楼主有没有兴趣

syitryu@qq.com
2009-11-28 11:19 | CU

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

这些代码有错误啊,实现不了
2010-01-02 20:29 | 健健

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

楼主能不能发一份完整的代码给我,运行不了,急需,谢谢啊,邮箱zhiyun1128@163.com
2010-01-02 20:34 | 小坏蛋

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

楼主~怎么运行不了的?
现在要做个类似的东西,那个急啊。。。
能不能直接发个完整的万年历给我?chanthuang@qq.com
2011-04-09 00:37 | chant

# re: 对前几天作的java万年历,再弄修改,加上农历[未登录]  回复  更多评论   

很感谢楼主,麻烦问下能不能再加上二十四节气?
2011-06-07 15:03 |

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

急求完整程序,楼主也给我发一份吧,邮箱:760623648@qq.com。谢谢啊!
2013-02-27 09:28 | 咫尺天涯

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

有没有点击那种下拉列表弹出时间选项卡的啊?求源码
2014-08-27 10:51 | zhangyi

# re: 对前几天作的java万年历,再弄修改,加上农历  回复  更多评论   

建议用冬月、腊月而不是十一月、十二月等
2015-02-23 15:24 | -

只有注册用户登录后才能发表评论。


网站导航:
博客园   IT新闻   Chat2DB   C++博客   博问