【2.使用VBA自动填充Excel工作表】

目录

  • 前言
  • 什么是VBA
  • 如何使用Excel中的VBA
  • 简单基础入门
    • 控制台输出信息
    • 定义过程(功能)
    • 定义变量
    • 常用的数据类型
    • Set
    • 循环
      • For To
  • 我的需求
  • 开发过程
    • 效果演示
    • 文件情况
      • 测试填充源文件
      • 测试填充目标文件
    • 全部完整的代码
      • sheet1中的代码,对应A公司工作表
      • UserForm1的代码
  • 待续
    • 基础逻辑完善
      • 获取行号
      • 遍历日期算法逻辑
      • 提升对单元格等对象的操作知识
    • 功能升级

前言

上一篇文章我使用了Excel中的Index和Match函数实现可以根据之前打过的数据去自动填充内容。在对账的时候可以帮助我们节省很多时间。但是,这样不够直接、更不够快
做过对账单和送货单的人都知道,送货单在出货的时候要打,可能一天一个公司打个一两次。要是我在打送货单的时候,对账单也能自动根据送货单的内容自动填充就好啦

如此细致的功能,我目前只想到了使用Python和VBA。

如果要实现日常的使用,VBA应该就够用了。

什么是VBA

VBA是Visual Basic for Applications的缩写,它是一种事件驱动的编程语言,由微软公司开发。VBA主要用于Microsoft Office套件中的应用程序,如Excel、Word、Access、PowerPoint等,允许用户通过编写宏和自定义函数来自动化任务和扩展应用程序的功能

VBA基于Visual Basic编程语言,它提供了一套丰富的编程工具和对象模型,使得开发者可以创建复杂的自动化脚本和自定义用户界面。使用VBA,用户可以:

自动化重复性任务:通过编写宏来减少手动操作,提高工作效率。
数据处理和分析:在Excel中进行复杂的数据操作和分析。
用户界面定制:在Word或Excel中创建自定义的对话框和表单
与其他应用程序交互:通过VBA编写的代码可以与其他应用程序或数据库进行交互

发明背景

VBA的发明时间可以追溯到1994年左右,当时随着Office 95的发布,VBA作为其宏语言的一部分被引入。它与传统的宏语言不同,提供了面向对象的程序设计方法,并拥有相当完整的程序设计语言。VBA易于学习掌握,可以使用宏记录器记录用户的各种操作并将其转换为VBA程序代码,从而实现工作自动化

VBA与Visual Basic(VB)具有相似的语言结构,从语言结构上讲,VBA是VB的一个子集,它们的语法结构是一样的。VB是一种独立的开发工具,而VBA则必须依附于Office应用程序。VBA专门用于Office的各应用程序中,如Word、Excel、Access等,它与主应用程序之间的通信变得简单而高效。

VBA的编程能力介于低级语言和高级语言之间,它继承了Visual Basic的部分特性和语法,因此具有更强的编程能力和灵活性。然而,VBA的运行速度相对较慢,因为它是通过解释执行而非编译执行的。此外,VBA主要用于Office软件的宏编程和自动化处理,对于其他领域的开发可能不太适用。

如何使用Excel中的VBA

参考链接小步教程
首先,要想使用VBA,需要新建启用宏的工作簿文件后缀名是xlsm
然后再文件-选项中-自定义功能区 选中开发工具

在这里插入图片描述

简单基础入门

控制台输出信息

使用

Debug.Print ""

以及立即窗口(控制台)

定义过程(功能)

Sub 函数名

End Sub

定义变量

Dim 变量名 As 数据类型
Dim a As String

常用的数据类型

  • Workbook 工作簿 表示整个Excel文件,包括所有的工作表
  • Worksheet 工作表 表示单个工作表
  • String 字符串
  • Long 存储大整数,比如行号 (-2,147,483,648 到 2,147,483,647)

Set

给自己定义的变量指向一个对象

' 设置源工作表
    Set wsSource = ThisWorkbook.Sheets(sourceSheetName)
    

    ' 打开目标工作簿
    Set wbTarget = Workbooks.Open("C:\Users\Administrator\Desktop\nosee\EXCEL\测试填充目标文件.xlsm")

循环

VBA循环结构包括四类结构:For To、For Each、While、Do While。

For To

在这里插入图片描述
在这里插入图片描述
退出当前循环

在这里插入图片描述
退出整个循环
在这里插入图片描述

我的需求

要有图形化界面

  • 可以自己选择需要对账的工作表和送货表
  • 自己可以选择要对账的日期和范围 (可能要用到正则表达式)
  • 使用用户界面交互 选择送货单的位置和工作表 对账单的位置和表

开发过程

效果演示

视频如下:

文件情况

测试填充源文件

(这是代码里的名称,在实际工作中就对应送货单)
在这里插入图片描述

如上图,比较值得注意的点是:

  • 序号那种字段名称居然占了两行,这就导致有效数据是日期行加3,不知道做这个送货单的人怎么想的🙄,当时没注意看坑死我了。
  • 还有就是第九行的内容看起来横跨了很多单元格,但是其实只是在A列而已,只是没有展开
  • 还有一个细节就是,对比一下可以看出,在DATE 2024/8/31的下面三行就是有用的数据,而且有用的数据那一行的C、D列都有数据,这些都是我后面用来判断数据有效性的依据

测试填充目标文件

(这个也是代码里的名称,实际中对应的是对账单)
在这里插入图片描述
这个没什么好说的,以后希望也能把日期附在内容前以方便分开

全部完整的代码

sheet1中的代码,对应A公司工作表

    '全局变量 用来接收文件信息
    Public targetSheetName As String
    Public sourceSheetName As String
    Public targetFilePath As String
    Public dateToCheck As String
    
Sub UserForm_Initialize()
    UserForm1.Show '打开用户界面
End Sub

Sub CopyDataBasedOnDate() '定义函数  用户界面
    Dim wsSource As Worksheet '定义变量 Worksheet 是Excel中工作表的对象类型,表示单个工作表。源工作簿不用设置
    Dim wbTarget As Workbook  'Workbook 是Excel中工作簿的对象类型,表示整个Excel文件,包括所有的工作表。
    Dim lastRow As Long
    Dim targetRowA As Long
    Dim targetRowC As Long
    Dim targetRow As Long 'Long 是VBA中的一种数据类型,用于存储较大的整数(-2,147,483,648 到 2,147,483,647)


    ' 设置源工作表
    Set wsSource = ThisWorkbook.Sheets(sourceSheetName)
    ' 打开目标工作簿
    Set wbTarget = Workbooks.Open(targetFilePath)
    ' 找到目标工作表
    Dim wssTarget As Worksheet
    
    
    ' ############设置异常?
    On Error Resume Next '这行代码是一个错误处理语句,它告诉VBA在遇到错误时不要显示错误消息,而是继续执行下一行代码。这是错误处理的开始。
    Set wssTarget = wbTarget.Sheets(targetSheetName) '通过 wsTarget 工作簿对象来访问名为 targetSheetName 的工作表,
    If wssTarget Is Nothing Then ' Nothing 表示一个变量没有引用任何对象。
        MsgBox "目标工作表不存在"
        Exit Sub  '这行代码会立即退出当前正在执行的子程序(Sub)。如果没有找到工作表,程序将不会继续执行后面的代码。
    End If
    On Error GoTo 0 '这行代码用于关闭之前启用的错误处理。在遇到错误时停止执行并显示错误消息。这行代码通常放在错误处理代码的最后,以确保错误处理的范围仅限于特定的代码块。



     ' ############获取源工作表的最后一行  这里可能要调
    lastRow = wsSource.Cells(wsSource.Rows.Count, "A").End(xlUp).Row
    targetRowA = wssTarget.Cells(wssTarget.Rows.Count, "A").End(xlUp).Row
    targetRowC = wssTarget.Cells(wssTarget.Rows.Count, "C").End(xlUp).Row
    targetRow = Application.Max(targetRowA, targetRowC) + 2    '因为最后一行也是两行合并
    ' ##############遍历源工作表中的所有行  应该要换个判断条件或者方式 根据日期,直到下一个日期
    Dim i As Long
    For i = 1 To lastRow '遍历循环
        If wsSource.Cells(i, "A").Value = dateToCheck Then '如果第i行是要对账的日期
            For j = i + 3 To lastRow '日期内循环
                If j - i > 13 Then '3是必要的内容 因为有客套话 假如内容十行的话 就够了
                '最好的办法还是判断日期内循环有没有到下一个日期 到了就直接跳 结束了
                    i = j - 1
                    Exit For
                End If
                
                
                If wsSource.Cells(j, "A").Value = dateToCheck Then '如果已经碰到下一个日期 比如同一天的 还有可能不是同一天的
                    i = j - 1 '节省遍历循环的时间 跳出循环
                    Exit For
                End If
                
                If Not IsEmpty(wsSource.Cells(j, "C").Value) And Not IsEmpty(wsSource.Cells(j, "D").Value) Then
                '如果没有到下一个日期并且数据有效  c列d列不都为空就可以了
                    wsSource.Rows(j).Copy Destination:=wssTarget.Rows(targetRow) '
                    targetRow = targetRow + 1
                End If
            Next j

        End If
    Next i
    


    ' ############保存并关闭目标工作簿
    wbTarget.Save
'    wbTarget.Close

    MsgBox "数据复制完成"
End Sub



UserForm1的代码

Sub dateToCheckCommandButton_Click()
    Sheet1.dateToCheck = dateToCheckTextBox.Text
End Sub

Sub sourceSheetNameCommandButton_Click()
    Sheet1.sourceSheetName = sourceSheetNameTextBox.Text ' 把文本框的内容赋值给源工作表名字变量
End Sub
Sub targetFilePathCommandButton_Click()
    Sheet1.targetFilePath = targetFilePathTextBox.Text
End Sub

Sub targetSheetNameCommandButton_Click()
    Sheet1.targetSheetName = targetSheetNameTextBox.Text ' 把文本框的内容赋值给目标工作表名字变量
End Sub

Sub CommitCommandButton_Click()
    Sheet1.CopyDataBasedOnDate '调用复制程序
End Sub


待续

基础逻辑完善

获取行号

获取源文件行号是从下往上找获取A、C两个必定会有内容的列的单元格不为空的逻辑,但是别人的工作表不一定是这样的

  • 现在使用的是cell和max函数
lastRow = wsSource.Cells(wsSource.Rows.Count, "A").End(xlUp).Row
targetRowA = wssTarget.Cells(wssTarget.Rows.Count, "A").End(xlUp).Row
targetRowC = wssTarget.Cells(wssTarget.Rows.Count, "C").End(xlUp).Row
targetRow = Application.Max(targetRowA, targetRowC) + 2    '因为最后一行也是两行合并

搜了一下还可以用下面这些方法,日后试试

  • 使用UsedRange属性:
    UsedRange属性可以返回工作表中使用过的范围,然后可以通过这个范围来找到最后一行。
Dim lastRow As Long
lastRow = ThisWorkbook.Worksheets("Sheet1").UsedRange.Rows.Count
  • 使用SpecialCells方法:
    这个方法可以用来找到最后一行,特别是当表格中有空行时。
Dim lastRow As Long
On Error Resume Next ' 如果没有找到单元格,避免错误
lastRow = ThisWorkbook.Worksheets("Sheet1").Cells.SpecialCells(xlCellTypeLastCell).Row
On Error GoTo 0 ' 重新启用错误报告

遍历日期算法逻辑

For i = 1 To lastRow '遍历循环
        If wsSource.Cells(i, "A").Value = dateToCheck Then '如果第i行是要对账的日期
            For j = i + 3 To lastRow '日期内循环
                If j - i > 13 Then '3是必要的内容 因为有客套话 假如内容十行的话 就够了
                '最好的办法还是判断日期内循环有没有到下一个日期 到了就直接跳 结束了
                    i = j - 1
                    Exit For
                End If
                
                
                If wsSource.Cells(j, "A").Value = dateToCheck Then '如果已经碰到下一个日期 比如同一天的 还有可能不是同一天的
                    i = j - 1 '节省遍历循环的时间 跳出循环
                    Exit For
                End If
                
                If Not IsEmpty(wsSource.Cells(j, "C").Value) And Not IsEmpty(wsSource.Cells(j, "D").Value) Then
                '如果没有到下一个日期并且数据有效  c列d列不都为空就可以了
                    wsSource.Rows(j).Copy Destination:=wssTarget.Rows(targetRow) '
                    targetRow = targetRow + 1
                End If
            Next j

        End If
    Next i

我在源文件遍历日期的方法是从第一行查找到最后一行,如果遇到匹配的字符串,就再内嵌一个循环,因为一个日期内有很多数据需要一行行地复制过去,还要判断数据的有效性

这里解释一下
For j = i +3是因为序号那一行是两行合并,所以有效数据是日期+3行 ,j-i>13是因为一般来说一次送货不会超过十种货,也就是对应十行数据了

提升对单元格等对象的操作知识

可以看出我对对单元格本身不了解 内置函数和属性不熟 所以判断数据有效性的条件很抽象

功能升级

  • 目前可以实现在页面不关闭的情况下一次插入一天的对账信息,但是这样也不够快,要是能一次插入一个时间范围内的对账信息就好了,估计要新增很多判断逻辑,以及对字符串的一些操作处理(因为有时候你的源工作表不一定就是2024/8/31这样的数据),最好是再做一个控件。

  • 输入文件路径有点麻烦,尝试改进成浏览文件的方式选中。

  • 未来要加入判断插入的数据是否是已有数据的逻辑才行,否则会重复插入 而且最好能提示用户“重复插入了”

  • 从送货单复制的数据到对账单并不是会完美一一对应,因为对账单和送货单所需要的信息不一定是一一对应的,可能会多或者少或者顺序不对,还有一些比较奇葩的是粘贴的目标列合并了,所以要怎么匹配列去进行粘贴 是一个问题

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/883368.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2024年最新Redis内存数据库主从复制、哨兵模式、集群部署等详细教程(更新中)

Centos 安装 Redis 检查安装 GCC 环境 [rootVM-4-17-centos ~]# gcc --version gcc (GCC) 8.5.0 20210514 (Red Hat 8.5.0-4) Copyright (C) 2018 Free Software Foundation, Inc. This is free software; see the source for copying conditions. There is NO warranty; no…

Cisco Secure Firewall Threat Defense Virtual 7.6.0 发布下载,新增功能概览

Cisco Secure Firewall Threat Defense Virtual 7.6.0 - 思科下一代防火墙虚拟设备 (FTDv) Firepower Threat Defense (FTD) Software for ESXi & KVM 请访问原文链接:https://sysin.org/blog/cisco-firepower-7/,查看最新版。原创作品&#xff0c…

信息安全管理工程师(工信部教育与考试中心)

在信息技术迅猛发展的时代,信息安全已经成为企业乃至国家安全不可或缺的一环。 工信部高级信息安全管理工程师认证,作为软考中的一项顶尖资格认证,对提升信息安全管理人员的专业能力、确保信息安全性具有至关重要的作用。 本文将深入探讨该…

神经网络(四):UNet图像分割网络

文章目录 一、简介二、网络结构2.1编码器部分2.2解码器部分2.3完整代码 三、实战案例 一、简介 UNet网络是一种用于图像分割的卷积神经网络,其特点是采用了U型网络结构,因此称为UNet。该网络具有编码器和解码器结构,两种结构的功能如下&#…

网络安全中的 EDR 是什么:概述和功能

专业知识:EDR、XDR、NDR 和 MDR_xdr edr ndr-CSDN博客 端点检测和响应 (EDR) 是一种先进的安全系统,用于检测、调查和解决端点上的网络攻击。它可以检查事件、检查行为并将系统恢复到攻击前的状态。EDR 使用人工智能、机器学习和威胁情报来避免再次发生攻…

矩阵分析 学习笔记4 内积与Gram矩阵

内积 定义 由于对称,第二变元线性那第一变元也线性了。例如这个:

Tomcat may not be running

一、问题背景 tomcat7运行在JDK1.7上,可启动tomcat,但是停止时报错误,如下: 二、适用条件 JDK1.7/JDK1.8 tomcat7 三、解决方法 1、查找java路径 which java 2、修改文件 找到/usr/lib/jvm/jdk1.7.0_80/jre/lib/security/j…

力扣P1706全排列问题 很好的引入暴力 递归 回溯 dfs

代码思路是受一个洛谷题解里面大佬的启发。应该算是一个dfs和回溯的入门题目&#xff0c;很好的入门题目了下面我会先给我原题解思路我想可以很快了解这个思路。下面是我自己根据力扣大佬写的。 我会进行详细讲解并配上图辅助理解大家请往下看 #include<iostream> #inc…

Java 注解详解:从基础到自定义及解析

注解&#xff1a;概述 目标 能够理解注解在程序中的作用 路径 什么是注解注解的作用 注解 什么是注解&#xff1f; 注解(Annotation)也称为元数据&#xff0c;是一种代码级别的说明注解是JDK1.5版本引入的一个特性&#xff0c;和类、接口是在同一个层次注解可以声明在包…

创意实现!在uni-app小程序商品详情页轮播中嵌入视频播放功能

背景介绍 通过uni-app框架实现商城小程序商品详情页的视频与图片轮播功能&#xff0c;以提升用户体验和增加商品吸引力。通过展示商品视频和图片&#xff0c;用户可以更全面地了解商品细节&#xff0c;从而提高购买决策的便利性和满意度。这种功能适用于各类商品&#xff0c;如…

Redis --- redis事务和分布式事务锁

redis事务基本实现 Redis 可以通过 MULTI&#xff0c;EXEC&#xff0c;DISCARD 和 WATCH 等命令来实现事务(transaction)功能。 > MULTI OK > SET USER "Guide哥" QUEUED > GET USER QUEUED > EXEC 1) OK 2) "Guide哥"使用 MULTI命令后可以输入…

嘉立创EDA-- 线宽、过孔和电流大小对比图

导线宽度和电流大小如何来考虑 1 电流大小需要考虑问题 1、允许的温升&#xff1a;如果能够允许的铜线升高的温度越高&#xff0c;那么允许通过的电流自然也就越高 2、走线的线宽&#xff1a;线越宽 &#xff0c;导线横截面积越大&#xff0c;电阻越小&#xff0c;发热越小&a…

影刀---实现我的第一个抓取数据的机器人

你们要的csdn自动回复机器人在这里文末哦&#xff01; 这个上传的资源要vip下载&#xff0c;如果想了解影刀这个软件的话可以私聊我&#xff0c;我发你 目录 1.网页对象2.网页元素3.相似元素组4.元素操作设置下拉框复选框滚动条获取元素的信息 5.变量6.数据的表达字符串变量列…

汽车免拆诊断案例 | 2016 款宾利GT车仪表盘上的多个故障灯点亮

故障现象 一辆2016款宾利欧陆GT车&#xff0c;搭载CYCB发动机&#xff0c;累计行驶里程约为4.5万km。据车主反映&#xff0c;发动机偶尔无法起动&#xff0c;仪表盘上的多个故障灯点亮&#xff08;图1&#xff09;。此外&#xff0c;刮水器、电动车窗及空调等电器设备功能失效…

代码随想录算法训练营第十一天|150. 逆波兰表达式求值 239. 滑动窗口最大值 347.前 K 个高频元素

150. 逆波兰表达式求值 根据 逆波兰表示法&#xff0c;求表达式的值。 有效的运算符包括 , - , * , / 。每个运算对象可以是整数&#xff0c;也可以是另一个逆波兰表达式。 说明&#xff1a; 整数除法只保留整数部分。 给定逆波兰表达式总是有效的。换句话说&#xff0c…

vue3+element-plus icons图标选择组件封装

一、最终效果 二、参数配置 1、代码示例 <t-select-icon v-model"selectVlaue" />2、配置参数&#xff08;Attributes&#xff09;继承 el-input Attributes 参数说明类型默认值v-model绑定值string-prefixIcon输入框前缀iconstringSearchisShowSearch是否显…

注册安全分析报告:人民卫生音像

前言 由于网站注册入口容易被黑客攻击&#xff0c;存在如下安全问题&#xff1a; 暴力破解密码&#xff0c;造成用户信息泄露短信盗刷的安全问题&#xff0c;影响业务及导致用户投诉带来经济损失&#xff0c;尤其是后付费客户&#xff0c;风险巨大&#xff0c;造成亏损无底洞…

Footprint Growthly Quest 工具:赋能 Telegram 社区实现 Web3 飞速增长

作者&#xff1a;Stella L (stellafootprint.network) 在 Web3 的快节奏世界里&#xff0c;社区互动是关键。而众多 Web3 社区之所以能够蓬勃发展&#xff0c;很大程度上得益于 Telegram 平台。正因如此&#xff0c;Footprint Analytics 精心打造了 Growthly —— 一款专为 Tel…

leaflet加载GeoServer的WMS地图服务.md

leaflet加载GeoServer的WMS地图服务&#xff0c;该示例涵盖了涵盖了 “WMS图层加载、WMS图层动态投影、图层index顺序调整、图层添加、高德地图、腾讯地图OpenStreet地图”&#xff0c;WMS图层加载看代码中标注的核心代码部分即可。 <!DOCTYPE html> <html xmlns&qu…

SpringBoot集成阿里easyexcel(一)基础导入导出

easyexcel主要用于excel文件的读写&#xff0c;可使用model实体类来定义文件读写的模板&#xff0c;对开发人员来说实现简单Excel文件的读写很便捷。可参考官方文档 https://github.com/alibaba/easyexcel 一、引入依赖 <!-- 阿里开源EXCEL --><dependency><gr…