MFC下CSocket编程详解

MFC下CSocket编程详解

分类: C/C++2008-03-13 09:01 34465人阅读 评论(34) 收藏 举报

mfc编程socket服务器socketsstream

MFC下CSocket编程详解:

1. 常用的函数和注意事项(详细的函数接口说明请查看MSDN):

CSocket::Create 初始化(一般写服务器程序都不要用为好,用下面的 CSocket::Socket 初始化)

CSocket::Socket初始化

CSocket::SetSockOpt 设置socket选项

CSocket::Bind 绑定地址端口

CSocket::Connect 连接

CSocket::Listen  监听

CSocket::Accept 接收外部连接的socket
 
    CSocket::Send 发送内容

CSocket::Receive 接收内容

CSocket::Close 关闭(不等于delete)

1) 在使用MFC编写socket程序时,必须要包含<afxsock.h>都文件。

2) AfxSocketInit() 这个函数,在使用CSocket前一定要先调用该函数,否则使用CSocket会出错;并且该函数还有一个重要的使用方式,
       就是在某个线程下使用 CSocket 前一定要调用,就算主线程调用了该函数,在子线程下使用 CSocket 也要先调用该函数,要不会出错。

3) 还要注意的是, Create 方法已经包含了 Bind 方法,如果是以 Create 方法初始化的前提下不能再调用 Bind ,要不一定出错。

2. 以下是使用例子代码,通过例子来学习如何使用 CSocket 进行编程, 并且附件上有完整的例子代码。例子的可以在我的发布资源中找到:MFC下CSocket编程例子 http://download.csdn.net/source/379597

1) 客户端主要代码:

 //初始化
 AfxSocketInit();
  
        //创建 CSocket 对象
 CSocket aSocket;

 CString strIP;
 CString strPort;
 CString strText;

 this->GetDlgItem(IDC_EDIT_IP)->GetWindowText(strIP);
 this->GetDlgItem(IDC_EDIT_PORT)->GetWindowText(strPort);
 this->GetDlgItem(IDC_EDIT_TEXT)->GetWindowText(strText);

 //初始化 CSocket 对象, 因为客户端不需要绑定任何端口和地址, 所以用默认参数即可
 if(!aSocket.Create())
 {
  char szMsg[1024] = {0};

  sprintf(szMsg, "create faild: %d", aSocket.GetLastError());

  AfxMessageBox(szMsg);
  return;
 }

 //转换需要连接的端口内容类型
 int nPort = atoi(strPort);
 
        //连接指定的地址和端口
 if(aSocket.Connect(strIP, nPort))
 {
  char szRecValue[1024] = {0};

                //发送内容给服务器
  aSocket.Send(strText, strText.GetLength());
  
  //接收服务器发送回来的内容(该方法会阻塞, 在此等待有内容接收到才继续向下执行)
  aSocket.Receive((void *)szRecValue, 1024);

  AfxMessageBox(szRecValue);
 }
 else
 {
  char szMsg[1024] = {0};
  
  sprintf(szMsg, "create faild: %d", aSocket.GetLastError());
  
  AfxMessageBox(szMsg);
 }

 //关闭
 aSocket.Close();

2)服务器端代码:

unsigned int StartServer(LPVOID lParam)
{

//初始化Winscok
    if (!AfxSocketInit())
    {
        AfxMessageBox(IDP_SOCKETS_INIT_FAILED);
        return 1;
    }

    m_exit = false;

    CServerDlg *aDlg = (CServerDlg *)lParam;

    CString strPort;
    
    aDlg->GetDlgItemText(IDC_EDIT_PORT, strPort);
    
    UINT nPort = atoi(strPort);
    
    //socket------------------------------------------------
    
    CSocket aSocket, serverSocket;
    //最好不要使用aSocket.Create创建,因为容易会出现10048错误
    if (!aSocket.Socket())
    {
        char szError[256] = {0};
        
        sprintf(szError, "Create Faild: %d", GetLastError());
        
        AfxMessageBox(szError);
        
        return 1; 
    }

    BOOL bOptVal = TRUE;
    int bOptLen = sizeof(BOOL);

     //设置Socket的选项, 解决10048错误必须的步骤
    aSocket.SetSockOpt(SO_REUSEADDR, (void *)&bOptVal, bOptLen, SOL_SOCKET);

//监听
    if(!aSocket.Listen(10))
    {    
        char szError[256] = {0};
        
        sprintf(szError, "Listen Faild: %d", GetLastError());
        
        AfxMessageBox(szError);
        
        return 1;
    }
    
    CString strText;
    
    aDlg->GetDlgItemText(IDC_EDIT_LOG, strText);
    
    strText += "Server Start!  ";
    
    aDlg->SetDlgItemText(IDC_EDIT_LOG, strText);

    while(!m_exit)
    {
        //接收外部连接
        if(!aSocket.Accept(serverSocket))
        {
            continue;
        }
        else
        {
            char szRecvMsg[256] = {0};
            char szOutMsg[256] = {0};    

//接收客户端内容:阻塞
            serverSocket.Receive(szRecvMsg, 256);

            sprintf(szOutMsg, "Receive Msg: %s  ", szRecvMsg);
            
            aDlg->GetDlgItemText(IDC_EDIT_LOG, strText);
            
            strText += szOutMsg;
            
            aDlg->SetDlgItemText(IDC_EDIT_LOG, strText);

//发送内容给客户端
            serverSocket.Send("Have Receive The Msg", 50);

//关闭
            serverSocket.Close();
        }
        
    }

//关闭
    aSocket.Close();
    serverSocket.Close();
    
    aDlg->GetDlgItemText(IDC_EDIT_LOG, strText);
    
    strText += "Have Close!";
    
    aDlg->SetDlgItemText(IDC_EDIT_LOG, strText);

    return 0;
}

    
    //绑定端口
    if (!aSocket.Bind(nPort))
    {
        char szError[256] = {0};
            
        sprintf(szError, "Bind Faild: %d", GetLastError());
            
        AfxMessageBox(szError);
            
        return 1; 
    }

3) SDK 下的服务器端代码

       //子线程函数
       unsigned int StartServer(LPVOID lParam)
       {
        
       //初始化Winsock, AfxSocketInit() 也是封装了这些语句, 不过 AfxSocketInit() 所做的事比这里多些

    WSADATA wsaData;
   
           //Winsock 的版本, 建议用1.1 ,兼容性好
    WORD wVersionRequested = MAKEWORD(1, 1);
    int nResult = WSAStartup(wVersionRequested, &wsaData);
    if (nResult != 0)
    {
  return 1;
    }

       //----------------------------------------------------- 

           m_exit = false;

    CServerDlg *aDlg = (CServerDlg *)lParam;

    CString strPort;
 
    aDlg->GetDlgItemText(IDC_EDIT_PORT, strPort);
 
    UINT nPort = atoi(strPort);
 
 //socket------------------------------------------------
           
           //接口对象
           SOCKET aSocket, serverSocket;

           //寻址相关结构
    sockaddr_in serverSockaddr;
    memset(&serverSockaddr, 0, sizeof(serverSockaddr));


    aSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);

    if (aSocket == INVALID_SOCKET)
    {
  char szError[256] = {0};
  
  sprintf(szError, "Create Faild: %d", GetLastError());
  
  AfxMessageBox(szError);
  
  return 1; 
    }

    //注意,该处非常重要,取值的正确与否决定关闭scoket后端口是否能正常释放
    BOOL bOptVal = TRUE;
    int bOptLen = sizeof(BOOL);
           
            //设置 socket 选项, SOL_SOCKET 和 SO_REUSEADDR 一起使用, 并且后面的参数如上, 
              关闭scoket后端口便能正常释放
     setsockopt(aSocket, SOL_SOCKET, SO_REUSEADDR, (char *)&bOptVal, bOptLen); 
 
           //寻址相关结构
    sockaddr_in aSockaddr;
    memset(&aSockaddr,0,sizeof(aSockaddr));

    aSockaddr.sin_family = AF_INET;

    aSockaddr.sin_addr.s_addr = htonl(INADDR_ANY);
 
    aSockaddr.sin_port = htons((u_short)nPort);
 
           //绑定: 注意参数的类型转换
    if(bind(aSocket,(sockaddr *)&aSockaddr, sizeof(aSockaddr)) == SOCKET_ERROR)
    {
  char szError[256] = {0};
  
  sprintf(szError, "Bind Faild: %d", GetLastError());
  
  AfxMessageBox(szError);
  
  return 1; 
    }

 
    //监听
    if(listen(aSocket, 10) == SOCKET_ERROR)
    { 
  char szError[256] = {0};
  
  sprintf(szError, "Listen Faild: %d", GetLastError());
  
  AfxMessageBox(szError);
  
  return 1;
    }
 
    CString strText;

    aDlg->GetDlgItemText(IDC_EDIT_LOG, strText);

    strText += "Server Start!  ";

    aDlg->SetDlgItemText(IDC_EDIT_LOG, strText);

    while(!m_exit)
    {
  //接收外部连接, 非阻塞
  serverSocket = accept(aSocket, (sockaddr *)&serverSockaddr, 0);
   
  if(serverSocket == INVALID_SOCKET)
  {
   continue;
  }
  else
  {
   char szRecvMsg[256] = {0};
   char szOutMsg[256] = {0}; 
   
   //接收客户端内容: 阻塞
   recv(serverSocket, szRecvMsg, 256, 0);

   sprintf(szOutMsg, "Receive Msg: %s  ", szRecvMsg);
   
   aDlg->GetDlgItemText(IDC_EDIT_LOG, strText);
   
   strText += szOutMsg;

   aDlg->SetDlgItemText(IDC_EDIT_LOG, strText);

                        //发送内容给客户端
   send(serverSocket, "Have Receive The Msg", 50, 0);

                        //关闭
   closesocket(serverSocket);
  }
  
    }
 
    //关闭
    closesocket(aSocket);
    closesocket(serverSocket);

    aDlg->GetDlgItemText(IDC_EDIT_LOG, strText);
 
    strText += "Have Close!";

    aDlg->SetDlgItemText(IDC_EDIT_LOG, strText);

           //当你使用完Winsock接口后,要调用下面的函数对其占用的资源进行释放
    WSACleanup();

    return 0;
       }

3. 总结
   1) MFC进行编程的确比较简单, 用的代码比较少, 又容易管理。唯一不好的地方在于很多细节上的东西在资料上不容易查出来, 关联性非常紧密, 象 AfxSocketInit() 函数就是,函数的实现里包含着很多不容易理解的类, 并且记录了非常多的环境信息, 比如创建的线程等等, 这样在主线程调用后子线程没有调用执行 CSocket 的操作就会出错。还有就是有些接口的设计非常离奇, 象 CSocket::Create 的接口就是, 实现上还执行了 CSocket::Bind , 非常不容易被发现。并且MSDN上对 CSocket::Bind 的说明又明显的提示需要显示执行 CSocket::Bind 操作。

2) SDK 编程能理解函数的调用顺序和代码的结构就比较容易,省去了MFC下封装了不知道什么东西的部分,使得代码的流程容易控制。但是从上面的例子来看非常明显的并且不是那么容易理解。不仅仅有很多奇怪的结构(微软的命名一直如此, 无所云云), 并且函数相关太过于紧密, 初学者想一下子熟悉使用并不容易, 对开发者来说代码管理起来非常麻烦。

时间: 2024-11-06 12:55:17

MFC下CSocket编程详解的相关文章

[顶]ORACLE PL/SQL编程详解之二:PL/SQL块结构和组成元素(为山九仞,岂一日之功)

原文:[顶]ORACLE PL/SQL编程详解之二:PL/SQL块结构和组成元素(为山九仞,岂一日之功) [顶]ORACLE PL/SQL编程详解之二: PL/SQL块结构和组成元素(为山九仞,岂一日之功) 继上四篇:ORACLE PL/SQL编程之八:把触发器说透                ORACLE PL/SQL编程之六:把过程与函数说透(穷追猛打,把根儿都拔起!)                [推荐]ORACLE PL/SQL编程之四:把游标说透(不怕做不到,只怕想不到) [推荐]

[推荐]ORACLE PL/SQL编程详解之一:PL/SQL 程序设计简介(千里之行,始于足下)

原文:[推荐]ORACLE PL/SQL编程详解之一:PL/SQL 程序设计简介(千里之行,始于足下) [推荐]ORACLE PL/SQL编程详解之一: PL/SQL 程序设计简介(千里之行,始于足下) ——通过知识共享树立个人品牌. 继上六篇: [顶]ORACLE PL/SQL编程详解之二:PL/SQL块结构和组成元素(为山九仞,岂一日之功) [推荐]ORACLE PL/SQL编程详解之三:PL/SQL流程控制语句(不给规则,不成方圆) [推荐]ORACLE PL/SQL编程之四:把游标说透(

Qt Quick 之 QML 与 C++ 混合编程详解

Qt Quick 技术的引入,使得你能够快速构建 UI ,具有动画.各种绚丽效果的 UI 都不在话下.但它不是万能的,也有很多局限性,原来 Qt 的一些技术,比如低阶的网络编程如 QTcpSocket ,多线程,又如 XML 文档处理类库 QXmlStreamReader / QXmlStreamWriter 等等,在 QML 中要么不可用,要么用起来不方便,所以呢,很多时候我们是会基于这样的原则来混合使用 QML 和 C++: QML 构建界面, C++ 实现非界面的业务逻辑和复杂运算. 请给

iOS定位服务编程详解

现在的移动设备很多都提供定位服务,使用iOS系统的iPhone.iPod Touch和iPad都可以提供位置服务,iOS设备能提供3种不同途径进行定位:Wifi, 蜂窝式移动电话基站, GPS卫星 iOS 不像Android系统在定位服务编程时,可以指定采用哪种途径进行定位.iOS的API把底层这些细节屏蔽掉了,开发人员和用户并不知道现在设备是采用 哪种方式进行定位的,iOS系统会根据设备的情况和周围的环境,采用一套最佳的解决方案.这个方案是这样的,如果能够接收GPS信息,那么设备优先采用 GP

Qt5串口编程详解【新版】

Qt5的串口比Qt4的好用得多,Qt4的貌似没有集成官方库. 之前我也写过Qt5的串口,不过有一些缺陷,这次试图改进.转载请保留链接:http://blog.csdn.net/qq363692146/article/details/26049355 本文发表于2014.5.17. 如果在linux下,记得使用root权限,可以用root权限打开可执行文件,或者用root权限打开Qt Creator.(原因是串口常常需要特权,有些串口有特权也只能度而不能写,这个可能是Qt本身的问题)[至于安卓端,

PHP面向对象编程详解:类和对象

PHP面向对象编程详解:类和对象 从OOP的视角看,不应区分语言.无论是C++.无论是Java.无论是.net还有更多面向对象的语言,只要你了解了OO的真谛,便可以跨越语言,让你的思想轻松的跳跃.便没有对于Java..net.PHP 之间谁强谁弱的争执了. 希望这个介绍PHP5面向对象编程(OOP)的资料能让初学者受益,能让更多的PHPer开始转向OO的编程过程. 相对PHP4,PHP5在面向对象方面改变了很多.我们将只介绍PHP5环境下的面向对象.而我们必须改变自己来跟随PHP5的发展.如果代

Java多线程编程详解

线程的同步 由于同一进程的多个线程共享同一片存储空间,在带来方便的同时,也带来了访问冲突这个严重的问题.Java语言提供了专门机制以解决这种冲突,有效避免了同一个数据对象被多个线程同时访问. 由于我们可以通过 private 关键字来保证数据对象只能被方法访问,所以我们只需针对方法提出一套机制,这套机制就是 synchronized 关键字,它包括两种用法:synchronized 方法和 synchronized 块. 1. synchronized 方法:通过在方法声明中加入 synch

xib Nib IB 可视化编程详解

Xibs 简单的说,Xib就是拖控件编程,也可以说是可视化编程.相对于代码,使用IB和xib文件来组织UI,可以省下大量代码和时间,从而得到更快的开发速度.如果你曾经受到过微软家Visual Basic或者其他Visual系的可视化界面的荼毒与残害,因此怀疑Interface Builder的纯正血统和工作能力,建议可以看看这些资料以纠正三观:Jean-Marie Hullot的Interface Builder神话以及西装革履的青涩乔帮主在NeXT时亲手用IB构建应用(需要翻墙).另外,不妨打

ORACLE PL/SQL编程详解

ORACLE PL/SQL编程详解 编程详解 SQL语言只是访问.操作数据库的语言,并不是一种具有流程控制的程序设计语言,而只有程序设计语言才能用于应用软件的开发.PL /SQL是一种高级数据库程序设计语言,该语言专门用于在各种环境下对ORACLE数据库进行访问.由于该语言集成于数据库服务器中,所以PL/SQL代码可以对数据进行快速高效的处理.除此之外,可以在ORACLE数据库的某些客户端工具中,使用PL/SQL语言也是该语言的一个特点.本章的主要内容是讨论引入PL/SQL语言的必要性和该语言的