摘要:用户自定义类(Class)是Visual Basic 6.0中实现软件重用的最基本方法,也是设计和使用Active X EXE/DLL部件的技术基础。在VB6.0中除了可定义一般功能的类外,还可以定义从外部源获取数据的数据源类(包括ODBC 源、ADO,或者任何 OLE DB 提供程序)。该文介绍了在VB 6.0中定义和使用类的属性过程、方法、特别是类的自定义事件的方法与技巧,最后特别介绍了包含两个外部数据成员的数据源类的定义、使用的若干技术。
关键词:类(Class),自定义事件,数据源类
一、 绪论
Visual Basic 6.0是面向对象的编程语言,允许开发人员设计自定义类。用户自定义类(Class)是VB 6.0中实现软件重用的最基本方法,也是设计和使用Active X EXE/DLL部件的技术基础。在VB 6.0中可以用手工的方式直接设计类,也可用类设计器设计类,而且,VB 6.0提供了类向导(如数据源类向导,复杂数据源类向导)供快速设计复杂类。本文基本上以笔者设计的通讯电源电压监控为例介绍定义和使用类(特别是类的自定义事件)的方法(包括属性、方法、自定义事件),并将讨论包含两个外部数据成员的数据源类的定义、使用技术。
二、自定义类的设计和使用
2.1 自定义类的设计
手工设计自定义类分以下四步:
第一步、设计类的属性或属性过程(Property)
在VB 6.0中有两种类型的属性,即用向类模块中添加Public变量的方法定义的一般属性,和用向类模块中添加Property GET/LET/SET过程的方法定义的属性过程,一般使用属性过程。属性过程分只读(Property GET)、只写(Property LET/SET)、读写(Property GET和LET/SET)属性过程三种,SET用于定义对象类型属性过程,而LET用于定义非对象类型属性过程。GET/LET/SET的含义是:当读取/使用(即Access存取)该属性值时,执行Property Get中的代码;当写入/修改(即Assign指派)该属性值时,执行Property Let或Set中的代码。因为在存取或指派属性值时能执行一段代码,我们就可以在属性过程中增加许多代码(如检验代码),这正是属性过程的优点。
例如:下面代码定义了两个属性CurVolume和PreviousVolume
Option Explicit
Public PreviousVolume As Integer
'定义了属性PreviousVolume
'下述代码定义属性过程CurVolume
Dim mVolume As Integer
'为属性CurVolume定义临时存贮区
Public Property Get CurVolume() As Integer
'读属性CurVolume
CurVolume = mVolume
End Property
Public Property Let CurVolume(ByVal vNewValue As Integer)
'写属性CurVolume
If vNewValue < 0 Then
MsgBox ("电压值为负,无法写入!")
Else
mVolume = vNewValue
End If
End Property
关于属性过程有几点说明:
读写属性过程的Property GET名称和LET/SET名称必须相同,且Property GET的参数个数比LET/SET的参数个数少一个,但Property GET的类型和LET/SET的最后一个参数的类型相同。例如:下例定义了一个数组属性mintNa(0 TO 99)
Option Explicit
Private mintN As Integer
Private mintNa(100) As String
Public Property Get NumberOfTeeth
(ByVal Newvalue As Integer) As String
NumberOfTeeth = mintNa(Newvalue)
End Property
Public Property Let NumberOfTeeth(ByVal Newvalue As
Integer, ByVal x As String)
mintNa(Newvalue) = x
End Property
由于窗体为类,故可以为窗体添加属性过程。
属性过程的作用域有三种:
Private:本类模块中有效
Public:本类模块和其他模块中均有效
Friend:主要用于Active X部件设计。对本工程中而言,相当于Public;对其他工程/应用而言,相当于Private
第二步、设计类的方法(Method)
向类模块中添加Sub 或 Function 过程即可。例如:下例定义了一个公用方法
Public Sub VolumeCheck()
If CurVolume < 200 Or CurVolume > 240 Then
MsgBox ("引发事件VolumeWarn")
RaiseEvent VolumeWarn
End If
End Sub
第三步、设计类的事件(Event)
在VB6中,设计和使用类模块中的自定义事件很特殊,不易理解。具体而言,向类模块中添加自定义事件有三步:
(1)在类模块的声明部分,用Public Event 定义一个事件,事件可以有参数。例如,下例定义了两个事件:
Option Explicit
Public Event VolumeWarn()
Public Event PercentDone(ByVal pERcent As
Integer, ByRef Cancel As Boolean)
(2)在该类模块的某一个模块中,用Raise Event引发该事件。例如,公用模块VolumeCheck引发事件VolumeWarn(如果属性CurVolume之值在200~240V范围之外则引发该事件):
Public Sub VolumeCheck()
If CurVolume < 200 Or CurVolume > 240 Then
MsgBox ("引发事件VolumeWarn")
RaiseEvent VolumeWarn
End If
End Sub
如果该事件被引发,那么将执行为该事件编写的事件代码。问题是:事件代码在哪儿编写?不是在该类模块中编写,而是在事件源(引发该事件的对象)中编写,在事件源的声明部分用WITHEVENT声明该类。
(3)在事件源中编写事件代码。例如,在项目中的Test窗体中引发该事件(即Test为事件源),那么:
首先在Test窗体的声明部分用WITHEVENT声明该类(Private WithEvents xVolume As ClsVolumeCheck),代码编辑窗口的左边将出现该WITHEVENT变量xVolume
然后在代码编辑窗口的左边选择WITHEVENT变量,在右边选择VolumeWarn模块,于是产生一个空模块(如下图):
在右边的VolumeWarn模块中为该事件编写的事件代码:
Private Sub xVolume_VolumeWarn()
MsgBox ("电压不正常!") '处理电压代码
' ..............
End Sub
从上可以发现,不同的事件源所编写的自定义事件代码可以不相同。如何使用事件将在后面讲述。
第四步、编写类的Initalize和Terminate事件代码
为该类编写初始化和中断事件。
2.2 自定义类的使用
使用自定义类分四步:
第一步、在适当的模块中(如通用)声明类变量。若类中有自定义事件,则使用用WITHEVENT声明该类,并为自定义事件编写事件代码(如前述的设计类的事件的第(3)步)。例如:
Private WithEvents xVolume As ClsVolumeCheck
Private yVolume As cls多态接口
Private zVolume As cls抽象类
第二步、在适当的模块中(如Form_Load模块)对类变量赋值。例如:
Private Sub Form_Load()
Set xVolume = New ClsVolumeCheck
End Sub
第三步、在适当的模块中(如Form_Load模块)使用类的Public或Friend属性、方法。例如:
Private Sub Command1_Click()
xVolume.CurVolume = Text1.Text
'对xVolume的CurVolume属性赋值
Call xVolume.VolumeCheck
'调用xVolume的VolumeCheck过程
End Sub
由于ClsVolumeCheck类中的VolumeCheck过程中有RaiseEvent VolumeWarn语句,故Call xVolume.VolumeCheck时可能引发事件VolumeWarn。当200 >Text1.Text >240时,将引发事件VolumeWarn,并执行在本窗体中为该类编写的事件代码:
Private Sub xVolume_VolumeWarn()
MsgBox ("电压不正常!")
'处理电压代码
End Sub
第四步、释放类对象变量。这一点非常重要,当不用类对象变量时一定要释放它,以释放内存资源。例如:
Private Sub Form_QueryUnload(Cancel As Integer,
UnloadMode As Integer)
Set xVolume = Nothing
End Sub
三、数据源类的设计与使用
数据源类是一个从外部源获取数据的类,这些数据将被其他对象所使用的,数据识别类不是必须有可视的外在表示,也不局限于某个特定的数据接口(如DAO 或者RDO)。数据识别类可以作为任何类型数据的数据源,包括传统的 ODBC 源、ActiveX Data Objects (ADO),或者任何 OLE DB 提供程序。
3.1创建步骤
第一步、增加一个新类,设置类的DataSourceBehavior 属性为vbDataSource,则类就可以作为其他对象的数据源;增加ActiveX Data Objects 2.0 Library的引用。
第二步、设计数据源类的Initialize事件代码、GetDataMember过程模块;
第三步、为数据源类设计一些Public方法。
例如下面的代码定义了一个包含两个外部数据成员的数据源类:
Option Explicit
Public Event MoveComplete()
'公共事件
Dim WithEvents adoEmployeesRS As Recordset
'第一个外部数据成员
Dim WithEvents adoVolumeRS As Recordset
'第二个外部数据成员
Private Sub Class_Initialize()
'类的初始化事件
Dim db As Connection
Set db = New Connection
db.CursorLocation = adUseClient
db.Open "PROVIDER=Microsoft.Jet.OLEDB.3.51;
Data Source=C:\Program Files\Microsoft
Visual Studio\VB98\Nwind.mdb;"
'创建一个具体的ADO连接
Set adoEmployeesRS = New Recordset
adoEmployeesRS.Open "select Address,BirthDate,City
from Employees", db, adOpenStatic, adLockOptimistic
'创建第一个具体的外部源所对应的
ADO Recordset(adoEmployeesRS)
DataMembers.Add " Employees"
Set adoVolumeRS = New Recordset
adoVolumeRS.Open "select Address,City,CompanyName from
Customers", db, adOpenStatic, adLockOptimistic
'创建第二个具体的外部源所对应的
ADO Recordset(adoVolumeRS)
DataMembers.Add " Customers "
End Sub
Private Sub Class_GetDataMember(DataMember As String,
Data As Object) '读数据源时发生
Select Case DataMember
Case "Employees" '第一个ADO Recordset
Set Data = adoEmployeesRS
Case "Customers" '第二个ADO Recordset
Set Data = adoVolumeRS
End Select
End Sub
'下面设计一些方法:
Private Sub adoPrimaryRS_MoveComplete()
RaiseEvent MoveComplete
End Sub
Public Sub RequeryEmployees ()
adoPrimaryRS.Requery
DataMemberChanged "Employees"
End Sub
Public Sub RequeryCustomers ()
adoVolumeRS.Requery
DataMemberChanged "Customers"
End Sub
3.2 如何在窗体模块中使用数据源类
可以使用下面两种方法之一完成数据源和数据消费者(如TextBox)之间的绑定:
方法一、利用数据属性来完成绑定。
假设Form中有两个下面文本框控件数组(txtFieldsEmployees,txtFieldsCustomers)将分别用于显示两个外部数据成员的三个字段数据,其代码为:
Option Explicit
Private WithEvents PrimaryCLS As clsEmployeesCustomers
Private Sub Form_Load()
Set PrimaryCLS = New clsEmployeesCustomers
Dim oText As TextBox
'绑定文本框控件数组txtFieldsEmployees到数据提供者
For Each oText In Me.txtFieldsEmployees
oText.DataMember = "Employees" '设置DataMember
Set oText.DataSource = PrimaryCLS '设置DataSource
Next
txtFieldsEmployees (0).DataField = "Address" '设置DataField
txtFieldsEmployees (1).DataField = "BirthDate"
txtFieldsEmployees (2).DataField = "City"
'绑定文本框控件数组txtFieldsCustomers到数据提供者
For Each oText In Me.txtFieldsCustomers
oText.DataMember = "Customers"
Set oText.DataSource = PrimaryCLS
Next
txtFieldsCustomers (0).DataField = "Address"
txtFieldsCustomers (1).DataField = "City"
txtFieldsCustomers (2).DataField = "CompanyName"
End Sub
Private Sub cmdRefresh_Click() '调用公用方法
On Error GoTo RefreshErr
PrimaryCLS.RequeryEmployees
PrimaryCLS.RequeryCustomers
Exit Sub
RefreshErr:
MsgBox Err.Description
End Sub
方法二、利用BindingCollection来完成绑定。
第一步、增加BindingCollection的引用,BindingCollection 是某个数据源和一个或多个数据使用者之间的绑定的集合。
第二步、设计窗体声明、Load代码。
第三步、在窗体的其他模块中使用数据源类的Public方法。
例如下面的代码使用上面的包含两个外部数据成员的数据源类:
Option Explicit
Private WithEvents PrimaryCLS As clsEmployeesCustomers
Private objBC As BindingCollection '定义一个绑定集合objBC
Private Sub Form_Load()
Set PrimaryCLS = New clsEmployeesCustomers
Set objBC = New BindingCollection '对绑定集合objBC赋值
objBC.DataMember = "Employees" '设置objBC 的DataMember
Set objBC.DataSource = PrimaryCLS '设置objBC 的DataSource
objBC.Add txtFieldsEmployees(0), "text", "address"
'使用objBC绑定集合的.Add方法绑定txtFieldsEmployees(0)
文本框的Text属性到Employees 记录集的address字段
objBC.Add txtFieldsEmployees(1), "text", "BirthDate"
'绑定txtFieldsEmployees(1)到BirthDate
objBC.Add txtFieldsEmployees(2), "text", "City"
'绑定txtFielldsEmployees(1)到City
objBC.DataMember = "Customers"
Set objBC.DataSource = PrimaryCLS
objBC.Add txtFieldsCustomers(0), "text", "address"
objBC.Add txtFieldsCustomers(1), "text", "CompanyName"
objBC.Add txtFieldsCustomers(2), "text", "City"
End Sub
上述两方法中第一种方法显得简洁,不象第二种方法需要BindingCollection的知识,建议使用第一种方法。