安卓怎么安装365BET-h365官方登录平台-365batapp

简约 · 精致 · 专注内容

金税盘、税控盘、税务UKey快速批量抄税清卡的一种方法分享

金税盘、税控盘、税务UKey快速批量抄税清卡的一种方法分享

大家好,又见面了,我是你们的朋友全栈君。

1. 引言 记得那是去年的春天,一位高中时代的老同学,也是相知相交多年的挚友,受某著名机构邀请,来到我生活的城市参加财税论坛峰会,并做学术报告。当时正值清明前后,正所谓是“春风吹柳万条斜” 的美好时节。活动结束之后,我带老友去这个时节,我们这个城市人气指数NO1的景点欣赏形态各异,五颜六色的郁金香,然后泛舟名湖,参观千年古寺,老友对这座城市的湖光山色和文化底蕴赞叹不已。晚上,我找了一个比较幽静的酒馆,准备一醉方休,为老友饯行。

酒过三巡,菜过五味,我们相互交流分享着踏入社会后这么多年,各自所经历的人、事、物,可谓“无可奈何花落去,似曾相识燕归来”,感叹不已!后来话题转移到各自的工作上来,由于我在IT行业混迹多年。老友说,现在有六七百家的客户委托他公司做账,有四五百个税盘(有白色的金税盘,黑色的税控盘和税务UKey)在公司里托管。每到月初心里有着莫名的恐惧感。他说每月月初都要将这些盘进行抄税和清卡,工作量极大。最开始的时候采用全手工操作,找到众多的盘相应的税盘,插到电脑上,然后打开相应的开票软件,进行抄税和清卡,顺利的话一个盘需要六七分钟的时间,如果遇到特殊的情况一个盘需要二十甚至三十多分钟才能做完,这样每个月都要投入七八天的时间,做这些机械化的,让人眼花缭乱的,又不得不做的操作,真是痛苦烦恼不已啊!

他为了能结束这种煎熬的过程,经过多方咨询和比较后,从一家知名的公司采购了税盘机柜和配套使用的软件,第一年投入十来万的费用,后续每年还要缴纳一定费用的维护费,以期解决这种只有经历过才知道的痛苦煎熬。此时,我问老友说:“现在你们有了这批硬件设备和配套的软件,应该轻松很多了?”不料,老友此时眉头紧皱,长叹一声,说道:“唉!远远未达我的预期,现在依然在煎熬中,也算花了冤枉钱。”我便问道:“此话怎讲呢?”。老友说道:“我感觉采购的硬件还算可以,配套使用的软件很不稳定,各种报错,卡死。现在每个月还要花费四五天的时间来处理这个事情。你懂软件,不知你能否抽时间到我那看看软件是否有改进的空间。”看到老友这么痛苦不堪的样子,我便答应了下来。

翌日一早,我便随老友一起到他的公司,分析问题出在哪里。老友的盛情款待不再言表,带着老友的嘱托和殷切的期盼,我通宵到旦地对配套软件进行跟踪分析,终于发现了其中的秘密。原来这款软件其实就是一个模拟器(专业术语叫RPA技术),通过自动打开相应的开票软件(会将票软件的界面隐藏),模拟鼠标键盘操作以实现抄税和清卡的功能,这款软件对各种场景的处理不太尽如人意,所以效率和稳定性都太差。

老友眼神流露出殷切的期盼,问我道:“是否有改进空间?”,其实,我心里此时也没底,便说道:“让我试试吧,给我一两个月的时间。”

带着老友的期待和嘱托,我回去后通过两个月的日夜兼程的努力,终于写了一款工具出来。通过给老友使用,该工具运行比较稳定,四五百个盘不需要做任何的人工干预,花费六七个小时的时间即可实现抄税和清卡。老友看到这个结果兴奋不已,感激的对我说:“你真的解决了我的难言之隐,辛苦你了,说实话不能让你白忙一场。这样把我按一个月X万的成本支付你,给你X万。”我哈哈一笑说:“你怎这么客气呢,要不这样把,你买一瓶好酒,我们一起喝酒就行了。”

于是老友买了几瓶享誉海内外的某酒,喊上三五好友作陪,开怀畅饮,一切不再言表。翌日,老友知道我爱喝酒,便买了一箱该酒偷偷放到我后备箱内。我发现后,再三推辞,但盛情难却,象征行的收了两瓶。直到今日,老友的规模也在不断扩大,工具依然在稳定的运行。

最近笔者稍得闲暇,将税盘批量抄税和清卡的相关经验和核心的代码分享给各位读者,希望能对有类似经历的朋友起到帮助作用。笔者同时也非常愿意和各位有类似业务场景需求,或者遇到技术瓶颈的,各界朋友交流业务或技术经验。

2. 税盘抄税和清卡业务流程 增值税开票软件迄今为止都需要税盘,税盘分为金税盘(白盘),税控盘(黑盘)和税务UKey三种(如下图所示,当然还有全电,不再本文讨论的范畴)。其中金税盘是航信旗下研发的开票软件,税控盘和税务UKey是百旺研发的开票软件。从市场占有率来说金税盘属于佼佼者,从趋势来看当属税务UKey首屈一指。

开票软件需要每月月初进行抄税,将当前开票软件中,上月所开的发票数据进行汇总上传(注意:在抄税之前,需要从税盘中修复所有的发票,并将发票全部上传)。在开票软件中,进行抄税之后,税务会计能登录到电子税务局,进行纳税申报(这一操作也称为“报税”)。在电子税务局进行纳税申报之后,方可在开票软件中,进行清卡操作,清卡成功后,税盘的锁死日期自动变更为次月。若未清卡,到税盘锁死日期后,不能再开具发票。抄税和清卡的操作流程如下图所示。

图-4 抄税和清卡流程

3. 常见的税盘抄税和清卡操作方式 税盘的抄税和清卡操作方式,根据笔者的理解和调研大致分为如下四种。第一种,在开票软件中手工操作进行抄税和清卡;第二种,使用RPA技术,模拟鼠标键盘操作进行抄税和清卡;第三种,通过注入技术进行税盘抄税和清卡进行操作;第四种,通过相关组件提供的抄税和清卡接口进行抄税和清卡进行操作。笔者,下面分别就这四种操作方式进行详细的阐述和对比。

图-5 抄税和清卡操作的几种方式

3.1 手工操作进行抄税和清卡 在开票软件里进行手工的抄税和清卡,是最便捷、最稳定的方式。这对于只有一个或数个盘的单个企业而言,是行之有效的操作手段。

这种操作,首先,找到税盘(金税盘,或者税控盘,或者税务UKey)插到电脑上,打开对应的开票软件(不同类型的税盘需要不同的开票软件)输入密码(包括软件密码,税控设备密码,证书密码)登录到开票软件。

然后,修复发票数据(因为有可能存在同一税盘在不同的电脑上开过票),以确保当月的所有的开票数据在当前开票软中已存在,如果这一点已确保,可以忽略修复发票。修复完发票数据后,需要确认是否存在未上传的发票数据,如果存在未上传的发票数据则要进行上传发票。

经过以上的准备和前奏操作之后,则进行抄税(在有的开票软件中或称之为“汇总上传”,不同的开票软件其入口或者叫法不同,笔者在此不再一一展开赘述)。抄税成功之后,则需要税务会计登录到电子税务局,进行纳税申报(亦称之为“申报”或“报税”)。

最后,在税务会计纳税申报完成之后,开票员插入税盘,打开并登录开票软件进行清卡(在有的开票软件中也称之为 “监控回传”或“监控回写”,不同的开票软件其操作入口也不同,笔者也不再展开赘述)。清卡完成后,则税盘的锁死日期自动变为次月的日期,若清卡失败,则到税盘锁死日期(一般是当月中旬,当然具体的日期可以查看税局的纳税申报截止日期)不再允许开票。

上面描述的则是手工进行抄税,清卡的完整的操作流程,这种方式的优点是稳定可控。但是,如果企业有数百甚至更多的盘(例如代账企业),其工作量也是可想而知的,正如我在引言中所描述的,老友的痛苦一样,在此笔者不再回忆老友的梦魇和痛苦。

3.2 通过RPA技术进行抄报和清卡 RPA技术(Robotic Process Automation)即机器人流程自动化,该技术用在税盘的抄税和清卡中,也即自动打开税盘对应的开票软件,通过模拟键盘鼠标操作,进行抄税和清卡。操作的流程和手工操作完全一致,首先登录开票软件,然后修复发票,判断发票是否上传,如果存在未上传的发票则上传,最后进行抄税和清卡操作。

这种方式的优点是,相对于纯手工操作来说会提升一定的效率,但效率提升是有限的。缺点是,开票软件种类繁多,和各种单步操作的各种业务场景(例如多报错信息的处理等)叠加在一起,有数不尽的细分场景,很难一一枚举,这也就导致RPA很不稳定。在加上开票软件本身会在使用各种反模拟操作的技术手段。开票软件升级后RPA也要做相关的调整和迭代,开票软件升级频繁(至少每月一个版本)。这些因素决定了,RPA技术很难在税盘的抄税,清卡的场景能做到稳定,同时提升的效率也是有限的。正如在引言中所描述的一样,我老友花“巨资”引入的产品,依然令他痛苦不堪,正所谓“花钱买了寂寞”。

3.3 通过注入技术进行抄税和清卡 通过注入技术进行税盘的抄税,清卡操作,相对于RPA技术来说更加考验开发着的技术功底,正所谓“没有金刚钻,别拦瓷器活”,注入需要找到正确的注入点,并进行相关的分析,这个过程没有一定技术底蕴很难胜任。当然注入也可以分为初级和高级两个境界,能做到高级注入的话,可以直接调用相关的底层接口,可以做到很稳定和很高效。然而,高级注入没有“大师”级的技术底蕴,很难胜任。

高级注入虽然可以做大稳定和高效,但是开票软件频繁升级,也是梦魇一般的存在,每次升级后,都需要进行重新的分析,这个过程也是极其耗时和繁杂的,所以这种技术也有很大的瓶颈。

3.4 通过组件接口进行抄税和清卡 相关的组件接口提供了抄税,清卡以及其它功能,使用组件接口可以获取税盘状态(例如各核定票种的最近抄报日期,锁死日期等),并依此来判断税盘的抄税,清卡状态。如果没有抄税,则可以调用相关接口进行未上传发票上传,抄税,清卡。组件接口的优点是稳定,高效,同时不用再考虑开票软件的升级问题,一劳永逸。另外组件接口调用简单,很容易上手。

综上所述,组件接口是最优方案。当然组件所提供的接口不止抄税和清卡的功能,例如开票,作废,冲红,库存查询,领购,上传,发票查询,版式文件下载等一应具全。本文着重介绍抄税和清卡。

4. 组件接口实现抄税清卡的核心代码分享 下面笔者对通过调用组件接口实现抄税和清卡的核心代码进行分享,上节提到的通过RPA技术和注入技术实现抄税,清卡的技术细节,有兴趣的朋友可以和笔者进行沟通交流。

4.1 金税盘组件接口4.1.1 抄税操作核心代码代码语言:javascript复制{**********************************************************

功能:金税盘抄税操作

参数:errMsg 输出参数,错误信息

返回值:ture - 成功

false - 失败

date: 2021-05-0430

author: 海之边 qq-3094353627

流程简述:

1. 打开金税盘

2. 获取金税盘盘状态

3. 如果尚未抄税则进行抄税

4. 抄税成功后获取金税盘状态,并缓存到本地数据库

**********************************************************}

function TCtrlItem.reportTax(var errMsg: string): boolean;

var idx: integer;

bReported: boolean;

sFplxdm, sCardClock: string;

oJsCardComponent: TJsCardComponent;

oTaxCardRespOpenCard: TJsCardResponse_OpenCard;

oJsCardRespGetClock: TJsCardResponse_getClock;

oJsCardState: TJsCardState;

oJsCardRespRep: TJsCardResponse_taxReport;

oHdFplxdmLst: TStrings;

oDBAJxsb: TDBAJxsb;

oDBAPzxx: TDBAPzxx;

begin

result := false;

oJsCardComponent := TJsCardComponent.Create;

oTaxCardRespOpenCard := nil;

oJsCardState := nil;

oJsCardRespRep := nil;

oHdFplxdmLst := nil;

oDBAJxsb := nil;

oDBAPzxx := nil;

try

oDBAJxsb := TDBAJssb.Create;

oDBAPzxx := TDBAPzxx.Create;

//1. 打开金税盘

oTaxCardRespOpenCard := oJsCardComponent.openTaxCard(Zsmm);

if not oTaxCardRespOpenCard.isSucc then

begin

errMsg := Format('打开金税盘%s.%s失败: %s。', [FNsrsbh, FMachineNo, errMsg]);

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

exit;

end;

if oTaxCardRespOpenCard.isCertNotPass then

begin

//证书密码错误,设置密码校验不通过状态

Zsmmjyjg := untGolbalConst.JYJG_NotPass;

updateToDb(oDBAJxsb);

Exit;

end;

//判断抄报状态

//2. 读取金税盘时钟

oJsCardRespGetClock := OJsCardComponent.getClock;

if not oJsCardRespGetClock.isSucc then

begin

errMsg := Format('读取金税盘时钟失败:%d-%s', [oJsCardRespGetClock.retCode,

oJsCardRespGetClock.retMsg]);

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end;

sCardClock := oJsCardRespGetClock.Clock;

//3 读取金税盘状态

oJsCardState := oJsCardComponent.queryTaxcardState_svr(errMsg);

if not Assigned(oJsCardState) then

begin

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end;

updateToDb(oDBAJxsb);

//更新设备状态信息

assignFromJsCardState(oJsCardState);

//保存票种核定信息

oJsCardState.Pzxxs.saveToDb(Sksblx, SksbNo);

//4 逐个核定票种判断是否能抄报

//4.1 判断是否已经抄报

bReported := True;

oHdFplxdmLst := spliteStr(hdfplxdm, ',');

for idx := 0 to oHdFplxdmLst.Count - 1 do

begin

sFplxdm := oHdFplxdmLst.Strings[idx];

if not oDBAPzxx.isReport(Sksblx, SksbNo, sFplxdm, sCardClock) then

begin

bReported := False;

Break;

end;

end;

if bReported then

begin

//已抄报

updateCzzt(oDBAJxsb, CZZT_CG, '已抄报,无需再执行抄报操作。');

result := true;

Exit;

end;

//4.2 是否到抄报期

if oTaxCardRespOpenCard.IsRepReached <> '1' then

begin

errMsg := '未到抄税期';

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end;

//5 执行抄税操作

oJsCardRespRep := OJsCardComponent.taxReport;

if not oJsCardRespRep.reportIsSucc then

begin

//抄税失败

errMsg := Format('抄报失败:%s-%s', [oJsCardRespRep.Code, oJsCardRespRep.Mess]);

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end

else

begin

//抄税成功

updateCzzt(oDBAJxsb, CZZT_CG, '');

//更新金税盘状态

if Assigned(taskThread) then

taskThread.setThirdPrompt('正在更新金税盘状态信息...');

updateCzlb(oDBAJxsb, CZLB_HQJSPZT);

oJsCardState := oJsCardComponent.queryTaxcardState(errMsg);

if not Assigned(oJsCardState) then

begin

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end;

updateToDb(oDBAJxsb);

//更新设备状态信息

assignFromJsCardState(oJsCardState);

//保存票种核定信息

oJsCardState.Pzxxs.saveToDb(Sksblx, SksbNo);

result := true;

end;

finally

if Assigned(oJsCardComponent) then

FreeAndNil(oJsCardComponent);

if Assigned(oTaxCardRespOpenCard) then

FreeAndNil(oTaxCardRespOpenCard);

if Assigned(oJsCardRespGetClock) then

FreeAndNil(oJsCardRespGetClock);

if Assigned(oJsCardState) then

FreeAndNil(oJsCardState);

if Assigned(oHdFplxdmLst) then

FreeAndNil(oHdFplxdmLst);

if Assigned(oJsCardRespRep) then

FreeAndNil(oJsCardRespRep);

if Assigned(oDBAJxsb) then

FreeAndNil(oDBAJxsb);

if Assigned(oDBAPzxx) then

FreeAndNil(oDBAPzxx);

end;

end; 4.1.2 清卡操作核心代码代码语言:javascript复制{**********************************************************

功能:金税盘清卡操作

参数:errMsg 输出参数,错误信息

返回值:ture - 成功

false - 失败

date: 2021-05-0430

author: 海之边 qq-3094353627

流程简述:

1. 打开金税盘

2. 获取金税盘盘状态

3. 如果尚未清卡则进行清卡

4. 清卡成功后获取金税盘状态,并缓存到本地数据库

**********************************************************}

function TCtrlItem.reportTaxJSP(var errMsg: string): boolean;

var idx: integer;

bCleared: boolean;

sFplxdm, sCardClock: string;

oJsCardComponent: TJsCardComponent;

oTaxCardRespOpenCard: TJsCardResponse_OpenCard;

oJsCardRespGetClock: TJsCardResponse_getClock;

oJsCardState: TJsCardState;

oJsCardRespClearCard: TJsCardResponse_clearReport;

oHdFplxdmLst: TStrings;

oDBAJxsb: TDBAJxsb;

oDBAPzxx: TDBAPzxx;

begin

result := false;

oJsCardComponent := TJsCardComponent.Create;

oTaxCardRespOpenCard := nil;

oJsCardState := nil;

oJsCardRespClearCard := nil;

oHdFplxdmLst := nil;

oDBAJxsb := nil;

oDBAPzxx := nil;

try

oDBAJxsb := TDBAJssb.Create;

oDBAPzxx := TDBAPzxx.Create;

//1 打开金税盘

oTaxCardRespOpenCard := oJsCardComponent.openTaxCard(Zsmm);

if not oTaxCardRespOpenCard.isSucc then

begin

errMsg := Format('打开金税盘%s.%s失败: %s。', [FNsrsbh, FMachineNo, errMsg]);

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end;

if oTaxCardRespOpenCard.isCertNotPass then

begin

//证书密码错误,设置密码校验不通过状态

Zsmmjyjg := untGolbalConst.JYJG_NotPass;

updateToDb(oDBAJxsb);

Exit;

end;

//2. 判断是否已经清卡

//读取金税盘时钟

oJsCardRespGetClock := OJsCardComponent.getClock;

if not oJsCardRespGetClock.isSucc then

begin

errMsg := Format('读取金税盘时钟失败:%d-%s', [oJsCardRespGetClock.retCode,

oJsCardRespGetClock.retMsg]);

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end;

sCardClock := oJsCardRespGetClock.Clock;

updateLastClock(sCardClock);

//读取金税盘状态

oJsCardState := oJsCardComponent.queryTaxcardState(errMsg);

if not Assigned(oJsCardState) then

begin

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end;

updateToDb(oDBAJxsb);

//更新设备状态信息,保存票种核定信息

assignFromJsCardState(oJsCardState);

oJsCardState.Pzxxs.saveToDb(Sksblx, SksbNo);

//这个核定票种判断是否已经抄报

bCleared := True;

oHdFplxdmLst := spliteStr(hdfplxdm, ',');

for idx := 0 to oHdFplxdmLst.Count - 1 do

begin

sFplxdm := oHdFplxdmLst.Strings[idx];

if not oDBAPzxx.isClearCard(Sksblx, SksbNo, sFplxdm, sCardClock) then

begin

bCleared := False;

Break;

end;

end;

if bCleared then

begin

//已经清卡

WriteLog('金税盘%s.%s已清卡,无需再次清卡。', [FNsrsbh, FMachineNo]);

updateCzzt(oDBAJxsb, CZZT_CG, '已清卡,无需再执行清卡操作。');

result := true;

Exit;

end;

//3 执行清卡操作

oJsCardRespClearCard := OJsCardComponent.clearCard;

if not oJsCardRespClearCard.reportIsSucc then

begin

//清卡失败

errMsg := Format('清卡失败:%s-%s', [oJsCardRespClearCard.Code,

oJsCardRespClearCard.Mess]);

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end

else

begin

//清卡成功

updateCzzt(oDBAJxsb, CZZT_CG, '');

//更新金税盘状态

if Assigned(oJsCardState) then

FreeAndNil(oJsCardState);

oJsCardState := oJsCardComponent.queryTaxcardState(errMsg);

if not Assigned(oJsCardState) then

begin

WriteLogError('更新金税盘状态信息失败: %s', [errMsg]);

updateCzzt(oDBAJxsb, CZZT_SB, errMsg);

Exit;

end;

updateToDb(oDBAJxsb);

oJsCardState.Pzxxs.saveToDb(Sksblx, SksbNo);

result := true;

end;

finally

if Assigned(oJsCardComponent) then

FreeAndNil(oJsCardComponent);

if Assigned(oTaxCardRespOpenCard) then

FreeAndNil(oTaxCardRespOpenCard);

if Assigned(oJsCardRespGetClock) then

FreeAndNil(oJsCardRespGetClock);

if Assigned(oJsCardState) then

FreeAndNil(oJsCardState);

if Assigned(oHdFplxdmLst) then

FreeAndNil(oHdFplxdmLst);

if Assigned(oJsCardRespRep) then

FreeAndNil(oJsCardRespRep);

if Assigned(oDBAJxsb) then

FreeAndNil(oDBAJxsb);

if Assigned(oDBAPzxx) then

FreeAndNil(oDBAPzxx);

end;

end;4.2 税控盘组件 税控盘组件的抄税和清卡相关的核心代码,和金税盘类似,笔者不再一一贴出代码,有兴趣或需求的朋友可以和笔者做进一步的沟通交流。

4.3 税务UKey组件 税务UKey组件的抄税和清卡相关的核心代码,和金税盘类似,笔者不再一一贴出代码,有兴趣或需求的朋友可以和笔者做进一步的沟通交流。

5. 后记 上文中笔者所述的组件,抄税和清卡只是其诸多功能之一,同时支持发票开具,作废,冲红,库存查询,发票上传,领购,版式文件下载,发票查询等功能。有兴趣的朋友可以和笔者进一步的沟通交流,同时笔者认知、技术水平有限,若文中有错误或不当之处,欢迎各位批评指正。

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/135138.html原文链接:https://javaforall.cn

相关推荐

曲水铁索桥

曲水铁索桥

h365官方登录平台 08-10
2025主流AI辅助编程工具大盘点

2025主流AI辅助编程工具大盘点

安卓怎么安装365BET 09-06
开关插座使用一个月后分享西门子睿致怎么样评测质量值得买吗?
发起 App 转让

发起 App 转让

安卓怎么安装365BET 01-10
上海苏宁易购加速线下布局!今年预计新开10家门店,打造17家苏宁FUN店
wps/word 如何在一个文档内的不同页 设置不同的页边距(已解决)