Python 读取 支付宝账单并存储到 Access 中
我有一个很多年前自己写的C#+Access的记账程序,用了很多年,现在花钱的机会多了,并且大部分走的支付宝,于是就想把账单从支付宝网站上下载下来,直接写入到Access,这样就很省心了。
记账程序是长这个样子的:

还有报表汇总模块儿:

Access 主要表结构如下:

支付宝支付流水下载下来如下:

具体代码如下:
table="收支记录表"
path="C:\\Users\\user\\Desktop\\tmp\\alipay_record_20190114.xlsx" #test_connect()
read_excel(path)
#-*-coding:utf-8 -*-
import pypyodbc
import xlrd
from ClassDef import *
import CommonUtil
import traceback

def read_excel(path):
workbook=xlrd.open_workbook(path)
sheet=workbook.sheet_by_index(0)
nrows=sheet.nrows
ncols=sheet.ncols
print(nrows, ncols) j=0 for i in range(1, nrows):
# pay_time=sheet.row_values(i,0)#付款时间
# print(pay_time)
#print(sheet.cell(i, 0).value) # 指定行和列获取数据 goodsName = sheet.cell(i, 5).value #商品名称
inOrOut = sheet.cell(i, 7).value ## 收、支
status = sheet.cell(i, 8).value ## 交易状态
# if (reason.find("收益发放") > -1):
# continue
if inOrOut is not None and len(inOrOut.strip()) >0 and goodsName.find("收益发放") == -1\
and goodsName.find("赚钱红包推荐奖励")==-1\
and (status.find("交易成功")>-1 or status.find("等待确认收货")>-1):
#print(i,inOrOut)
ioRecord = InOutRecord() ioRecord.reason=goodsName################################## reason
ioRecord.inOrOut=inOrOut[0:1]########################### 收/支 payTimeStr = sheet.cell(i, 0).value
if payTimeStr is not None :
# print (i, payTimeStr)
payTimeStr = payTimeStr[0:8]
dateStr = payTimeStr[0:4]
monthStr = payTimeStr[4:6]
dayStr = payTimeStr[6:8]
# value = datetime.datetime(payTimeStr) XXXX
# print(payTimeStr)
# print(dateStr)
# print(monthStr)
# print(dayStr) #自动生成编号:先查当天有几条记录了
filterDate = dateStr[2:4]+monthStr+dayStr
numberStr,connStr=genAccessKeyAndTradeDate(filterDate, dateStr, monthStr, dayStr)
# print("编号===" + numberStr)
if len(numberStr) > 0:
ioRecord.numberStr = numberStr ######################### 编号
else:
print("eeeeeeeeeeeeeeeeeeeeeeee " + "编号自动生成错误" + "eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee") ioRecord.payDate=connStr################################### 日期
##########################################################################
amount = sheet.cell(i, 6).value#金额
ioRecord.amount=amount
# print("金额==="+str(amount)) #交易来源地
tradeSource = sheet.cell(i, 3).value # 交易来源地 #获取交易对方,来确认收支类型
tradeObject = sheet.cell(i, 4).value # 交易对方 #获取收支具体类型,如衣、食、住、用、行等
ioRecord.type=getType(tradeObject)
#修正 type
if ioRecord.reason is not None:
if ioRecord.reason.find('打车')>-1:
ioRecord.type='行'
#完善
ioRecord.reason=ioRecord.reason.strip()+"【"+tradeObject.strip()+"】"+"【"+tradeSource.strip()+"】"
# if len(ioRecord.reason)>15:#商品名称只取前15个字符
# ioRecord.reason=ioRecord.reason[0:15]
ioRecord.comment=tradeObject.strip() ret=insertRecord(ioRecord)#存储
#ret=insertTest();
#ret=test_insertRecord(ioRecord)
if ret==1:
print(str(j + 1) + "--" + str(
i) + " " + ioRecord.numberStr + " " + ioRecord.payDate + " " + ioRecord.inOrOut + " "
+ str(ioRecord.amount) + " " + ioRecord.type + " " + ioRecord.reason + "---存储成功") print("==============================================================================================") j=j+1
#记录当前已经记了几条账目了
def getRecordCountByDate(date):
conn,cur=connect_access()
#sql="SELECT * FROM " + table+" where 编号 like"+date+""
sql = "select * from 收支记录表 where 编号 like "+"'"+"%"+date+"%"+"'"
#sql = "select * from 收支记录表 where 编号 = "+"'"+date+"'"
#sql = "select * from 收支记录表 where 编号 like " + date
#print(sql)
cur.execute(sql)
#cur.execute(sql,date)
alldata = cur.fetchall()
total_rows = len(alldata)
# total_cols = len(alldata[0])
# print("总行数 = %d" % ( total_rows))
return total_rows
#根据日期自动生成编号,并生成交易日期
def genAccessKeyAndTradeDate(filterDate, dateStr, monthStr, dayStr):
recordCount = getRecordCountByDate(filterDate)
print("filterDate==="+filterDate)
print("recordCount==="+str(recordCount))
numberStr = ""
if recordCount == 0:
numberStr = filterDate + ""
elif recordCount > 0 and recordCount < 9:
numberStr = filterDate + "" + str(recordCount + 1)
elif recordCount >= 9:
numberStr = filterDate + str(recordCount + 1) connStr = dateStr + "/" + monthStr + "/" + dayStr return numberStr, connStr
def getType(tradeObject):
type=""
if tradeObject is not None:
if tradeObject.find('超市') > -1 or tradeObject.find('汇悦') > -1 or tradeObject.find(
'大王') > -1 or tradeObject.find('便利') > -1: \
type = '食' if tradeObject.find('商品') > -1 or tradeObject.find('蔬菜') > -1 or tradeObject.find('奶吧') > -1:
type = "食" if tradeObject.find('条码支付-总部') or tradeObject.find('食品') > -1 or tradeObject.find(
'餐吧') > -1 or tradeObject.find('切面') > -1:
type = "食" if tradeObject.find('德青源') > -1 or tradeObject.find('食') > -1:
type = "食" if tradeObject.find('童') > -1 or tradeObject.find('母') > -1:
type = '宝宝' if tradeObject.find('移动') > -1:
type = "通信费用" if tradeObject.find('地铁') > -1:
type = "行" if tradeObject.find('市妇幼') > -1 or tradeObject.find('御和堂') > -1 or tradeObject.find('医院') > -1:
type = '医疗健康' if tradeObject.find('七匹狼') > -1 or tradeObject.find('南极人') > -1:
type = '衣' if tradeObject.find('图书') > -1 or tradeObject.find('当当') > -1:
type = '自我完善' if tradeObject.find('供电') > -1:
type = '电费' return type
#存储一条数据
def insertRecord(ioRecord):
if (ioRecord is not None):
#printItemAllInfo(ioRecord)
conn, cursor = connect_access()
#sql = "insert into 收支记录表 (编号, 日期, 收支原因, 收支, 金额, 类型) values( %s, %s, %s, %s, %s, %s)"
# sql = "insert into 收支记录表 (编号, 日期, 收支原因, 收支, 金额, 类型) " \
# "values(%s, %s, %s, %s, %s, %s)"
createDate = CommonUtil.getCurrentDateStr()
sql = "insert into 收支记录表 (编号, 日期, 收支原因, 收支, 金额, 类型,备注) values("\
+ "'"+ str(ioRecord.numberStr) + "'" +","\
+ "'" + str(ioRecord.payDate) + "'" + ','\
+ "'" + str(ioRecord.reason) + "'" + ','\
+ "'" + str(ioRecord.inOrOut) + "'" + ','\
+ "'" + str(ioRecord.amount) + "'" + ','\
+ "'" + str(ioRecord.type) + "'"+ ',' \
+ "'" + str(ioRecord.comment+"_"+createDate) + "'" + ')' #sql = 'insert into 收支记录表 (编号, 日期, 收支原因, 收支, 金额, 类型) values('+str(ioRecord.numberStr)+','
# +ioRecord.payDate+','+ioRecord.reason+','+ioRecord.inOrOut+','+float(ioRecord.amount)+','+ioRecord.type+');' # print("=======================================================================")
# print(dataRow[16], dataRow[17], dataRow[18])
#print(sql)
#print(ioRecord.numberStr,ioRecord.payDate,ioRecord.reason,ioRecord.inOrOut,ioRecord.amount,ioRecord.type) #print(createDate)
try:
# cursor.execute(sql, (
# str(ioRecord.numberStr),str(ioRecord.payDate),str(ioRecord.reason),
# str(ioRecord.inOrOut),str(ioRecord.amount),str(ioRecord.type)))
cursor.execute(sql)
conn.commit()
cursor.close()
conn.close()
return 1
except Exception as e:
traceback.print_exc()
print("insertOnRecord() 出错,具体记录内容如下XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX")
printItemAllInfo(ioRecord)
cursor.close()
conn.close()
return -1
#获取数据库连接
def connect_access():
mdb = 'Driver={Microsoft Access Driver (*.mdb,*.accdb)};' \
'DBQ=D:\\个人\\我的软件\\PersonalAsistance_2012_04_13\\PersonalAsistance\\PersonalAsistance\\bin\\Debug\\PADB.mdb'
conn = pypyodbc.win_connect_mdb(mdb)
cur = conn.cursor()
return conn,cur
总结:
这里面有几个关键点:
1、读Excel(这个不算,比较淘常见)
2、生成Access表的主键,这个主键的规则是这样的:取年的最后两位,月的两位,日的两位,然后从01开始往后编码,有几条编到几;
3、连接Access,读、写Access;
python 代码已经放出来了,记账程序有需要的可QQ我:861712499
Python 读取 支付宝账单并存储到 Access 中的更多相关文章
- Python读取文件内容与存储
Python读取与存储文件内容 一..csv文件 读取: import pandas as pd souce_data = pd.read_csv(File_Path) 其中File_path是文件的 ...
- python 读取SQLServer数据插入到MongoDB数据库中
# -*- coding: utf-8 -*-import pyodbcimport osimport csvimport pymongofrom pymongo import ASCENDING, ...
- Python读取PE文件(exe/dll)中的时间戳
代码原文地址: https://www.snip2code.com/Snippet/144008/Read-the-PE-Timestamp-from-a-Windows-Exe https://gi ...
- Python科学计算结果的存储与读取
Python科学计算结果的存储与读取 总结于2019年3月17日 荆楚理工学院 计算机工程学院 一.前言 显然,作为一名工科僧,执行科学计算,需用Python.PS:快忘记Matlab吧.我用了二十 ...
- python读取caffemodel文件
caffemodel是二进制的protobuf文件,利用protobuf的python接口可以读取它,解析出需要的内容 不少算法都是用预训练模型在自己数据上微调,即加载"caffemodel ...
- python读取xml文件
关于python读取xml文章很多,但大多文章都是贴一个xml文件,然后再贴个处理文件的代码.这样并不利于初学者的学习,希望这篇文章可以更通俗易懂的教如何使用python 来读取xml 文件. 什么是 ...
- Python读取SQLite文件数据
近日在做项目时,意外听说有一种SQLite的数据库,相比自己之前使用的SQL Service甚是轻便,在对数据完整性.并发性要求不高的场景下可以尝试! 1.SQLite简介: SQLite是一个进程内 ...
- python读取与写入csv,txt格式文件
python读取与写入csv,txt格式文件 在数据分析中经常需要从csv格式的文件中存取数据以及将数据写书到csv文件中.将csv文件中的数据直接读取为dict类型和DataFrame是非常方便也很 ...
- python 读取文件时报错UnicodeDecodeError: 'gbk' codec can't decode byte 0x80 in position 205: illegal multib
python 读取文件时报错UnicodeDecodeError: 'gbk' codec can't decode byte 0x80 in position 205: illegal multib ...
随机推荐
- 黑色半透明镂空遮罩指引效果实现jQuery小插件
/*! * by zhangxinxu(.com) 2017-05-18 * 新版上线时候的黑色半透明镂空遮罩指引效果实现jQuery小插件 * 兼容到IE8+ * MIT使用协议,使用时候保留版权 ...
- TopCoder SRM502 Div1 500 贪心 01背包
原文链接https://www.cnblogs.com/zhouzhendong/p/SRM502-500.html SRM502 Div1 500 好题. 首先,如果已经确定了解决所有问题的优先级, ...
- UOJ#62. 【UR #5】怎样跑得更快 数论 莫比乌斯反演
原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ62.html 题解 太久没更博客了,该拯救我的博客了. $$\sum_{1\leq j \leq n} \ ...
- NOIP2017提高组Day1T3 逛公园 洛谷P3953 Tarjan 强连通缩点 SPFA 动态规划 最短路 拓扑序
原文链接https://www.cnblogs.com/zhouzhendong/p/9258043.html 题目传送门 - 洛谷P3953 题目传送门 - Vijos P2030 题意 给定一个有 ...
- Quratz的理解
一:公共术语 1.为什么使用Qurztz 在某一个有规律的时间点干某件事.并且时间的触发的条件可以非常复杂(比如每月最后一个工作日的17:50),复杂到需要一个专门的框架来干这个事. Quartz就是 ...
- 无向图的 DFS 和 BFS实现 (以邻接表存储的图)
#include <iostream> #include <queue> using namespace std; #define MaxVertexNum 10 typede ...
- 我今天遇到的条件语句Integer类型的
两个Integer类型的值进行比较时,应该用equals进行判断,用"=="判断是错误的,后来想了一下就明白了,Integer毕竟是对象, 而不是int基本数据类型,可以直接比较, ...
- 【JavaScript】jQuery
No1: jQuery能帮我们干这些事情: 消除浏览器差异:你不需要自己写冗长的代码来针对不同的浏览器来绑定事件,编写AJAX等代码: 简洁的操作DOM的方法:写$('#test')肯定比docume ...
- Gym 102091K The Stream of Corning 2【线段树】
<题目链接> 题目大意: 进行两种操作:1.给定一个数的出现时间.价值.消失时间: 2.进行一次询问,问你当前时间,第K大的数的价值. 解题分析: 采用离线集中处理,将每个数的出现时间和它 ...
- poj 3067 Japan 【树状数组】
<题目链接> 题目大意: 有两个点集,这两个点集从上至下分别从1~n,1~m编号,现在给出n组数据,(x,y),表示左边点集编号为x的点与右边点集编号为y的点相连,现在要求计算这些线段的交 ...