我在vb中这样声明:
Private Declare Function CryptAcquireContext Lib "advapi32.dll" Alias "CryptAcquireContextA" (ByVal phProv As Long, ByVal pszContainer As String, ByVal pszProvider As String, ByVal dwProvType As Long, ByVal dwFlags As Long) As Long
当时在使用时,不论输入什么参数,即使csp根本不存在,都会返回一个大于0的Long,但是取不到容器句柄,phProv始终=0
请问我坐在哪里?

解决方案 »

  1.   

    只是个完整的调用例子,
    但是没怎么看懂  :(
    那位能解释解释???????????'Example Name:Crypto
    'Paste this code in a Class Module, named clsCryptoFilterBoxOption Explicit
    Private Declare Function CryptAcquireContext Lib "advapi32.dll" Alias "CryptAcquireContextA" (phProv As Long, pszContainer As String, pszProvider As String, ByVal dwProvType As Long, ByVal dwFlags As Long) As Long
    Private Declare Function CryptCreateHash Lib "advapi32.dll" (ByVal hProv As Long, ByVal Algid As Long, ByVal hKey As Long, ByVal dwFlags As Long, phHash As Long) As Long
    Private Declare Function CryptDeriveKey Lib "advapi32.dll" (ByVal hProv As Long, ByVal Algid As Long, ByVal hBaseData As Long, ByVal dwFlags As Long, phKey As Long) As Long
    Private Declare Function CryptDestroyHash Lib "advapi32.dll" (ByVal hHash As Long) As Long
    Private Declare Function CryptDestroyKey Lib "advapi32.dll" (ByVal hKey As Long) As Long
    Private Declare Function CryptEncrypt Lib "advapi32.dll" (ByVal hKey As Long, ByVal hHash As Long, ByVal Final As Long, ByVal dwFlags As Long, ByVal pbData As String, pdwDataLen As Long, ByVal dwBufLen As Long) As Long
    Private Declare Function CryptDecrypt Lib "advapi32.dll" (ByVal hKey As Long, ByVal hHash As Long, ByVal Final As Long, ByVal dwFlags As Long, ByVal pbData As String, pdwDataLen As Long) As Long
    Private Declare Function CryptExportKey Lib "advapi32.dll" (ByVal hKey As Long, ByVal hExpKey As Long, ByVal dwBlobType As Long, ByVal dwFlags As Long, ByVal pbData As String, pdwDataLen As Long) As Long
    Private Declare Function CryptGenKey Lib "advapi32.dll" (ByVal hProv As Long, ByVal Algid As Long, ByVal dwFlags As Long, phKey As Long) As Long
    Private Declare Function CryptGetProvParam Lib "advapi32.dll" (ByVal hProv As Long, ByVal dwParam As Long, pbData As Any, pdwDataLen As Long, ByVal dwFlags As Long) As Long
    Private Declare Function CryptGetUserKey Lib "advapi32.dll" (ByVal hProv As Long, ByVal dwKeySpec As Long, phUserKey As Long) As Long
    Private Declare Function CryptHashData Lib "advapi32.dll" (ByVal hHash As Long, ByVal pbData As String, ByVal dwDataLen As Long, ByVal dwFlags As Long) As Long
    Private Declare Function CryptReleaseContext Lib "advapi32.dll" (ByVal hProv As Long, ByVal dwFlags As Long) As Long
    Private Declare Function CryptSignHash Lib "advapi32.dll" Alias "CryptSignHashA" (ByVal hHash As Long, ByVal dwKeySpec As Long, ByVal sDescription As String, ByVal dwFlags As Long, ByVal pbSignature As String, pdwSigLen As Long) As Long
    Private Declare Function CryptVerifySignature Lib "advapi32.dll" Alias "CryptVerifySignatureA" (ByVal hHash As Long, ByVal pbSignature As String, ByVal dwSigLen As Long, ByVal hPubKey As Long, ByVal sDescription As String, ByVal dwFlags As Long) As Long'API error function
    Private Declare Function GetLastError Lib "kernel32" () As Long'API memory functions
    Private Declare Function GlobalAlloc Lib "kernel32" (ByVal wFlags As Long, ByVal dwBytes As Long) As Long
    Private Declare Function GlobalFree Lib "kernel32" (ByVal hMem As Long) As Long
    Private Declare Function GlobalLock Lib "kernel32" (ByVal hMem As Long) As Long
    Private Declare Function GlobalUnlock Lib "kernel32" (ByVal hMem As Long) As Long
    Private Declare Sub CpyMemValAdrFromRefAdr Lib "kernel32" Alias "RtlMoveMemory" (ByVal hpvDest As Any, hpvSource As Any, ByVal cbCopy As Long)
    Private Declare Sub CpyMemRefAdrFromValAdr Lib "kernel32" Alias "RtlMoveMemory" (hpvDest As Any, ByVal hpvSource As Any, ByVal cbCopy As Long)'constants for API memory functions
    Private Const GMEM_MOVEABLE = &H2
    Private Const GMEM_ZEROINIT = &H40
    Private Const GHND = (GMEM_MOVEABLE Or GMEM_ZEROINIT)'constants for Cryptography API functions
    Private Const MS_DEF_PROV = "Microsoft Base Cryptographic Provider v1.0"
    Private Const PROV_RSA_FULL = 1
    Private Const CRYPT_NEWKEYSET = &H8
    Private Const PP_CONTAINER = 6
    Private Const AT_KEYEXCHANGE = 1
    Private Const AT_SIGNATURE = 2Private Const SIMPLEBLOB = 1Private Const ALG_CLASS_DATA_ENCRYPT = 24576
    Private Const ALG_CLASS_HASH = 32768
    Private Const ALG_TYPE_ANY = 0
    Private Const ALG_TYPE_BLOCK = 1536
    Private Const ALG_TYPE_STREAM = 2048
    Private Const ALG_SID_RC2 = 2
    Private Const ALG_SID_RC4 = 1
    Private Const ALG_SID_MD5 = 3Private Const CALG_MD5 = ((ALG_CLASS_HASH Or ALG_TYPE_ANY) Or ALG_SID_MD5)
    Private Const CALG_RC2 = ((ALG_CLASS_DATA_ENCRYPT Or ALG_TYPE_BLOCK) Or ALG_SID_RC2)
    Private Const CALG_RC4 = ((ALG_CLASS_DATA_ENCRYPT Or ALG_TYPE_STREAM) Or ALG_SID_RC4)'constants from WinErr.h
    Private Const NTE_NO_KEY As Long = -2146893811  '0x8009000DL
    Private Const NTE_BAD_SIGNATURE As Long = -2146893818'clsCryptoFilterBox constants
    Private Const CFB_BUSY = 0
    Private Const CFB_READY = 1
    Private Const CFB_VALID = 2Private Const ENCRYPT_ALGORITHM = CALG_RC4
    Private Const ENCRYPT_BLOCK_SIZE = 1Private Const CRYPT_EXPORTABLE = 1'private property buffers
    Private sInBuffer As String
    Private sOutBuffer As String
    Private sPassword As String
    Private sSignature As String
    Private lStatus As Long
    Public Property Get InBuffer() As String
        InBuffer = sInBuffer
    End Property
    Public Property Let InBuffer(vNewValue As String)
        sInBuffer = vNewValue
    End Property
    Public Property Get OutBuffer() As String
        OutBuffer = sOutBuffer
    End Property
    Public Property Get Signature() As String
        Signature = sSignature
    End Property
    Public Property Let Signature(vNewValue As String)
        sSignature = vNewValue
    End Property
    Public Sub Sign()
    'Create a signature for Inbuffer and place in SignatureDim sContainer As String, sDescription As String, sProvider As String, lHCryptprov As Long
    Dim lHHash As Long, lResult As Long, lSignatureLen As LongOn Error GoTo ErrSign'switch Status property
    lStatus = CFB_BUSY'init Signature property
    sSignature = ""'Get handle to the default provider.
    sContainer = vbNullChar
    sProvider = MS_DEF_PROV & vbNullChar
    If Not CBool(CryptAcquireContext(lHCryptprov, ByVal sContainer, ByVal sProvider, PROV_RSA_FULL, 0)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptAcquireContext!")
        GoTo ReleaseHandles:
    End If'Create a hash object.
    If Not CBool(CryptCreateHash(lHCryptprov, CALG_MD5, 0, 0, lHHash)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptCreateHash!")
        GoTo ReleaseHandles:
    End IfIf Not CBool(CryptHashData(lHHash, sInBuffer, Len(sInBuffer), 0)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptHashData!")
        GoTo ReleaseHandles:
    End If'Sign hash object.
    'Determine size of signature.
    sDescription = vbNullChar
    lResult = CryptSignHash(lHHash, AT_SIGNATURE, sDescription, 0, sSignature, lSignatureLen)sSignature = String(lSignatureLen, vbNullChar)'Sign hash object (with signature key).
    If Not CBool(CryptSignHash(lHHash, AT_SIGNATURE, sDescription, 0, sSignature, lSignatureLen)) Then
        MsgBox ("Error " & CStr(GetLastError()) & " during CryptSignHash")
        GoTo ReleaseHandles:
    End IfReleaseHandles:
    'Destroy hash object.
    If lHHash Then lResult = CryptDestroyHash(lHHash)
    'Release provider handle.
    If lHCryptprov Then lResult = CryptReleaseContext(lHCryptprov, 0)'switch Status property
    lStatus = CFB_READYExit SubErrSign:
    MsgBox ("ErrSign " & Error$)
    GoTo ReleaseHandles
    End Sub
      

  2.   

    接上:Public Sub Validate()
    'Validate InBuffer with Signature and assign Status with result
    Dim bValid As Boolean, sContainer As String, sDescription As String, sProvider As String
    Dim lDataLen As Long, lDataPoint As Long, lHCryptprov As Long, lHHash As Long
    Dim lResult As Long, lSignatureLen As Long, lHCryptKey As LongReDim aByteData(0) As ByteOn Error GoTo ErrValidate'switch Status property
    lStatus = CFB_BUSY'init internal valid flag
    bValid = True'Get handle to the default provider.
    sContainer = vbNullChar
    sProvider = MS_DEF_PROV & vbNullChar
    If Not CBool(CryptAcquireContext(lHCryptprov, ByVal sContainer, ByVal sProvider, PROV_RSA_FULL, 0)) Then
        bValid = False
        MsgBox ("Error " & CStr(GetLastError) & " during CryptAcquireContext!")
        GoTo ReleaseHandles:
    End If'Create a hash object.
    If Not CBool(CryptCreateHash(lHCryptprov, CALG_MD5, 0, 0, lHHash)) Then
        bValid = False
        MsgBox ("Error " & CStr(GetLastError) & " during CryptCreateHash!")
        GoTo ReleaseHandles:
    End If'Add data to hash object.
    If Not CBool(CryptHashData(lHHash, sInBuffer, Len(sInBuffer), 0)) Then
        bValid = False
        MsgBox ("Error " & CStr(GetLastError) & " during CryptHashData!")
        GoTo ReleaseHandles:
    End If'Determine size of signature.
    'sDescription = vbNullChar
    'lResult = CryptSignHash(lHHash, AT_SIGNATURE, sDescription, 0, 0, lSignatureLen)'Get handle to signature key.
    If Not CBool(CryptGetUserKey(lHCryptprov, AT_SIGNATURE, lHCryptKey)) Then
        bValid = False
        MsgBox ("Error " & CStr(GetLastError) & " during CryptGetUserKey!")
        GoTo ReleaseHandles:
    End IflSignatureLen = Len(sSignature)'Verify signature.
    If Not CBool(CryptVerifySignature(lHHash, sSignature, lSignatureLen, lHCryptKey, sDescription, 0)) Then    If GetLastError = NTE_BAD_SIGNATURE Then
            bValid = False
            GoTo ReleaseHandles:
        Else
            bValid = False
            MsgBox ("Error " & CStr(GetLastError) & " during CryptVerifySignature!")
            GoTo ReleaseHandles:
        End IfEnd IfReleaseHandles:
    'Release signature key.
    If lHCryptKey Then lResult = CryptDestroyKey(lHCryptKey)
    'Destroy hash object.
    If lHHash Then lResult = CryptDestroyHash(lHHash)
    'Release provider handle.
    If lHCryptprov Then lResult = CryptReleaseContext(lHCryptprov, 0)Select Case bValid
        Case True
            lStatus = CFB_VALID
        Case Else
            lStatus = CFB_READY
    End SelectExit SubErrValidate:
    MsgBox ("ErrValidate " & Error$)
    ResumeEnd Sub
    Public Sub Encrypt()
    'Encrypt InBuffer into OutBufferDim lHExchgKey As Long, lHCryptprov As Long, lHHash As Long, lHkey As Long
    Dim lResult As Long, sContainer As String, sProvider As String, sCryptBuffer As String
    Dim lCryptLength As Long, lCryptBufLen As LongOn Error GoTo ErrEncrypt'switch Status property
    lStatus = CFB_BUSY'Get handle to the default provider
    sContainer = vbNullChar
    sProvider = vbNullChar
    sProvider = MS_DEF_PROV & vbNullChar
    If Not CBool(CryptAcquireContext(lHCryptprov, ByVal sContainer, ByVal sProvider, PROV_RSA_FULL, 0)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptAcquireContext!")
        GoTo Done
    End If'Create a hash object.
    If Not CBool(CryptCreateHash(lHCryptprov, CALG_MD5, 0, 0, lHHash)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptCreateHash!")
        GoTo Done
    End If'Hash in the password data.
    If Not CBool(CryptHashData(lHHash, sPassword, Len(sPassword), 0)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptHashData!")
        GoTo Done
    End If'Derive a session key from the hash object.
    If Not CBool(CryptDeriveKey(lHCryptprov, ENCRYPT_ALGORITHM, lHHash, 0, lHkey)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptDeriveKey!")
        GoTo Done
    End If'Destroy the hash object.
    CryptDestroyHash (lHHash)
    lHHash = 0'Prepare a string buffer for the CryptEncrypt function
    lCryptLength = Len(sInBuffer)
    lCryptBufLen = lCryptLength * 2
    sCryptBuffer = String(lCryptBufLen, vbNullChar)
    LSet sCryptBuffer = sInBuffer'Encrypt data
    If Not CBool(CryptEncrypt(lHkey, 0, 1, 0, sCryptBuffer, lCryptLength, lCryptBufLen)) Then
        MsgBox ("bytes required:" & CStr(lCryptLength))
        MsgBox ("Error " & CStr(GetLastError) & " during CryptEncrypt!")
        'GoTo Done
    End IfsOutBuffer = Mid$(sCryptBuffer, 1, lCryptLength)Done:'Destroy session key.
    If (lHkey) Then lResult = CryptDestroyKey(lHkey)'Release key exchange key handle.
    If lHExchgKey Then CryptDestroyKey (lHExchgKey)'Destroy hash object.
    If lHHash Then CryptDestroyHash (lHHash)'Release provider handle.
    If lHCryptprov Then lResult = CryptReleaseContext(lHCryptprov, 0)'switch Status property
    lStatus = CFB_READYExit SubErrEncrypt:MsgBox ("ErrEncrypt " & Error$)
    ResumeEnd Sub
      

  3.   

    接上:Public Sub Decrypt()
    'Decrypt InBuffer into OutBuffer
    Dim lHExchgKey As Long, lHCryptprov As Long, lHHash As Long, lHkey As Long
    Dim lResult As Long, sContainer As String, sProvider As String
    Dim sCryptBuffer As String, lCryptBufLen As Long, lCryptPoint As Long
    Dim lPasswordPoint As Long, lPasswordCount As LongOn Error GoTo ErrDecrypt'switch Status property
    lStatus = CFB_BUSY'Init sOutBuffer
    sOutBuffer = ""'Get handle to the default provider.
    sContainer = vbNullChar
    sProvider = vbNullChar
    sProvider = MS_DEF_PROV & vbNullChar
    If Not CBool(CryptAcquireContext(lHCryptprov, ByVal sContainer, ByVal sProvider, PROV_RSA_FULL, 0)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptAcquireContext!")
        GoTo Done
    End If'Create a hash object.
    If Not CBool(CryptCreateHash(lHCryptprov, CALG_MD5, 0, 0, lHHash)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptCreateHash!")
        GoTo Done
    End If'Hash in the password data.
    If Not CBool(CryptHashData(lHHash, sPassword, Len(sPassword), 0)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptHashData!")
        GoTo Done
    End If'Derive a session key from the hash object.
    If Not CBool(CryptDeriveKey(lHCryptprov, ENCRYPT_ALGORITHM, lHHash, 0, lHkey)) Then
        MsgBox ("Error " & CStr(GetLastError) & " during CryptDeriveKey!")
        GoTo Done
    End If'Destroy the hash object.
    CryptDestroyHash (lHHash)
    lHHash = 0'Prepare sCryptBuffer for CryptDecrypt
    lCryptBufLen = Len(sInBuffer) * 2
    sCryptBuffer = String(lCryptBufLen, vbNullChar)
    LSet sCryptBuffer = sInBuffer'Decrypt data
    If Not CBool(CryptDecrypt(lHkey, 0, 1, 0, sCryptBuffer, lCryptBufLen)) Then
        MsgBox ("bytes required:" & CStr(lCryptBufLen))
        MsgBox ("Error " & CStr(GetLastError) & " during CryptDecrypt!")
        GoTo Done
    End If'Apply decrypted string from sCryptBuffer to private buffer for OutBuffer property
    sOutBuffer = Mid$(sCryptBuffer, 1, Len(sInBuffer))Done:'Destroy session key.
    If (lHkey) Then lResult = CryptDestroyKey(lHkey)'Release key exchange key handle.
    If lHExchgKey Then CryptDestroyKey (lHExchgKey)'Destroy hash object.
    If lHHash Then CryptDestroyHash (lHHash)'Release provider handle.
    If lHCryptprov Then lResult = CryptReleaseContext(lHCryptprov, 0)'switch Status property
    lStatus = CFB_READYExit SubErrDecrypt:
    MsgBox ("ErrDecrypt " & Error$)
    GoTo DoneEnd Sub
    Public Property Get Status() As Long
        Status = lStatus
    End Property
    Private Function InitUser() As Long
        Dim lHCryptprov As Long, lHCryptKey As Long, avProviderData(1000) As Byte
        Dim lProviderDataAddress As Long, lProviderDataLen As Long, lDataSize As Long
        Dim lResult As Long, sContainer As String, sProvider As String
        Dim sUserName As String, lPoint As Long, lMemHandle As Long
        Dim lReturn As Long, sBuffer As String    On Error GoTo ErrInitUser
        'prepare string buffers    sContainer = vbNullChar
        sProvider = MS_DEF_PROV & vbNullChar    'Attempt to acquire a handle to the default key container.
        If Not CBool(CryptAcquireContext(lHCryptprov, ByVal sContainer, ByVal sProvider, PROV_RSA_FULL, 0)) Then        'Create default key container.
            If Not CBool(CryptAcquireContext(lHCryptprov, ByVal sContainer, ByVal sProvider, PROV_RSA_FULL, CRYPT_NEWKEYSET)) Then
                MsgBox ("Error creating key container! " & CStr(GetLastError))
                Exit Function
            End If        'Get name of default key container.
            lProviderDataLen = Len(avProviderData(0)) * (UBound(avProviderData) + 1)
            If Not CBool(CryptGetProvParam(lHCryptprov, PP_CONTAINER, avProviderData(0), lProviderDataLen, 0)) Then
                MsgBox ("Error getting user name! " & CStr(GetLastError))
                avProviderData(0) = 0
            End If        'Get sUserName from avProviderData()
            lPoint = LBound(avProviderData)
            While lPoint <= UBound(avProviderData)
                If avProviderData(lPoint) <> 0 Then
                    sUserName = sUserName & Chr$(avProviderData(lPoint))
                Else
                    lPoint = UBound(avProviderData)
                End If
                lPoint = lPoint + 1
            Wend        MsgBox ("Create key container " & sUserName)    End If    'Attempt to get handle to signature key
        If Not CBool(CryptGetUserKey(lHCryptprov, AT_SIGNATURE, lHCryptKey)) Then
            If GetLastError = NTE_NO_KEY Then
                MsgBox ("Create key exchange key pair")
                If Not CBool(CryptGenKey(lHCryptprov, AT_SIGNATURE, 0, lHCryptKey)) Then
                    MsgBox ("Error during CryptGenKey! " & CStr(GetLastError))
                    Exit Function
                Else
                    lResult = CryptDestroyKey(lHCryptprov)
                End If
            Else
                MsgBox ("Error during CryptGetUserKey! " & CStr(GetLastError))
                Exit Function
            End If
        End If    'Attempt to get handle to exchange key
        If Not CBool(CryptGetUserKey(lHCryptprov, AT_KEYEXCHANGE, lHCryptKey)) Then
            If GetLastError = NTE_NO_KEY Then
                MsgBox ("Create key exchange key pair")
                If Not CBool(CryptGenKey(lHCryptprov, AT_KEYEXCHANGE, 0, lHCryptKey)) Then
                    MsgBox ("Error during CryptGenKey! " & CStr(GetLastError))
                    Exit Function
                Else
                    lResult = CryptDestroyKey(lHCryptprov)
                End If
            Else
                MsgBox ("Error during CryptGetUserKey! " & CStr(GetLastError))
                Exit Function
            End If
        End If    'release handle to provider
        lResult = CryptReleaseContext(lHCryptprov, 0)
        InitUser = TrueExit FunctionErrInitUser:
        MsgBox ("ErrInitUser " & Error$)
        ResumeEnd Function
    Private Sub Class_Initialize()
        If InitUser = True Then
            MsgBox ("InitUser OK")
        Else
            MsgBox ("InitUser failed")
        End If
    End Sub
    Public Property Get Password() As String
        Password = sPassword
    End Property
    Public Property Let Password(vNewValue As String)
        sPassword = vNewValue
    End Property
      

  4.   

    ms在1996年通过引入Cryptography API(Crypto API)首次涉及了数据保密性问题
    加密可达到一下目的:
    1.保密性
    2.数据完整性
    3.身份验证
    为了达到这些目的,通信双方使用算法和惯列的组合来创建加密方案.
    常用的加密单元:
    私钥加密(对称加密)
    公钥加密(非对称加密)
    加密签名
    加密哈希
      

  5.   

    to passer_wave(路人) 
    那个代码有什么问题??
      

  6.   

    这里有函数的说明:
    http://www.mentalis.org/apilist/CryptAcquireContext.shtml
      

  7.   

    哦,,看了写楼上给的资料,我的理解是这个api是关于数据安全方面的。
    而通讯两端都在相应的加解密。是不是有点像现在ATM网络中adsl调制解调器的那种作用??
      

  8.   

    我是在vb.net下开发的,上面的代码加进去后有些错误,修改后虽能执行但却得不到正确的结果。麻烦大家能在.net下试一下,看看应该怎么做。谢谢!
      

  9.   

    to ctops() 
    .net下有System.Security.Cryptography命名空间,就不需要上面的api了.net下我有vb.net的例子
      

  10.   

    非常感谢大家,正如online所说,.net下有System.Security.Cryptography命名空间,而CspParameters 类可以实现我要的这个函数的功能,我刚才已经调通,可以结贴了。