百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

c#中使用miniExcel和fastreport实现付款审批单的批量打印

bigegpt 2024-11-30 18:51 4 浏览

有这么一个需求,公司财务会根据银行回单来打印付款申批单,没有软件的情况下只能在Excel表格中把付款审批单的样子做出来,然后再把银行回单的内容填进去,基本上录入一个打印一个,没什么效率不说,也无法实现查重,所以有时候会有付款给付的重复的情况出现;现在我们要给他解决一下这问题,就是通过excel把表格内容导入到软件中,在进行保存时进行查重,重复的记录就不保存了,然后可以按需要多选或单选记录来进行打印,通过fastreport设计好的模版来保障打印质量的可靠稳定,在打印时可以预览打印内容,也可以不预览直接连续打印。

使用c#语言进行开发,选择.net4.8+winform,数据库使用的是SQL 2016,就一个数据表,用来存放审批单的内容;使用MiniExcel来实现读取excel文件并把内容加载到DataGridView中,使用Ef Core 3.1来进行数据查重和保存,通过fastreport来实现按模板打印,可以使用AntdUI来美化下界面,下面开工。

界面设计

大概如下,通过【打开Excel】来实现加载表格中的数据,搜索按钮则是查询数据库中已经存在的数据,保存按钮在保存时会检测是从excel中打开的数据还是从数据库中查询出来的,如果是数据库中查询出来的,就不进行保存,然后单选或多选记录后点击打印,就能实现按模板连续或单个打印了。

NuGet包引用

由于是在.net4.8中使用ef core,所以要引用的包有些多,好些包是自动安装的,需要引用Microsoft.EntityFrameworkCore和Microsoft.EntityFrameworkCore.SqlServer,这两个都选3.1.32版本,还需要引用MiniExcel,不然没法方便导入数据。

包引用完成后,编写数据实体类,使用的是中文表名,所以写法如下

public class CaiWu
{
}

[Table("付款审批表")]
public class ShenPi
{
    [Key]
    public int Id { get; set; }
    public DateTime 日期 { get; set; }
    public string 申请部门 { get; set; }
    public string 项目内容 { get; set; }
    public decimal 金额 { get; set; }
    public string 收款单位 { get; set; }
    public string 账号 { get; set; }
    public string 开户行 { get; set; }
    public string 付款方式 { get; set; }
    public string 备注 { get; set; }
}

编写数据库上下文,jxc.connStr是数据库连接字符串,直接写或写配置文件中都可以

 public class MyDbContext : DbContext
 {
     public MyDbContext()
     {

     }


     protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
     {
         optionsBuilder.UseSqlServer(jxc.connStr);
     }
     /// <summary>
     /// 付款审批表
     /// </summary>
     public DbSet<ShenPi> ShenPis { get; set; }
      
 }

编写Excel导入的代码

 public Boolean isExcel = true;
 List<ShenPi> books = new List<ShenPi>();

点击打开Excel时

 string fileName = "";
 OpenFileDialog ofd = new OpenFileDialog();
 ofd.Filter = "Excel文件|*.xlsx";
 ofd.Multiselect = false;
 var dr = ofd.ShowDialog();
 if (dr == DialogResult.OK)
 {
     fileName = ofd.FileName;
 }
 else
 {
     return;
 }
 try
 {
     books = MiniExcel.Query<ShenPi>(fileName).ToList();
     DG_show.DataSource = books;
     isExcel = true;
 }
 catch (Exception ex)
 {
     MessageBox.Show("打开表格中的数据时出现以下错误:\r\n" + ex.Message.ToString(), jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
     return;
 }

使用miniExcel直接把表格数据转换成list<ShenPi>,在转换时会自动检查数据格式是否正确,不正确的话会报错,会提示第几行第几列出错了。

查询

查询不管是单条件还是多条件,在ef core中实现起来简直太简单了,几行语句搞定,如果你还是使用SQL语句来实现的话,那一定要学下ef core了,两种方法都写了。

//搜索数据库,字符串拼接的方法实现
//string sql;
//string name = "";
//name = txt_name_ck.Text.ToString().Trim();
//if (name.Length < 1)
//{
//    sql = "select top 100 * from 付款审批表 order by id desc ";
//}
//else 
//{
//    sql = "select * from 付款审批表 where 项目内容 like '%" + name + "%' or 收款单位 like '%" + name + "%'";
//}

//DG_show.DataSource = jxc.GetTable(sql);
//isExcel = false;


//使用ef Core进行查询
string name = txt_name_ck.Text.Trim();
if (name.Length < 1)
{
    using (var context = new MyDbContext())
    {
        books = context.ShenPis
            .Take(100)
            .ToList();
    }
}

if (name.Length > 0)
{
    using (var context = new MyDbContext())
    {
        books = context.ShenPis
            .Where(x => x.申请部门.Contains(name) || x.项目内容.Contains(name))
            .ToList();
    }
}
DG_show.DataSource = books;
isExcel = false;

保存

保存最主要的是查重,以前传统方式实现时,想要批量保存的话,查重就好难搞,但是用ef core就很方便了。

int num = 0;
if (!isExcel)
{
    MessageBox.Show("只有从Excel中加载的数据才需要保存哦!", jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
    return;
}
if (books.Count < 1)
{
    MessageBox.Show("当前表格中没有数据哦!", jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
    return;
}

//保存数据
using (var context = new MyDbContext())
{
    books.ForEach(book =>
    {
        var t = context.ShenPis.FirstOrDefault(p => p.项目内容 == book.项目内容 && p.账号 == book.账号 && p.金额 == book.金额 && p.日期 == book.日期);
        if (t != null)
        {
            num += 1;
        }
        else
        {
            context.ShenPis.Add(book);

        }
    });

    try
    {
        context.SaveChanges();
        MessageBox.Show("数据已成功保存,其中有" + num.ToString() + "条重复记录已跳过!", jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
    }
    catch (Exception ex)
    {
        MessageBox.Show("保存数据时出现如下错误:\r\n" + ex.Message.ToString(), jxc.Title, MessageBoxButtons.OK, MessageBoxIcon.Information);
        return;
    }

}

打印

打印主要通过fastreport来实现,自用的话从网上找个大家都懂的版本就行了,之所以不用.net8.和ef core8.0,就是因为找不下大家都懂的版本[捂脸],设计模板这个就不写了,自己按需要和大小画就行了。

点击打印时先循环一下表格,看哪些被选中了,然后再打印,变量是随意起的,主打一个能跑就行[呲牙]

//开始打印
for (int i = 0; i <= DG_show.Rows.Count - 1; i++)
{
    if (DG_show.Rows[i].Selected)
    {
        var row = DG_show.Rows[i];
        PrintExcel(row);
    }

}
 public void PrintExcel(DataGridViewRow row)
 {
     string bm, nr, je, skdw, zh, khh, fkfs, bz, rq;
     string DaXie;
     string path = Environment.CurrentDirectory + @"\report\" + "huidan.frx";

     bm = row.Cells["申请部门"].Value.ToString();
     nr = row.Cells["项目内容"].Value.ToString();
     je = Convert.ToDecimal( row.Cells["金额"].Value).ToString("N2");
     skdw = row.Cells["收款单位"].Value.ToString();
     zh = row.Cells["账号"].Value.ToString();
     khh = row.Cells["开户行"].Value.ToString();
     bz = row.Cells["备注"].Value.ToString();
     rq = Convert.ToDateTime(row.Cells["日期"].Value).ToString("D");
     fkfs = row.Cells["付款方式"].Value.ToString();
     DaXie = jxc.MoneyToChinese(Convert.ToDecimal(je));
     Report rp = new Report();
     rp.Clear();
     rp.Load(path);
     rp.SetParameterValue("申请部门", bm);
     rp.SetParameterValue("项目内容", nr);
     rp.SetParameterValue("金额", je);
     rp.SetParameterValue("收款单位", skdw);
     rp.SetParameterValue("账号", zh);
     rp.SetParameterValue("开户行", khh);
     if (fkfs == "现金")
     {
         rp.SetParameterValue("fs_xj", true);
     }
     else if (fkfs == "汇票")
     {
         rp.SetParameterValue("fs_hp", true);
     }
     else
     {
         rp.SetParameterValue("fs_yh", true);
     }
     rp.SetParameterValue("日期", rq);
     rp.SetParameterValue("备注", bz);
     rp.SetParameterValue("金额大写", DaXie);

     rp.Prepare();
     if (chk_view.Checked)
     {
         rp.ShowPrepared();
     }
     else 
     {
         rp.Print();
     }
    
     //
 }

最后的效果




#挑战30天在头条写日记##职场达人说##职场#

相关推荐

悠悠万事,吃饭为大(悠悠万事吃饭为大,什么意思)

新媒体编辑:杜岷赵蕾初审:程秀娟审核:汤小俊审签:周星...

高铁扒门事件升级版!婚宴上‘冲喜’老人团:我们抢的是社会资源

凌晨两点改方案时,突然收到婚庆团队发来的视频——胶东某酒店宴会厅,三个穿大红棉袄的中年妇女跟敢死队似的往前冲,眼瞅着就要扑到新娘的高额钻石项链上。要不是门口小伙及时阻拦,这婚礼造型团队熬了三个月的方案...

微服务架构实战:商家管理后台与sso设计,SSO客户端设计

SSO客户端设计下面通过模块merchant-security对SSO客户端安全认证部分的实现进行封装,以便各个接入SSO的客户端应用进行引用。安全认证的项目管理配置SSO客户端安全认证的项目管理使...

还在为 Spring Boot 配置类加载机制困惑?一文为你彻底解惑

在当今微服务架构盛行、项目复杂度不断攀升的开发环境下,SpringBoot作为Java后端开发的主流框架,无疑是我们手中的得力武器。然而,当我们在享受其自动配置带来的便捷时,是否曾被配置类加载...

Seata源码—6.Seata AT模式的数据源代理二

大纲1.Seata的Resource资源接口源码2.Seata数据源连接池代理的实现源码3.Client向Server发起注册RM的源码4.Client向Server注册RM时的交互源码5.数据源连接...

30分钟了解K8S(30分钟了解微积分)

微服务演进方向o面向分布式设计(Distribution):容器、微服务、API驱动的开发;o面向配置设计(Configuration):一个镜像,多个环境配置;o面向韧性设计(Resista...

SpringBoot条件化配置(@Conditional)全面解析与实战指南

一、条件化配置基础概念1.1什么是条件化配置条件化配置是Spring框架提供的一种基于特定条件来决定是否注册Bean或加载配置的机制。在SpringBoot中,这一机制通过@Conditional...

一招解决所有依赖冲突(克服依赖)

背景介绍最近遇到了这样一个问题,我们有一个jar包common-tool,作为基础工具包,被各个项目在引用。突然某一天发现日志很多报错。一看是NoSuchMethodError,意思是Dis...

你读过Mybatis的源码?说说它用到了几种设计模式

学习设计模式时,很多人都有类似的困扰——明明概念背得滚瓜烂熟,一到写代码就完全想不起来怎么用。就像学了一堆游泳技巧,却从没下过水实践,很难真正掌握。其实理解一个知识点,就像看立体模型,单角度观察总...

golang对接阿里云私有Bucket上传图片、授权访问图片

1、为什么要设置私有bucket公共读写:互联网上任何用户都可以对该Bucket内的文件进行访问,并且向该Bucket写入数据。这有可能造成您数据的外泄以及费用激增,若被人恶意写入违法信息还可...

spring中的资源的加载(spring加载原理)

最近在网上看到有人问@ContextConfiguration("classpath:/bean.xml")中除了classpath这种还有其他的写法么,看他的意思是想从本地文件...

Android资源使用(android资源文件)

Android资源管理机制在Android的开发中,需要使用到各式各样的资源,这些资源往往是一些静态资源,比如位图,颜色,布局定义,用户界面使用到的字符串,动画等。这些资源统统放在项目的res/独立子...

如何深度理解mybatis?(如何深度理解康乐服务质量管理的5个维度)

深度自定义mybatis回顾mybatis的操作的核心步骤编写核心类SqlSessionFacotryBuild进行解析配置文件深度分析解析SqlSessionFacotryBuild干的核心工作编写...

@Autowired与@Resource原理知识点详解

springIOCAOP的不多做赘述了,说下IOC:SpringIOC解决的是对象管理和对象依赖的问题,IOC容器可以理解为一个对象工厂,我们都把该对象交给工厂,工厂管理这些对象的创建以及依赖关系...

java的redis连接工具篇(java redis client)

在Java里,有不少用于连接Redis的工具,下面为你介绍一些主流的工具及其特点:JedisJedis是Redis官方推荐的Java连接工具,它提供了全面的Redis命令支持,且...