vba - 如何使用VBA中的CopyMemory在内存映射文件中存储数据和从内存映射文件中取出数据?

标签 vba excel distributed-computing memory-mapping

我正在尝试构建一个分布式计算系统,该系统使用内存映射文件来通过 VBA 协调多台联网 PC 之间的工作。换句话说,我想让一组联网的计算机以协调的方式同时在一个可以轻松划分为不同部分的项目上工作。一台 PC 需要 13 个多小时才能完成该项目,这对我的客户来说不切实际。

我想将信息存储在内存映射文件中,这将有助于电脑以协调的方式处理项目(即没有重复工作,避免竞争问题等)。我尝试过使用其他类型的文件来完成此操作,但它会导致文件竞争问题或花费太长时间。因此,按照本论坛的建议,我正在尝试内存映射文件。

我对内存映射文件和分布式计算是全新的。必须在VBA中完成。据我所知,我必须指定该文件保存在我们网络上所有 PC 都可以访问的目录(此处为 Z 驱动器)上。我从不同的地方拼凑了一些代码:

Option Explicit

Private Const PAGE_READWRITE As Long = &H4
Private Const FILE_MAP_WRITE As Long = &H2
Private Const GENERIC_READ = &H80000000
Private Const GENERIC_WRITE = &H40000000
Private Const OPEN_ALWAYS = 4
Private Const FILE_ATTRIBUTE_NORMAL = &H80

Private Declare Function CreateFile Lib "kernel32" Alias "CreateFileA" (ByVal lpFileName As String, _
                                         ByVal dwDesiredAccess As Long, ByVal dwShareMode As Long, _
                                         ByVal lpSecurityAttributes As Long, ByVal dwCreationDisposition As Long, _
                                         ByVal dwFlagsAndAttributes As Long, ByVal hTemplateFile As Long) As Long

Private Declare Function CreateFileMapping Lib "kernel32.dll" Alias "CreateFileMappingA" ( _
     ByVal hFile As Long, _
     ByVal lpFileMappigAttributes As Long, _
     ByVal flProtect As Long, _
     ByVal dwMaximumSizeHigh As Long, _
     ByVal dwMaximumSizeLow As Long, _
     ByVal lpName As String) As Long

Private Declare Function MapViewOfFile Lib "kernel32.dll" ( _
     ByVal hFileMappingObject As Long, _
     ByVal dwDesiredAccess As Long, _
     ByVal dwFileOffsetHigh As Long, _
     ByVal dwFileOffsetLow As Long, _
     ByVal dwNumberOfBytesToMap As Long) As Long

#If VBA7 Then
Public Declare PtrSafe Sub CopyMemory Lib "kernel32" Alias _
    "RtlMoveMemory" (destination As Any, source As Any, _
    ByVal length As Long)
#Else
Public Declare Sub CopyMemory Lib "kernel32" Alias _
    "RtlMoveMemory" (destination As Any, source As Any, _
    ByVal length As Long)
    #End If

Private Declare Function UnmapViewOfFile Lib "kernel32.dll" ( _
     ByRef lpBaseAddress As Any) As Long

Private Declare Function CloseHandle Lib "kernel32.dll" ( _
     ByVal hObject As Long) As Long

Private hMMF As Long
Private pMemFile As Long

Sub IntoMemoryFileOutOfMemoryFile()

    Dim sFile As String
    Dim hFile As Long

    sFile = "Z:\path\test1.txt"

    hFile = CreateFile(sFile, GENERIC_READ Or GENERIC_WRITE, 0, 0, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0)
    hMMF = CreateFileMapping(hFile, 0, PAGE_READWRITE, 0, 1000000, "MyMemoryMappedFile")

    pMemFile = MapViewOfFile(hMMF, FILE_MAP_WRITE, 0, 0, 0)

    Dim buffer As String

    buffer = "testing1"
    CopyMemory pMemFile, ByVal buffer, 128

    hMMF = CreateFileMapping(-1, 0, PAGE_READWRITE, 0, 1000000, "MyMemoryMappedFile")
    pMemFile = MapViewOfFile(hMMF, FILE_MAP_WRITE, 0, 0, 0)

     Dim buffer2 As String

    buffer2 = String$(128, vbNullChar)

     CopyMemory ByVal buffer2, pMemFile, 128

     MsgBox buffer2 & " < - it worked?"

     UnmapViewOfFile pMemFile
     CloseHandle hMMF
End Sub

作为一个小例子,上面的代码尝试将字符串“testing1”放入文件 test1.txt 中,然后检索该字符串并将其存储在变量 buffer2 中,最后通过 msgbox 显示该字符串。 super 简单。但是,我不知道自己在做什么。

我们所有的电脑均为 64 位、Windows 7、Office/Excel 2013。

问题/疑问:

  1. 当我运行 IntoMemoryFileOutOfMemoryFile 时,消息框为空
  2. 子完成后,我打开 test1.txt 并得到:“该进程无法访问该文件,因为它正在被另一个进程使用。”这告诉我我没有正确使用 UnmapViewOfFile 和/或 CloseHandle 。
  3. 我想让这些内存文件持久化,这样如果所有电脑都被中断,我可以重新启动进程并从中断的地方继续。

以下是我用来到达现在位置的一些链接:

有趣但不重要的信息:该“项目”是针对对冲基金客户的。我是一名从事基础定量研究的金融人士。我们每天分析 1250 多个数据字段中的 2000 多只股票,以发出宏观经济信号/预测来买卖股票、 future 和期权。

更新:如果我分别像这样更改两个 CopyMemory 行(按值传递 pMemFile):

CopyMemory ByVal pMemFile, buffer, 128

还有...

CopyMemory buffer2, ByVal pMemFile, 128

我在文件 test1.txt 中收到一堆疯狂的字符,并且 Excel 崩溃。

最佳答案

对于您的第一个问题(还没有对其进行太多探索),这与您尝试将缓冲区传递到RtlMoveMemory的方式有关。它需要一个指针,但您向它传递了 BSTR 的副本。 。另请记住,VBA 中的字符串是 Unicode,因此您将得到交织的空字符。我通常使用字节数组或变体(它们将被编码为 CSTR)。

对于您的第二个问题,文件被锁定,因为您从未释放 hFile 的句柄。事实上,一旦将其传递给 CreateFileMappingA,您就可以对 hFile 调用 CloseHandle

对于第三个问题,当您进行第二次调用时,您将覆盖句柄 hMMF 和指针 pMemFile。理论上,它们应该返回与您在同一进程中相同的句柄和指针,但这并不能真正测试您是否获得了 map View 。

至于内存访问,我可能会建议将整个内容包装在一个类中,并将指针映射到比调用RtlMoveMemory更有用的东西。我将您在问题中链接的代码改编为一个类,这应该使它更安全、更可靠、更方便使用(尽管它仍然需要通过错误检查来充实):

'Class MemoryMap
Option Explicit

Private Type SafeBound
    cElements As Long
    lLbound As Long
End Type

Private Type SafeArray
    cDim As Integer
    fFeature As Integer
    cbElements As Long
    cLocks As Long
    pvData As Long
    rgsabound As SafeBound
End Type

Private Const VT_BY_REF = &H4000&
Private Const FILE_ATTRIBUTE_NORMAL = &H80
Private Const OPEN_ALWAYS = &H4
Private Const GENERIC_READ = &H80000000
Private Const GENERIC_WRITE = &H40000000
Private Const PAGE_READWRITE = &H4
Private Const FILE_MAP_WRITE = &H2
Private Const FADF_FIXEDSIZE = &H10

Private cached As SafeArray
Private buffer() As Byte
Private hFileMap As Long
Private hMM As Long
Private mapped_file As String
Private bound As Long

Public Property Get FileName() As String
    FileName = mapped_file
End Property

Public Property Get length() As Long
    length = bound
End Property

Public Sub WriteData(inVal As String, offset As Long)
    Dim temp() As Byte
    temp = StrConv(inVal, vbFromUnicode)

    Dim index As Integer
    For index = 0 To UBound(temp)
        buffer(index + offset) = temp(index)
    Next index
End Sub

Public Function ReadData(offset, length) As String
    Dim temp() As Byte
    ReDim temp(length)

    Dim index As Integer
    For index = 0 To length - 1
        temp(index) = buffer(index + offset)
    Next index

    ReadData = StrConv(temp, vbUnicode)
End Function

Public Function OpenMapView(file_path As String, size As Long, mapName As String) As Boolean
    bound = size
    mapped_file = file_path

    Dim hFile As Long
    hFile = CreateFile(file_path, GENERIC_READ Or GENERIC_WRITE, 0, 0, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0)
    hFileMap = CreateFileMapping(hFile, 0, PAGE_READWRITE, 0, size, mapName)
    CloseHandle hFile
    hMM = MapViewOfFile(hFileMap, FILE_MAP_WRITE, 0, 0, 0)

    ReDim buffer(2)
    'Cache the original SafeArray structure to allow re-mapping for garbage collection.
    If Not ReadSafeArrayInfo(buffer, cached) Then
        'Something's wrong, close our handles.
        CloseOpenHandles
        Exit Function
    End If

    Dim temp As SafeArray
    If ReadSafeArrayInfo(buffer, temp) Then
        temp.cbElements = 1
        temp.rgsabound.cElements = size
        temp.fFeature = temp.fFeature And FADF_FIXEDSIZE
        temp.pvData = hMM
        OpenMapView = SwapArrayInfo(buffer, temp)
    End If    
End Function

Private Sub Class_Terminate()
    'Point the member array back to its own data for garbage collection.
    If UBound(buffer) = 2 Then
        SwapArrayInfo buffer, cached
    End If
    SwapArrayInfo buffer, cached
    CloseOpenHandles
End Sub

Private Sub CloseOpenHandles()
    If hMM > 0 Then UnmapViewOfFile hMM
    If hFileMap > 0 Then CloseHandle hFileMap
End Sub

Private Function GetBaseAddress(vb_array As Variant) As Long
    Dim vtype As Integer
    'First 2 bytes are the VARENUM.
    CopyMemory vtype, vb_array, 2
    Dim lp As Long
    'Get the data pointer.
    CopyMemory lp, ByVal VarPtr(vb_array) + 8, 4
    'Make sure the VARENUM is a pointer.
    If (vtype And VT_BY_REF) <> 0 Then
        'Dereference it for the actual data address.
        CopyMemory lp, ByVal lp, 4
        GetBaseAddress = lp
    End If
End Function

Private Function ReadSafeArrayInfo(vb_array As Variant, com_array As SafeArray) As Boolean
    If Not IsArray(vb_array) Then Exit Function

    Dim lp As Long
    lp = GetBaseAddress(vb_array)
    If lp > 0 Then
        With com_array
            'Copy it over the passed structure
            CopyMemory .cDim, ByVal lp, 16
            'Currently doesn't support multi-dimensional arrays.
            If .cDim = 1 Then
                CopyMemory .rgsabound, ByVal lp + 16, LenB(.rgsabound)
                ReadSafeArrayInfo = True
            End If
        End With
    End If
End Function

Private Function SwapArrayInfo(vb_array As Variant, com_array As SafeArray) As Boolean
    If Not IsArray(vb_array) Then Exit Function
    Dim lp As Long
    lp = GetBaseAddress(vb_array)

    With com_array
        'Overwrite the passed array with the SafeArray structure.
        CopyMemory ByVal lp, .cDim, 16
        If .cDim = 1 Then
            CopyMemory ByVal lp + 16, .rgsabound, LenB(.rgsabound)
            SwapArrayInfo = True
        End If
    End With    
End Function

用法是这样的:

Private Sub MMTest()
    Dim mm As MemoryMap

    Set mm = New MemoryMap
    If mm.OpenMapView("C:\Dev\test.txt", 1000, "TestMM") Then
        mm.WriteData "testing1", 0
        Debug.Print mm.ReadData(0, 8)
    End If

    Set mm = Nothing
End Sub

您还需要在某个地方进行以下声明:

Public Declare Function MapViewOfFile Lib "kernel32.dll" ( _
    ByVal hFileMappingObject As Long, _
    ByVal dwDesiredAccess As Long, _
    ByVal dwFileOffsetHigh As Long, _
    ByVal dwFileOffsetLow As Long, _
    ByVal dwNumberOfBytesToMap As Long) As Long

Public Declare Sub CopyMemory Lib "kernel32" Alias _
    "RtlMoveMemory" (Destination As Any, Source As Any, _
    ByVal length As Long)

Public Declare Function CloseHandle Lib "kernel32.dll" ( _
    ByVal hObject As Long) As Long

Public Declare Function UnmapViewOfFile Lib "kernel32.dll" ( _
    ByVal lpBaseAddress As Any) As Long

要记住的另一件事 - 由于您使用的是网络驱动器,因此您需要确保缓存机制不会干扰对文件的访问。具体来说,您需要确保所有客户端都关闭网络文件缓存。您可能还想确定性地刷新内存映射,而不是依赖操作系统(请参阅 FlushViewOfFile )。

关于vba - 如何使用VBA中的CopyMemory在内存映射文件中存储数据和从内存映射文件中取出数据?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30218821/

相关文章:

excel - 在上面添加一个新行,同时为新行的第一个和最后一个单元格添加后缀

excel - 选中复选框时写入复选框旁边的单元格

excel - 打开多个 Excel 实例会导致 VBA 问题吗?

file - 在 Hadoop 中搜索/查找文件和文件内容

c++ - 在多个主机之间分配进程时,打开 MPI 程序不起作用

java - 网络、集群和虚拟分区

excel - 刻度标签数字格式excel

excel - 以编程方式注释 Excel 图表

arrays - Excel VBA将多个组合框值传递给动态数组

java - 为什么有些条目会被替换?