Winsock协议目录

时间:2022-04-22
本文章向大家介绍Winsock协议目录,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

SPI提供3种协议:

1 分层协议

    在基础协议的上层,依靠底层基础协议实现更高级的通信服务。

2 基础协议

    能够独立安全地,和远程断电实现数据通信的协议,相对于分层协议而言的

 3 协议连

    将一系列的基础协议,分层协议按特定的顺序连接在一起的链状结构。

协议特性:

  winsock用WSAPROTOCOL_INFO结构描述特定协议的完整信息。称为一个winsock目录入口。

  两个重要标识:

ProviderId:服务开发商提供的全局唯一标识

dwCatalogEntryId为每个WSAPROTOCOL_INFO结构安排的唯一标识,称为目录入口ID。

使用winsockapi函数枚举协议

  使用WSAEnumProtocols仅能枚举基础协议和协议链,不能枚举分层协议

执行结果:

使用winsock SPI函数枚举协议

 winsock SPI提供的枚举协议的函数是WSCEnumProtocols,能够枚举各种协议,包括分层协议,基础协议,协议链。

  与WSAEnumProtocols相比,只多了一个 lpError的参数

下面是WSCEnumProtocols代码:

///////////////////////////////////////////////////////////
// WSCEnumProtocols.cpp文件


#include <Ws2spi.h>        //  SPI函数定义在Ws2spi.h文件中
#include <windows.h>
#include <stdio.h>
#pragma comment(lib, "WS2_32")    // 链接到WS2_32.lib

LPWSAPROTOCOL_INFOW GetProvider(LPINT lpnTotalProtocols)
{
    int nError;
    DWORD dwSize = 0;
    LPWSAPROTOCOL_INFOW pProtoInfo = NULL;
    
    //  取得需要的缓冲区长度
    if(::WSCEnumProtocols(NULL, pProtoInfo, &dwSize, &nError) == SOCKET_ERROR)
    {
        if(nError != WSAENOBUFS)
            return NULL;
    }
    // 申请缓冲区,再次调用WSCEnumProtocols函数
    pProtoInfo = (LPWSAPROTOCOL_INFOW)::GlobalAlloc(GPTR, dwSize);
    *lpnTotalProtocols = ::WSCEnumProtocols(NULL, pProtoInfo, &dwSize, &nError);
    return pProtoInfo;
}

void FreeProvider(LPWSAPROTOCOL_INFOW pProtoInfo)
{
    ::GlobalFree(pProtoInfo);
}

void main()
{
    LPWSAPROTOCOL_INFOW pProtoInfo;
    int nProtocols;
    pProtoInfo = GetProvider(&nProtocols);

    for(int i=0; i<nProtocols; i++)
    {
        
        printf(" Protocol: %ws n", pProtoInfo[i].szProtocol);
        printf(" CatalogEntryId: %d        ChainLen: %d nn", 
            pProtoInfo[i].dwCatalogEntryId, pProtoInfo[i].ProtocolChain.ChainLen);
        
    }
}

执行结果: