VB6.0中用户类和数据源类的设计和使用技术

本文转自: http://www.pcworld.com.cn/99/script/9907/072601b.asp

 摘要:用户自定义类(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的知识,建议使用第一种方法。

 

你可能感兴趣的:(数据源)