京公网安备 11010802034615号
经营许可证编号:京B2-20210330
来源:【公众号】
Python技术
决定和女朋友去领证了,心里那个激动啊,无以言表!我们俩都是比较随性的,准备拿到户口本就去领。
可谁知女朋友回家拿户口本的时候,跟我说:最近可能领不了了!
what?到手的鸭子要飞了?我心里咯噔一下。
询问后才知道,丈母娘说领证可以,但是要选择一个良辰吉日,要求有俩:一个是看万年历,选取宜“婚假”的日子;一个是需要选择农历的双数日期,双数代表吉利。
听了之后,我拍着胸脯说没问题。接着准备去翻万年历了,可不想这时候女朋友来一句:你个呆子,还准备一天天地去翻啊?写个小程序不就1秒钟的事吗?
我拍了拍脑袋,对哦,还是老婆聪明!话不多说,打开电脑就开干。
我在百度输入框输入“万年历”查询,弹出的第一个当然是百度自己的万年历咯,但是我不想在百度上耗时间,因为时间紧,任务重,我选取一个相对容易的。
这个网站看起来信息比较全,并且不是那种热门的大网站,所以应该获取信息相对容易些。
首页也比较清晰明了,我所需要的几大信息(日历、农历日期、黄历)都有。
接着,我们来看看页面的请求,来定位我们所需信息的请求。
我在这个页面不算多的请求里面发现了这个请求(https://staticwnl.tianqistatic.com/Public/Home/js/api/yjs/2021.js):
这好像就是我们的目标请求,我们来看看返回:
非常好,人家一次性把一年的数据都返回了,感觉好简单啊,都不用咱们一天天请求了。
我再仔细看了下这个返回,发现并没有那么简单,我没找到农历的日期:
{
"y": [
"祭祀",
"塑绘",
"开光",
"裁衣",
"冠笄",
"嫁娶",
"纳采",
"拆卸",
"修造",
"动土",
"竖柱",
"上梁",
"安床",
"移徙",
"入宅",
"安香",
"结网",
"捕捉",
"畋猎",
"伐木",
"进人口",
"放水" ],
"j": [
"出行",
"安葬",
"修坟",
"开市" ],
"ts": "占房床房内北",
"c": "冲猪",
"s": "煞东",
"zc": "丁亥",
"zh": "执",
"yq": "五富 益後",
"yj": "劫煞 小耗 复日 重日 元武" }
我收起了天真,又开始寻找获取农历日期的方法。我没有找到获取农历日期的请求,但是我发现了一个特别的请求:
这个请求并没有直接获取农历日期,而是用 JavaScript 计算的,我们可以看到这个请求里面的计算方法:
//====================================== 算出农历, 传入日期控件, 返回农历日期控件 // 该控件属性有 .year .month .day .isLeap //sDObj = new Date(y,m,i+1); 当月1日日期 function Lunar(objDate) {
var i, leap = 0, temp = 0;
var offset = (Date.UTC(objDate.getFullYear(), objDate.getMonth(), objDate
.getDate()) - Date.UTC(1900, 0, 31)) / 86400000;
for (i = 1900; i < 2100 && offset > 0; i++) {
temp = lYearDays(i);
offset -= temp;
}
if (offset < 0) {
offset += temp;
i--;
}
this.year = i;
leap = leapMonth(i); //闰哪个月 this.isLeap = false;
for (i = 1; i < 13 && offset > 0; i++) {
//闰月 if (leap > 0 && i == (leap + 1) && this.isLeap == false) {
--i;
this.isLeap = true;
temp = leapDays(this.year);
} else {
temp = monthDays(this.year, i);
}
//解除闰月 if (this.isLeap == true && i == (leap + 1)) {
this.isLeap = false;
}
offset -= temp;
}
if (offset == 0 && leap > 0 && i == leap + 1) {
if (this.isLeap) {
this.isLeap = false;
} else {
this.isLeap = true;
--i;
}
}
if (offset < 0) {
offset += temp;
--i;
}
this.month = i;
this.day = offset + 1;
}
当然,这个 js 文件里面还有好多其他诸如计算星期、节假日之类的方法,我们可以把这个 js 里面的方法实现用 python 来实现就可以计算出农历日期以及节假日之类的信息了。但是我的时间比较紧迫,所以我选择用最简单的办法——百度。将度娘里面别人写的方法直接拿来用,就不用重复造轮子了。
搜索可以发现好多计算万年历信息的方法,我从其中选取了一个作为工具类来用。
前奏已经弄完了,接下来就好办了,我们来看看代码怎么写。
第一步是获取某一年每天的信息:
def get_data(year): url = 'https://staticwnl.tianqistatic.com/Public/Home/js/api/yjs/%d.js' % year
response = requests.get(url)
text = response.text
start_str = 'lmanac["%d"] =' % year
his_end_str = ';if(typeof(lmanac_2345)!="undefined"){lmanac_2345();}' cur_end_str = ';if(typeof(lmanac_2345)!="undefined"){lmanac_2345()};' cur_year = datetime.datetime.now().year
jsonstr = text.replace(start_str, '')
if cur_year == year:
jsonstr = jsonstr.replace(cur_end_str, '')
else:
jsonstr = jsonstr.replace(his_end_str, '')
return jsonstr
这里需要注意的是,获取到的结果数据在 JSON 数据的前后都加了字符串干扰信息,我们需要将这些字符串给去掉才能解析 JSON。
你以为这样就完了吗?是不是发现用解析2021年的数据的方法去2020年的数据不行?你没看错,这里网站开发人员开了一个小玩笑,他们把结尾字符串里面的一个分号换了个位置。据我仔细观察发现,当年的返回结果中这个分号是在最后的,而其他年份的返回数据中这个分号是在大括号里面的。
获取到数据之后,我们就来计算日期:
def choose_day(year, jsonstr):
jobj = json.loads(jsonstr)
for day in jobj.keys():
y = jobj[day]['y']
if '嫁娶' in y:
dtime = datetime.datetime(year, int(day[1:3]), int(day[3:5]))
# 获取农历日期
ludar_date = lunarUtils.get_ludar_date(dtime)
# 取得日,然后看是否是双数
if ludar_date[2] % 2 == 0:
print('公历日期:%s,农历日期:%s' % (day, ludar_date))
这里面就相对比较简单了,先解析返回的 JSON 数据,然后遍历日期,获取每天的信息,看哪天宜“嫁娶”,就再获取这天的农历日期,看是不是双数,如果是的话,这就是我们的目标日期。
我最后获得的日期是这样子的:
公历日期:d0107,农历日期:(2020, 11, 24) 公历日期:d0122,农历日期:(2020, 12, 10) 公历日期:d0124,农历日期:(2020, 12, 12) 公历日期:d0126,农历日期:(2020, 12, 14) 公历日期:d0203,农历日期:(2020, 12, 22) 公历日期:d0209,农历日期:(2020, 12, 28) 公历日期:d0225,农历日期:(2021, 1, 14) 公历日期:d0305,农历日期:(2021, 1, 22) 公历日期:d0311,农历日期:(2021, 1, 28) 公历日期:d0318,农历日期:(2021, 2, 6) 公历日期:d0324,农历日期:(2021, 2, 12) 公历日期:d0401,农历日期:(2021, 2, 20) 公历日期:d0419,农历日期:(2021, 3, 8) 公历日期:d0425,农历日期:(2021, 3, 14) 公历日期:d0507,农历日期:(2021, 3, 26) 公历日期:d0513,农历日期:(2021, 4, 2) 公历日期:d0525,农历日期:(2021, 4, 14) 公历日期:d0531,农历日期:(2021, 4, 20) 公历日期:d0606,农历日期:(2021, 4, 26) 公历日期:d0613,农历日期:(2021, 5, 4) 公历日期:d0617,农历日期:(2021, 5, 8) 公历日期:d0619,农历日期:(2021, 5, 10) 公历日期:d0625,农历日期:(2021, 5, 16) 公历日期:d0701,农历日期:(2021, 5, 22) 公历日期:d0711,农历日期:(2021, 6, 2) 公历日期:d0713,农历日期:(2021, 6, 4) 公历日期:d0717,农历日期:(2021, 6, 8) 公历日期:d0723,农历日期:(2021, 6, 14) 公历日期:d0725,农历日期:(2021, 6, 16) 公历日期:d0729,农历日期:(2021, 6, 20) 公历日期:d0804,农历日期:(2021, 6, 26) 公历日期:d0811,农历日期:(2021, 7, 4) 公历日期:d0813,农历日期:(2021, 7, 6) 公历日期:d0815,农历日期:(2021, 7, 8) 公历日期:d0823,农历日期:(2021, 7, 16) 公历日期:d0827,农历日期:(2021, 7, 20) 公历日期:d0914,农历日期:(2021, 8, 8) 公历日期:d0926,农历日期:(2021, 8, 20) 公历日期:d1013,农历日期:(2021, 9, 8) 公历日期:d1015,农历日期:(2021, 9, 10) 公历日期:d1025,农历日期:(2021, 9, 20) 公历日期:d1029,农历日期:(2021, 9, 24) 公历日期:d1106,农历日期:(2021, 10, 2) 公历日期:d1110,农历日期:(2021, 10, 6) 公历日期:d1112,农历日期:(2021, 10, 8) 公历日期:d1116,农历日期:(2021, 10, 12) 公历日期:d1124,农历日期:(2021, 10, 20) 公历日期:d1130,农历日期:(2021, 10, 26) 公历日期:d1207,农历日期:(2021, 11, 4) 公历日期:d1211,农历日期:(2021, 11, 8) 公历日期:d1219,农历日期:(2021, 11, 16) 公历日期:d1223,农历日期:(2021, 11, 20) 公历日期:d1231,农历日期:(2021, 11, 28)
看了一下,今天就是个好日子,公历是0126,农历是1214,12+14=26,我觉得挺好,可惜今天错过了,只能推后了。我看了一下,要赶在年前领证的话,只有两个日期可选了,我想选2月3日,这天正好立春,是个好日子。
Python 应用无处不在,只要善于运用,我们的生活会更高效美好!我马上要领证了,大家可否点个赞祝福一下?
数据分析咨询请扫描二维码
若不方便扫码,搜微信号:CDAshujufenxi
在互联网产品运营、用户增长的实战场景中,很多从业者都会陷入一个误区:盲目投入资源做推广、拉新,却忽视了“拉新后的用户激活 ...
2026-02-06在机器学习建模过程中,特征选择是决定模型性能的关键环节——面对动辄几十、上百个特征的数据(如用户画像的几十项维度、企业经 ...
2026-02-06在CDA(Certified Data Analyst)数据分析师的日常实操中,表格结构数据是贯穿全流程的核心载体,而对表格数据类型的精准识别、 ...
2026-02-06在日常办公数据分析中,我们经常会面对杂乱无章的批量数据——比如员工月度绩效、产品销售数据、客户消费金额、月度运营指标等。 ...
2026-02-05在分类模型(如风控反欺诈、医疗疾病诊断、客户流失预警)的实操落地中,ROC曲线是评估模型区分能力的核心工具,而阈值则是连接 ...
2026-02-05对CDA(Certified Data Analyst)数据分析师而言,数据分析的价值不仅在于挖掘数据背后的规律与洞察,更在于通过专业的报告呈现 ...
2026-02-05在数据分析实战中,我们经常会遇到“多指标冗余”的问题——比如分析企业经营状况时,需同时关注营收、利润、负债率、周转率等十 ...
2026-02-04在数据分析场景中,基准比是衡量指标表现、评估业务成效、对比个体/群体差异的核心工具,广泛应用于绩效评估、业务监控、竞品对 ...
2026-02-04业务数据分析是企业日常运营的核心支撑,其核心价值在于将零散的业务数据转化为可落地的业务洞察,破解运营痛点、优化业务流程、 ...
2026-02-04在信贷业务中,违约率是衡量信贷资产质量、把控信用风险、制定风控策略的核心指标,其统计分布特征直接决定了风险定价的合理性、 ...
2026-02-03在数字化业务迭代中,AB测试已成为验证产品优化、策略调整、运营活动效果的核心工具。但多数业务场景中,单纯的“AB组差异对比” ...
2026-02-03企业战略决策的科学性,决定了其长远发展的格局与竞争力。战略分析方法作为一套系统化、专业化的思维工具,为企业研判行业趋势、 ...
2026-02-03在统计调查与数据分析中,抽样方法分为简单随机抽样与复杂抽样两大类。简单随机抽样因样本均匀、计算简便,是基础的抽样方式,但 ...
2026-02-02在数据驱动企业发展的今天,“数据分析”已成为企业经营决策的核心支撑,但实践中,战略数据分析与业务数据分析两个概念常被混淆 ...
2026-02-02在数据驱动企业发展的今天,“数据分析”已成为企业经营决策的核心支撑,但实践中,战略数据分析与业务数据分析两个概念常被混淆 ...
2026-02-02B+树作为数据库索引的核心数据结构,其高效的查询、插入、删除性能,离不开节点间指针的合理设计。在日常学习和数据库开发中,很 ...
2026-01-30在数据库开发中,UUID(通用唯一识别码)是生成唯一主键、唯一标识的常用方式,其标准格式包含4个短横线(如550e8400-e29b-41d4- ...
2026-01-30商业数据分析的价值落地,离不开标准化、系统化的总体流程作为支撑;而CDA(Certified Data Analyst)数据分析师,作为经过系统 ...
2026-01-30在数据分析、质量控制、科研实验等场景中,数据波动性(离散程度)的精准衡量是判断数据可靠性、稳定性的核心环节。标准差(Stan ...
2026-01-29在数据分析、质量检测、科研实验等领域,判断数据间是否存在本质差异是核心需求,而t检验、F检验是实现这一目标的经典统计方法。 ...
2026-01-29