组合查询——如何使用窗体的继承达到事半功倍?

2024-05-25 14:08

本文主要是介绍组合查询——如何使用窗体的继承达到事半功倍?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  一个优秀的程序员就是要尽可能减少自己的代码量。我们日常的工作或者学习中都有可能遇到多个窗体结构类似,或者大同小异的情况,这时候如果在每个窗体都写一遍代码,或者纯粹的复制粘贴就太水了。想要偷懒的时候正是我们发现和创造的机会,我们的祖先也是这么过来的。

我在做机房收费系统个人重构版的时候,就遇到了基本数据维护、查看上机状态、上机信息统计和操作员工作记录四个窗体都组要用到组合查询而且窗体结构一致。

四个窗体中除了字段名不一致,其他的内容都一样,这种情况下我们用窗体的继承最合适了。在父窗体中写好公共的部分,不一致的地方可以写一个虚方法,然后让子类重写父类的虚方法。这样我们只需要一个U层,一个B层,一个接口,一个抽象工厂,一个D层然后子类重写虚方法就可以实现四个窗体各自的功能了。

U层主要判断一下控件是否为空,给实体参数和B层方法传值传值,以及定义转换数据库字段的虚方法和获取表名的虚方法,这些虚方法在不同的子类中被重写以实现不同功能。

'************************************************* 
'作者:邵鸿鑫
'小组:  
'说明:组合查询模板
'创建日期:2014.7.9
'版本号:
'**********************************************/
Public Class frmGroupQueryPublic thisgroupquery As New Entity.GroupQueryEnpty '定义一个实体参数Private Sub Button1_Click_1(sender As Object, e As EventArgs) Handles btnQuery.Click'第一行组合关系为空If cmbRelation1.Text = "" Then'判断第一行控件内容是否为空If cmbField1.Text = "" ThenMsgBox("请输入字段名")ElseIf cmbOperation1.Text = "" ThenMsgBox("请输入操作符")ElseIf txtContent1.Text = "" ThenMsgBox("请输入要查询的内容")End IfEnd If' 当第一个组合关系不为空时  If cmbRelation1.Text <> "" ThenIf cmbField1.Text = "" ThenMsgBox("请输入字段名")ElseIf cmbOperation1.Text = "" ThenMsgBox("请输入操作符")ElseIf txtContent1.Text = "" ThenMsgBox("请输入要查询的内容")ElseIf cmbField2.Text = "" ThenMsgBox("请输入字段名")ElseIf cmbOperation2.Text = "" ThenMsgBox("请输入操作符")ElseIf txtContent2.Text = "" ThenMsgBox("请输入要查询的内容")End IfEnd If' 当第二个组合关系不为空时  If cmbRelation2.Text <> "" ThenIf cmbField1.Text = "" ThenMsgBox("请输入字段名")ElseIf cmbOperation1.Text = "" ThenMsgBox("请输入操作符")ElseIf txtContent1.Text = "" ThenMsgBox("请输入要查询的内容")ElseIf cmbField2.Text = "" ThenMsgBox("请输入字段名")ElseIf cmbOperation2.Text = "" ThenMsgBox("请输入操作符")ElseIf txtContent2.Text = "" ThenMsgBox("请输入要查询的内容")ElseIf cmbField3.Text = "" ThenMsgBox("请输入字段名")ElseIf cmbOperation3.Text = "" ThenMsgBox("请输入操作符")ElseIf txtContent3.Text = "" ThenMsgBox("请输入要查询的内容")End IfEnd If'给实体层传参Dim thisgroupquery As New Entity.GroupQueryEnptythisgroupquery.cmbField1_text = GetDBName(cmbField1.Text.Trim())thisgroupquery.cmbField2_text = GetDBName(cmbField2.Text.Trim())thisgroupquery.cmbField3_text = GetDBName(cmbField3.Text.Trim())thisgroupquery.cmbOperation1_text = cmbOperation1.Text.Trim()thisgroupquery.cmbOperation2_text = cmbOperation2.Text.Trim()thisgroupquery.cmbOperation3_text = cmbOperation3.Text.Trim()thisgroupquery.txtContent1_text = txtContent1.Text.Trim()thisgroupquery.txtContent2_text = txtContent2.Text.Trim()thisgroupquery.txtContent3_text = txtContent3.Text.Trim()thisgroupquery.cmbRelation1_text = GetDBName(cmbRelation1.Text.Trim())thisgroupquery.cmbRelation2_text = GetDBName(cmbRelation2.Text.Trim())thisgroupquery.GetTable = GetTable()'给B层GroupQuery方法传递参数Dim db As New BLL.GroupQueryBLLIf db.GroupQuery(thisgroupquery) Is Nothing ThenMsgBox("没有记录,请重新设置查询条件", vbOKOnly, vbExclamation)DataGridView1.DataSource = NothingElseDataGridView1.DataSource = db.GroupQuery(thisgroupquery)End IfEnd Sub' 定义虚函数GetDBName,获取不同数据库的字段名  Protected Overridable Function GetDBName(ByVal control As String) As StringReturn ""End Function' 定义虚函数GetDBName,获取不同数据库的表名  Protected Overridable Function GetTable() As StringReturn ""End FunctionPrivate Sub Button2_Click_1(sender As Object, e As EventArgs) Handles btnExit.ClickMe.Hide()End Sub'组合关系一不为空后,显示第二行查询条件Private Sub cmbRelation1_SelectedIndexChanged(sender As Object, e As EventArgs) Handles cmbRelation1.SelectedIndexChangedcmbField2.Enabled = TruecmbOperation2.Enabled = TruecmbRelation2.Enabled = TruetxtContent2.Enabled = TrueEnd Sub'组合关系二不为空后,显示第三行查询条件Private Sub cmbRelation2_SelectedIndexChanged(sender As Object, e As EventArgs) Handles cmbRelation2.SelectedIndexChangedcmbField3.Enabled = TruecmbOperation3.Enabled = TruetxtContent3.Enabled = TrueEnd Sub
End Class

  B层写法跟以前分层写法一样,引用工厂和接口返回查询到的表。

<span style="font-family:KaiTi_GB2312;font-size:18px;">'************************************************* 
'作者:邵鸿鑫
'小组:  
'说明:组合查询B层
'创建日期:2014.7.17
'版本号:
'**********************************************/
Imports IDAL
Imports Entity
Public Class GroupQueryBLLPublic Function GroupQuery(ByVal queries As Entity.GroupQueryEnpty) As DataTableDim factory As New Factory.GroupQueryFactoryDim igroupquery As IGroupQuery = factory.GroupQueryDim table = igroupquery.GroupQuery(queries)'检查D层返回的数据表中是否有数据If table.Rows.Count = 0 ThenReturn NothingElseReturn tableEnd IfEnd Function
End Class</span>

  D层实现接口,调用存储过程实现对数据库的访问。

<span style="font-family:KaiTi_GB2312;font-size:18px;">'************************************************* 
'作者:邵鸿鑫
'小组:  
'说明:组合查询D层
'创建日期:2014.7.17
'版本号:
'**********************************************/
Imports System.Data.SqlClient
Imports IDAL
Public Class SqlserverGroupQueryDAL : Implements IGroupQueryFunction GroupQuery(ByVal queries As Entity.GroupQueryEnpty) As DataTable Implements IGroupQuery.GroupQueryDim strSQL As String = "PROC_GroupQuery" '调用存储过程Dim prams As SqlParameter() = {New SqlParameter("@cmbField1", queries.cmbField1_text),New SqlParameter("@cmbField2", queries.cmbField2_text),New SqlParameter("@cmbField3", queries.cmbField3_text),New SqlParameter("@cmbOperation1", queries.cmbOperation1_text),New SqlParameter("@cmbOperation2", queries.cmbOperation2_text),New SqlParameter("@cmbOperation3", queries.cmbOperation3_text),New SqlParameter("@txtContent1", queries.txtContent1_text),New SqlParameter("@txtContent2", queries.txtContent2_text),New SqlParameter("@txtContent3", queries.txtContent3_text),New SqlParameter("@cmbRelation1", queries.cmbRelation1_text),New SqlParameter("@cmbRelation2", queries.cmbRelation2_text),New SqlParameter("@tableName", queries.GetTable)} '设置参数Dim helper As New SqlHelperDim table As New DataTabletable = helper.GetDataTable(strSQL, CommandType.StoredProcedure, prams)Return tableEnd Function
End Class</span>

  存储过程

<span style="font-family:KaiTi_GB2312;font-size:18px;">-- =============================================
-- Author:		邵鸿鑫
-- Create date: 20140719
-- Description:	组合查询
-- =============================================
ALTER PROCEDURE [dbo].[PROC_GroupQuery]-- Add the parameters for the stored procedure here@cmbField1 varchar(10),@cmbOperation1 varchar(10),@txtContent1 varchar(10),@cmbField2 varchar(10),@cmbOperation2 varchar(10),@txtContent2 varchar(10),@cmbField3 varchar(10),@cmbOperation3 varchar(10),@txtContent3 varchar(10),@cmbRelation1 varchar(10),@cmbRelation2 varchar(10),@tableName varchar(20)
ASdeclare @TempSql varchar(500)--临时存放sql语句  
BEGINSET @TempSql='SELECT * FROM '+@tableName +' WHERE ' +@cmbField1 +@cmbOperation1+char(39) + @txtContent1 + char(39)  if @cmbRelation1 != ''BEGIN  SET @TempSql=@TempSql+@cmbRelation1+CHAR(32)+@cmbField2 +@cmbOperation2+CHAR(39)+@txtContent2+CHAR(39)  if @cmbRelation2!= ''BEGIN  SET @TempSql=@TempSql+@cmbRelation2+CHAR(32)+@cmbField3+@cmbOperation3+CHAR(39)+@txtContent3+CHAR(39)  END  END  
EXECUTE(@TempSql) 
END</span>

  抽象工厂应用配置文件,实现灵活更换数据库。

<span style="font-family:KaiTi_GB2312;font-size:18px;">'************************************************* 
'作者:邵鸿鑫
'小组:  
'说明:组合查询抽象工厂
'创建日期:2014.7.17
'版本号:
'**********************************************/
Imports System.Reflection
Imports System.Configuration
Imports IDAL
Public Class GroupQueryFactoryPrivate Shared ReadOnly AssemblyName As String = "DAL" '声明程序集名称Private Shared ReadOnly db As String = ConfigurationManager.AppSettings("DB") '读取配置文件到dbPublic Function GroupQuery() As IGroupQueryDim className As String = AssemblyName + "." + db + "GroupQueryDAL"Dim igroupquery As IGroupQueryigroupquery = CType(Assembly.Load(AssemblyName).CreateInstance(className), IGroupQuery)Return igroupqueryEnd Function
End Class
</span>

  子类代码以操作员工作记录为例,首先添加继承窗体

然后选择被继承的窗体

之后就是重写父类的虚方法

<span style="font-family:KaiTi_GB2312;font-size:18px;">'************************************************* 
'作者:邵鸿鑫
'小组:  
'说明:查询操作员工作记录U层
'创建日期:2014.7.18
'版本号:
'**********************************************/
Public Class frmworklog'重写转换成数据库字段方法Protected Overrides Function GetDBName(ByVal control As String) As StringSelect Case (control)Case "教师"Return "userName"Case "上班日期"Return "loginTime"Case "上班时间"Return "loginTime"Case "下班日期"Return "offTime"Case "下班时间"Return "offTime"Case "机器号"Return "computer"Case "与"Return "and"Case "或"Return "or"Case ElseReturn ""End SelectEnd Function'重写获得表名方法Protected Overrides Function GetTable() As Stringthisgroupquery.GetTable = "Work_info"Return thisgroupquery.GetTableEnd FunctionPublic Sub frmworklog_load(sender As Object, e As EventArgs) Handles MyBase.Load'给字段赋初值cmbField1.Items.Add("教师")cmbField1.Items.Add("上班日期")cmbField1.Items.Add("上班时间")cmbField1.Items.Add("下班日期")cmbField1.Items.Add("下班时间")cmbField1.Items.Add("机器号")cmbField2.Items.Add("教师")cmbField2.Items.Add("上班日期")cmbField2.Items.Add("上班时间")cmbField2.Items.Add("下班日期")cmbField2.Items.Add("下班时间")cmbField2.Items.Add("机器号")cmbField3.Items.Add("教师")cmbField3.Items.Add("上班日期")cmbField3.Items.Add("上班时间")cmbField3.Items.Add("下班日期")cmbField3.Items.Add("下班时间")cmbField3.Items.Add("机器号")End Sub
End Class</span>
  为了避免用户不按顺序组合查询条件,把第二行和第三行查询条件以及第二个组合关系设的属性置为不可用,只有第一个组合关系不为空时第二行查询条件和第二个组合关系才能使用,同理第二个组合关系不为空时第三行组合条件才能使用。

  言归正传这一切的一切都是为了减少我们的工作量,人类的进步其实就是一次次为了“偷懒”而不断探索的过程。文章中有不足之处还请大神们批评指正,希望能对大家有所帮助。

这篇关于组合查询——如何使用窗体的继承达到事半功倍?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

MySQL 多列 IN 查询之语法、性能与实战技巧(最新整理)

《MySQL多列IN查询之语法、性能与实战技巧(最新整理)》本文详解MySQL多列IN查询,对比传统OR写法,强调其简洁高效,适合批量匹配复合键,通过联合索引、分批次优化提升性能,兼容多种数据库... 目录一、基础语法:多列 IN 的两种写法1. 直接值列表2. 子查询二、对比传统 OR 的写法三、性能分析

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

prometheus如何使用pushgateway监控网路丢包

《prometheus如何使用pushgateway监控网路丢包》:本文主要介绍prometheus如何使用pushgateway监控网路丢包问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录监控网路丢包脚本数据图表总结监控网路丢包脚本[root@gtcq-gt-monitor-prome

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数

SpringBoot中如何使用Assert进行断言校验

《SpringBoot中如何使用Assert进行断言校验》Java提供了内置的assert机制,而Spring框架也提供了更强大的Assert工具类来帮助开发者进行参数校验和状态检查,下... 目录前言一、Java 原生assert简介1.1 使用方式1.2 示例代码1.3 优缺点分析二、Spring Fr

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件

SpringBoot线程池配置使用示例详解

《SpringBoot线程池配置使用示例详解》SpringBoot集成@Async注解,支持线程池参数配置(核心数、队列容量、拒绝策略等)及生命周期管理,结合监控与任务装饰器,提升异步处理效率与系统... 目录一、核心特性二、添加依赖三、参数详解四、配置线程池五、应用实践代码说明拒绝策略(Rejected

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

Ubuntu如何分配​​未使用的空间

《Ubuntu如何分配​​未使用的空间》Ubuntu磁盘空间不足,实际未分配空间8.2G因LVM卷组名称格式差异(双破折号误写)导致无法扩展,确认正确卷组名后,使用lvextend和resize2fs... 目录1:原因2:操作3:报错5:解决问题:确认卷组名称​6:再次操作7:验证扩展是否成功8:问题已解