Excel VBA变量与数据类型:专家级深度解析
发布时间: 2024-11-30 04:46:33 阅读量: 27 订阅数: 35
Excel:宏与VBA编程入门-自动化与数据处理
![Excel VBA](https://img-blog.csdnimg.cn/20200323220932545.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3N4emxj,size_16,color_FFFFFF,t_70)
参考资源链接:[Excel VBA编程指南:从基础到实践](https://wenku.csdn.net/doc/6412b491be7fbd1778d40079?spm=1055.2635.3001.10343)
# 1. Excel VBA基础概述
在信息时代,自动化工具是提高工作效率的关键。Excel VBA(Visual Basic for Applications)作为Microsoft Office产品中的一项强大编程语言,它能够让用户在Excel环境中通过编程来实现复杂的自动化任务,从而极大提升工作效率。VBA支持多种编程范式,包括面向对象编程,这使得它在处理电子表格数据、生成报告、以及实现自动化任务方面具有极大的灵活性。
VBA语言最初是为了简化Microsoft Office应用程序的重复任务而设计,但它现在已经演变成为一个功能丰富的编程环境,可以让开发者开发出复杂的应用程序。通过使用VBA,用户能够控制Excel中的几乎每一个方面,从单元格的格式设置到与其他Office应用程序的交互。
要开始使用VBA,首先需要了解基础知识,例如如何打开VBA编辑器,如何创建和运行宏以及如何调试代码。在此基础上,进一步探讨变量和数据类型是接下来学习的重要步骤,因为它们是构建任何程序的基石。让我们开始探索VBA的旅程,从基础概念开始,逐步深入了解它如何成为自动化和提高工作效率的强大工具。
# 2. ```
# 第二章:深入理解VBA变量
在VBA编程中,变量是存储数据值的基本单元。掌握变量的声明、作用域、数据类型以及它们的高级应用,对于编写高效且可维护的代码至关重要。本章将详细探讨这些主题,并提供实际编程中的应用示例。
## 2.1 变量的声明与作用域
### 2.1.1 变量命名规则与实践
在VBA中,变量的命名需要遵循一定的规则,如名称必须以字母开头、不能包含点号、长度不超过255个字符等。虽然可以使用任意字符作为变量名,但使用具有描述性的名称是最佳实践。
```vba
Dim numberOfStudents As Integer
Dim totalMarks As Double
Dim isApproved As Boolean
```
在上述代码中,`numberOfStudents`、`totalMarks` 和 `isApproved` 分别用于存储学生人数、总分和是否通过的信息,这样命名清晰且易于理解变量用途。
### 2.1.2 变量的作用域及其影响
变量的作用域决定了其在程序中的可见性和生命周期。VBA中的变量作用域主要有两种:局部作用域和全局作用域。局部变量的作用范围限于其声明的程序块中,而全局变量则可以在整个模块,甚至整个工程中访问。
```vba
Sub MySub()
Dim localVariable As Integer '局部变量
'局部变量的使用
End Sub
Public globalVariable As String '全局变量
```
在上述例子中,`localVariable` 是在 `MySub` 过程中声明的局部变量,而 `globalVariable` 由于使用了 `Public` 关键字,它在整个工程中都是可见的。
## 2.2 不同数据类型的变量使用
### 2.2.1 常用数据类型的定义和特点
VBA支持多种数据类型,如整型、双精度浮点型、字符串等。正确使用数据类型可以优化内存使用并减少运行时错误。
```vba
Dim anInteger As Integer '整型
Dim aDouble As Double '双精度浮点型
Dim aString As String '字符串
```
### 2.2.2 类型转换与数据类型的兼容性问题
当不同类型的数据进行操作时,可能需要显式地进行类型转换。例如,将字符串转换为数字类型。
```vba
Dim number As Integer
number = CInt("123") '将字符串转换为整数
```
## 2.3 变量的高级应用
### 2.3.1 动态变量声明与内存管理
动态声明变量允许在运行时根据条件创建变量。这可以用于处理不确定数量的数据项。动态变量声明应谨慎使用,以免导致内存泄漏。
```vba
Dim i As Integer
For i = 1 To 10
Dim myVarName As String
myVarName = "Item" & i
'使用变量 myVarName
Next i
```
### 2.3.2 使用变体( Variant )类型的优势与风险
Variant 类型是一种特殊的类型,它可以存储任何类型的数据。它的优势在于灵活性,但使用不当可能导致性能下降和出错概率提高。
```vba
Dim myVariant As Variant
myVariant = "This is a string"
myVariant = 123 'Variant 可以存储任何类型
```
在使用Variant时,需要注意它的灵活性也意味着在执行类型敏感的操作前,需要进行适当的类型检查或转换,以避免意外的行为。
```
这一部分深入探讨了VBA中变量的使用,强调了变量命名的重要性、作用域的作用以及不同数据类型的定义和特点。同时,也对变量的高级应用进行了说明,包括动态变量声明和Variant类型变量的使用。这为VBA编程提供了坚实的基础,并为进一步学习数据类型和变量在实践中的应用打下了良好的基础。
# 3. VBA数据类型的高级特性
## 3.1 用户定义类型和枚举类型
### 3.1.1 用户定义类型的声明与应用
在VBA中,用户定义类型(UDT)允许开发者组合多个数据类型以创建一个复合的数据结构。这在处理相关数据集时非常有用,比如个人记录、配置设置等。声明一个UDT时,你必须先使用`Type`关键字定义其结构,然后使用`Dim`或`Private`关键字来声明这个类型的变量。
**示例代码:**
```vba
Type PersonalInfo
FirstName As String
LastName As String
BirthDate As Date
Address As String
End Type
Dim person As PersonalInfo
```
在上面的代码段中,我们声明了一个`PersonalInfo`类型的用户定义类型,包括四个字段:`FirstName`、`LastName`、`BirthDate`和`Address`。然后我们声明了这个类型的一个变量`person`。使用UDT可以将相关数据封装在一起,并且可以像操作普通变量一样操作UDT变量。
### 3.1.2 枚举类型的定义及其在代码中的使用
枚举类型(Enum)提供了一种在VBA中定义和使用一组命名常量的方法,这些常量代表了一组相关的值。声明枚举类型可以提高代码的可读性和易维护性。
**示例代码:**
```vba
Enum Weekdays
Sunday = 1
Monday
Tuesday
Wednesday
Thursday
Friday
Saturday
End Enum
Sub TestEnum()
Dim today As Weekdays
today = Friday
MsgBox "It's " & today
End Sub
```
在该示例中,我们声明了一个名为`Weekdays`的枚举类型,其中包含了工作周的每一天。在`TestEnum`子程序中,我们声明了一个`Weekdays`类型的变量`today`,将其赋值为`Friday`,然后使用消息框显示出来。使用枚举类型可以避免使用硬编码的数值,使得代码更加清晰易懂。
## 3.2 数组和集合的使用
### 3.2.1 静态数组与动态数组的区别和应用
数组是用于存储和操作一系列相同类型数据项的变量。在VBA中,数组可以是静态的(在编写代码时大小已知且不可更改)或动态的(大小在运行时可以改变)。
**静态数组的示例代码:**
```vba
Dim numbers(1 To 5) As Integer ' 定义了一个包含5个整数的静态数组
numbers(1) = 10 ' 赋值操作
```
**动态数组的示例代码:**
```vba
Dim dynamicArray() As Integer ' 定义了一个动态数组
ReDim dynamicArray(1 To 10) ' 运行时调整大小为10个元素
dynamicArray(1) = 10
```
使用动态数组时,你需要使用`ReDim`语句来重新设置数组的大小,这对于不确定数据数量的情况非常有用。静态数组则适合于数据量固定不变的情况。
### 3.2.2 集合对象的使用方法和性能考量
集合(Collection)是一种特殊的对象,用于存储多个对象或数据项的无序集合。不同于数组,集合中的每个项可以通过一个键(Key)来引用,这使得在集合中查找、添加和删除项更加灵活。
**示例代码:**
```vba
Dim col As New Collection ' 创建一个新的集合
col.Add "Value1", "Key1" ' 添加项并指定一个键
col.Add "Value2" ' 添加项,VBA会自动创建一个键
```
在处理大量数据时,集合可以提供更好的性能,因为它们在内部是通过哈希表实现的,这使得集合操作的平均时间复杂度为O(1)。然而,数组的随机访问速度非常快,对于连续访问,数组可能仍然是更好的选择。
## 3.3 字符串处理和正则表达式
### 3.3.1 字符串处理技巧与函数
VBA提供了许多内置函数来处理字符串,如`Left`、`Right`、`Mid`、`Len`、`Trim`等。这些函数可以用来裁剪、拼接、计算长度和清除空白字符。
**示例代码:**
```vba
Dim originalString As String
originalString = "Hello, World!"
Dim trimmedString As String
trimmedString = Trim(originalString) ' 移除字符串前后的空格
```
字符串处理是日常编程工作中非常常见的任务,VBA的字符串函数能够帮助开发者高效地进行文本操作。
### 3.3.2 正则表达式在VBA中的实现与应用
VBA支持正则表达式(RegExp),这是一种用于描述字符串匹配模式的工具。正则表达式适用于复杂的字符串验证、搜索和替换操作。
**示例代码:**
```vba
Dim regex As Object
Set regex = CreateObject("VBScript.RegExp")
regex.Pattern = "\b[A-Z][a-z]+ [A-Z][a-z]+\b" ' 匹配首字母大写的名字模式
regex.Global = True
Dim text As String
text = "John Smith goes to the market, Mary Jones is there too."
MsgBox regex.Test(text) ' 检查文本是否符合模式
```
在这个示例中,我们使用正则表达式来匹配两个首字母大写的单词,可能代表人名。正则表达式在处理文本数据时具有巨大的灵活性和强大的功能。
下一章节将继续探讨变量与数据类型在实践中的应用,包括与Excel工作表数据的交互、自动化数据报告生成以及错误处理和数据验证策略等实际应用场景。
# 4. 变量与数据类型在实践中的应用
## 4.1 处理Excel工作表中的数据
在处理Excel工作表中的数据时,变量和数据类型的应用是至关重要的。理解它们如何工作可以极大提高数据操作的效率和准确性。
### 4.1.1 与单元格数据交互的变量应用
为了与单元格数据交互,首先需要声明一个或多个变量,并给它们赋予合适的Excel对象类型。例如,使用 `Range` 对象来引用特定的单元格或单元格区域,并利用 `Value` 属性来操作这些单元格中的数据。
```vba
Dim cellValue As Variant
Dim targetCell As Range
' 假设我们要引用A1单元格
Set targetCell = ThisWorkbook.Sheets("Sheet1").Range("A1")
cellValue = targetCell.Value ' 读取单元格的值
' 修改单元格的值
targetCell.Value = "新的值"
```
在上面的代码中,`cellValue` 用来存储单元格数据,`targetCell` 是一个 `Range` 对象,用来引用特定单元格。通过这种方式,可以方便地读取和修改单元格中的数据。
### 4.1.2 遍历和操作工作表中的数据类型
遍历工作表中的数据类型通常使用 `For Each` 循环结合 `Range` 对象。这在处理大量数据时特别有用,因为它允许我们逐个单元格地操作数据。
```vba
Dim ws As Worksheet
Dim cell As Range
Set ws = ThisWorkbook.Sheets("Sheet1")
For Each cell In ws.Range("A1:A10")
If IsNumeric(cell.Value) Then
' 如果是数字,我们可能想要进行数学运算
cell.Value = cell.Value * 2 ' 双倍这个数字
Else
' 如果不是数字,可能需要进行其他类型的操作
cell.Interior.Color = RGB(255, 255, 0) ' 非数字单元格高亮显示
End If
Next cell
```
在上述示例中,我们对工作表中A1到A10区域的每个单元格进行了遍历,并根据单元格的内容执行了不同的操作。如果是数字,我们将其乘以2;如果非数字,则将单元格的背景设置为黄色。
## 4.2 自动化数据报告和图表
自动化数据报告和图表是VBA应用中的一项重要技能,能够显著提高工作效率。
### 4.2.1 利用变量动态更新图表数据
使用变量动态更新图表数据时,可以灵活地根据数据变化调整图表,而无需手动编辑。这对于定期更新报告非常有用。
```vba
Dim chartObj As ChartObject
Dim dataRange As Range
Set chartObj = ThisWorkbook.Sheets("Sheet1").ChartObjects("Chart 1")
Set dataRange = ThisWorkbook.Sheets("Sheet1").Range("A2:A11")
' 更新图表数据源
chartObj.Chart.SetSourceData Source:=dataRange
' 可以进一步修改图表的其他属性
chartObj.Chart.HasTitle = True
chartObj.Chart.ChartTitle.Text = "季度销售数据"
```
在这段代码中,`chartObj` 是一个 `ChartObject` 对象,用于引用特定的图表。通过 `SetSourceData` 方法,我们动态地将图表的数据源更新为 `dataRange` 变量指定的区域。
### 4.2.2 数据报告生成中的数据类型处理
在生成数据报告时,需要特别注意各种数据类型的处理,确保它们在报告中以正确的方式展示。
```vba
Dim reportSheet As Worksheet
Dim reportRange As Range
Dim i As Long
Set reportSheet = ThisWorkbook.Sheets.Add(After:=ThisWorkbook.Sheets(ThisWorkbook.Sheets.Count))
Set reportRange = reportSheet.Range("A1")
reportRange.Value = "销售报告" ' 报告标题
reportRange.Font.Size = 14
reportRange.Font.Bold = True
reportRange.HorizontalAlignment = xlCenter
' 填充报告的其他数据
For i = 1 To 5
Set reportRange = reportRange.Offset(1, 0)
reportRange.Value = "第 " & i & " 月"
reportRange.Offset(0, 1).Value = "销售数据"
Next i
' 在报告中插入数据类型
reportRange.Offset(1, 0).Value = "数据类型"
reportRange.Offset(1, 1).Value = "示例"
reportRange.Offset(1, 2).Value = "说明"
```
在生成报告的过程中,我们使用循环来填充报告内容,并确保每项数据的格式和类型都是正确的。代码中使用了`Offset`方法在工作表上移动单元格引用,以方便地填充数据。
## 4.3 错误处理与数据验证
在编写涉及变量和数据类型的代码时,错误处理和数据验证是保证代码质量的重要方面。
### 4.3.1 变量和数据类型在错误处理中的角色
正确地使用变量和数据类型可以有效预防和处理错误,从而避免程序崩溃。
```vba
On Error GoTo ErrorHandler ' 启用错误处理
Dim myData As Long
myData = CDec("123.456") ' 将字符串转换为数字
' 其他代码...
Exit ErrorHandler
ErrorHandler:
MsgBox "发生错误: " & Err.Description
End ' 结束程序
```
在这个示例中,我们首先使用 `On Error GoTo` 语句为代码块设置了错误处理程序。如果在代码执行过程中发生错误,程序将跳转到标签 `ErrorHandler` 下执行,显示错误信息。
### 4.3.2 实现数据验证和输入确认的策略
实现数据验证和输入确认可以确保用户输入的数据是合法和有效的,从而保证程序的健壮性。
```vba
Dim userInput As String
userInput = InputBox("请输入您的姓名:")
If Len(userInput) < 3 Then
MsgBox "输入的姓名太短,请重新输入。"
ElseIf Not IsNameValid(userInput) Then
MsgBox "输入的姓名不合法,请包含真实姓名。"
Else
MsgBox "感谢您的输入。" & vbCrLf & "姓名: " & userInput
End If
Function IsNameValid(name As String) As Boolean
' 假设合法姓名只包含字母和空格
IsNameValid = (name Like "*[A-Za-z ]*")
End Function
```
以上代码利用 `InputBox` 函数获取用户输入,并使用 `Len` 和 `Like` 函数进行验证。如果用户输入不合法,会弹出提示信息,并要求重新输入。
这一章的实践应用展示了变量和数据类型在Excel VBA中处理实际问题时的重要性,包括数据交互、报告自动化以及错误处理和验证。下一章将深入探讨性能优化与代码维护的策略,以确保代码的高效运行和长期的可管理性。
# 5. 性能优化与代码维护
在软件开发的实践中,代码的性能优化和维护是确保应用程序长期稳定运行的关键。VBA(Visual Basic for Applications)虽然作为一种较为老旧的技术,但其在Office自动化中仍占据重要地位。本章节将深入探讨如何优化变量和数据类型的使用,以及如何重构代码和维护项目的最佳实践。
## 5.1 优化变量和数据类型的使用
变量和数据类型的优化使用是提高代码性能的有效手段。这涉及到如何高效地管理内存和减少不必要的资源消耗。
### 5.1.1 提升代码性能的变量使用技巧
在VBA中,理解变量的作用域可以有效减少内存的占用。局部变量相比于全局变量,在执行完毕后可以更快地被垃圾回收机制回收,因此在大型的循环或子程序中,尽量使用局部变量。
```vb
Sub PerformanceTest()
Dim i As Long, sum As Long
Dim startTime As Double, endTime As Double
startTime = Timer
' 使用局部变量进行计算
For i = 1 To 1000000
sum = sum + i
Next i
endTime = Timer
Debug.Print "Time taken with local variable: " & (endTime - startTime)
sum = 0
startTime = Timer
' 使用全局变量进行计算
For i = 1 To 1000000
sum = sum + i
Next i
endTime = Timer
Debug.Print "Time taken with global variable: " & (endTime - startTime)
End Sub
```
### 5.1.2 数据类型的优化选择
选择合适的数据类型对于优化性能至关重要。例如,使用`Long`类型而非`Integer`类型在处理大数值时可以避免溢出,并且性能更为优异。
```vb
Sub DataTypePerformance()
Dim startTime As Double, endTime As Double
Dim intSum As Integer, longSum As Long
Dim i As Integer
startTime = Timer
' 使用Integer类型进行计算
For i = 1 To 10000
intSum = intSum + i
Next i
endTime = Timer
Debug.Print "Time taken with Integer: " & (endTime - startTime)
' 重置计时器
startTime = Timer
' 使用Long类型进行计算
For i = 1 To 10000
longSum = longSum + i
Next i
endTime = Timer
Debug.Print "Time taken with Long: " & (endTime - startTime)
End Sub
```
在代码执行中,要注意数据类型转换的开销。特别是在将数据类型从字符串转换为数值时,应该尽量避免不必要的转换。
## 5.2 代码的重构与模块化
重构和模块化是保持代码易于管理和扩展的重要实践。在VBA中,这可以通过拆分过长的过程、定义用户定义函数和类模块来实现。
### 5.2.1 理解代码重构的重要性
重构是改善现有代码的设计和结构而不改变其行为的过程。在VBA中,重构可以带来以下几个好处:
- **提高可读性**:清晰的代码结构更容易理解,有助于维护。
- **减少复杂性**:通过简化代码,减少bug的发生概率。
- **提高复用性**:良好的代码设计可以使函数和子程序被多个地方复用。
### 5.2.2 构建模块化代码的实践技巧
模块化是将程序拆分为独立、可复用的部分的过程。在VBA中,可以利用`Module`、`Class Module`和`UserForm`来实现模块化设计。
```vb
' Class Module: DataHandler
Option Explicit
Public Function SumValues(ParamArray values() As Variant) As Double
Dim sum As Double
Dim i As Integer
For i = LBound(values) To UBound(values)
sum = sum + values(i)
Next i
SumValues = sum
End Function
```
在上述示例中,我们创建了一个`Class Module`,该模块包含了计算数组元素和的功能。这样的模块化设计使得我们可以在不修改原有代码的基础上,进行功能扩展或优化。
## 5.3 代码维护与版本控制
随着项目的持续发展,代码的维护和版本控制变得不可或缺。VBA项目虽然相对较小,但良好的维护习惯可以延长代码的使用寿命。
### 5.3.1 确保代码可维护性的最佳实践
代码的可维护性是保证软件长期稳定运行的基石。以下是一些提高VBA代码可维护性的实践:
- **注释和文档**:为过程、函数和关键代码行添加注释。
- **清晰的命名**:使用有意义的变量和过程名称,遵循命名规范。
- **避免硬编码**:在代码中使用常量代替硬编码的值。
### 5.3.2 使用版本控制系统管理VBA代码
尽管VBA本身不提供内置的版本控制功能,但可以通过外部工具或手动方式来实现。例如,使用`git`对VBA项目进行版本控制,可以利用`gitignore`文件忽略`.xlsm`文件的二进制更改,只跟踪文本文件的更改。
```mermaid
flowchart LR
A[开始] --> B[创建VBA项目]
B --> C[添加版本控制]
C --> D[开发新功能]
D --> E[提交更改到版本库]
E --> F{需要回退版本?}
F --> |是| G[从版本库恢复旧版本]
F --> |否| H[继续开发]
H --> I[发布新版本]
G --> J[结束]
I --> J
```
在上图中,我们使用了流程图来描述使用版本控制系统管理VBA项目的基本过程。通过定期的版本控制,可以确保代码的更改可追溯,并且在出错时能够快速回退到稳定状态。
## 结语
本章节介绍了性能优化与代码维护的多种实践方法,涉及变量和数据类型优化、代码重构和模块化以及版本控制。通过上述实践,能够显著提升VBA代码的性能和可维护性,延长其应用生命周期。接下来的章节将继续深入探讨VBA的高级应用技巧和最新特性。
# 6. 案例分析与高级技巧
在前几章中,我们深入探讨了Excel VBA的基础知识、变量和数据类型的概念、以及高级特性等。本章将通过案例分析与高级技巧,进一步深入理解VBA编程中的变量与数据类型,并探索内存管理和VBA的最新特性。
## 6.1 复杂案例的变量与数据类型分析
### 6.1.1 复杂自动化任务中变量的应用
在处理复杂的自动化任务时,正确地应用变量是成功的关键。考虑一个案例,其中需要自动化处理一系列销售数据,并生成月度报告。变量在这里扮演了重要角色。
```vba
Sub GenerateMonthlySalesReport()
Dim ws As Worksheet
Set ws = ThisWorkbook.Sheets("Data")
Dim lastRow As Long
lastRow = ws.Cells(ws.Rows.Count, "A").End(xlUp).Row
Dim i As Long
For i = 2 To lastRow 'Assuming first row is headers
Dim salesRepName As String
salesRepName = ws.Cells(i, 1).Value
' More variable usage for different calculations and manipulations...
Next i
End Sub
```
### 6.1.2 数据类型在解决实际问题中的作用
在处理不同类型的数据时,选择合适的变量类型可以提升代码的效率和准确性。
```vba
Dim amount As Currency
amount = ws.Cells(i, 3).Value ' Assuming amount is in a currency format
Dim isPromotionApplied As Boolean
isPromotionApplied = ws.Cells(i, 4).Value > 0
```
## 6.2 VBA代码中的内存管理
### 6.2.1 识别和处理内存泄漏
内存泄漏是长期运行的脚本中常见的问题。在VBA中,应当注意及时释放不再使用的对象变量,以避免内存泄漏。
```vba
Dim obj As Object
Set obj = CreateObject("Some.Application")
' ...
' Properly release the object to avoid memory leaks
Set obj = Nothing
```
### 6.2.2 提高代码效率的内存管理策略
在代码设计时,应尽量避免不必要的对象创建和销毁,使用数组替代集合等。
```vba
Dim myArray(1 To 100) As Integer
' Populate the array...
```
## 6.3 探索VBA的最新特性
### 6.3.1 对于最新版本的VBA新特性的概览
VBA定期更新其功能集,最新版本的VBA包含了许多强大的新特性。例如,对JSON的支持、以及更复杂的错误处理机制。
```vba
' JSON Support Example (assuming a JSON string named jsonString)
Dim jsonObject As Object
Set jsonObject = JsonConverter.ParseJson(jsonString)
' Error handling using the Err keyword
On Error GoTo ErrorHandler
' Code that might raise an error
' ...
ExitHandler:
' Clean up code
Set jsonObject = Nothing
Exit Sub
ErrorHandler:
' Error handling code
MsgBox "Error: " & Err.Description
Resume ExitHandler
```
### 6.3.2 应用新特性进行创新实践
利用VBA的新特性,开发者可以编写更加高效、简洁的代码,从而提升整体的工作效率。
```vba
' Example of using Early Binding with a defined type library
' The following code requires a reference to the "Microsoft Scripting Runtime" library
Dim fso As Scripting.FileSystemObject
Set fso = New Scripting.FileSystemObject
' ...
' Remember to set object references to Nothing when done
Set fso = Nothing
```
通过本章的案例分析与高级技巧,我们可以看到VBA编程中变量与数据类型的深入应用。这些内容不仅为初学者提供了坚实的基础,也为有经验的开发人员提供了进一步提升和创新的机会。在实际应用中,不断实践和学习这些高级技巧,是提高个人技术水平和解决复杂问题能力的有效途径。
0
0