我的理解:把服务器和客户端的交互工程比喻成外来人员访问公司,每来一个客户端访问,需要服务器的前台经理接待此客户,然后前台经理呼叫一个接待员来将客户带上楼。服务器的两个角色前台经理和接待员就是服务器的两个CSocket对象。

1、需要生成两个类对象,一个用来监听客户的访问,一个用来接待客户。

  在类向导中新建类名:CSockL和CSockC,重写CSockL的OnAccept函数和CSockC的OnReceive和OnClose函数:

  CSockL(1)、

    void CSockL::OnAccept(int nErrorCode)//有客户连接,就回调此函数

    {

    // TODO: 在此添加专用代码和/或调用基类

    CSockC *p=new CSockC;//每有一个访问,就生成一个CSockC对象指针

    this->Accept(*p);

    //p->GetPeerName()可以用来获取刚刚连接的客户端ip和端口

    CSocket::OnAccept(nErrorCode);

    }

  CSockC(2)、

    void CSockC::OnReceive(int nErrorCode)//有客户发送消息,就回调

    {

    // TODO: 在此添加专用代码和/或调用基类

    char rcv[100];//装从客户端传来的数据

    int rcvLenth=Receive(rcv,sizeof(rcv));//核心处

    rcv[rcvLenth]=0;//消除乱码

    CString ipAddrss;//装发送消息的客户端ip地址

    UINT ipPort;//装端口

    int result=GetPeerName(ipAddrss,ipPort);//获取发送消息客户端的ip和端口

    CTestCSoketServerDlg *p=(CTestCSoketServerDlg*)AfxGetMainWnd();//获取主窗口类指针

    p->ShowInfo(rcv,ipAddrss,ipPort);//调用主窗口类指针的显示函数

    CSocket::OnReceive(nErrorCode);

    }

    (3)、

    void SocketS::OnClose(int nErrorCode)//有客户端断开连接回调

    {

    // TODO: 在此添加专用代码和/或调用基类

    CSocket::OnClose(nErrorCode);

    }

2、现在已经得到一个我自己定义的类CSockL:CSocket,在Dlg.h中声明一个CSockL的对象,在Dlg.cpp中的打开服务器button中定义

  if(!m_sockl.Create(8080))

  {

  AfxMessageBox(CString("创建套接字失败"));

  return;

  }

  m_sockl.Listen();

  注意:服务器不需要指定ip,指定端口创建套接字之后,就可以用于捕获所有本地客户端活动

  所以直接create之后listen就相当于是服务器了。

  另外使用Create之后不能再使用Bind()函数

3、如果需要从服务器传数据到客户端

  这时所有的客户端都长连接在了服务器上面,在Accept中可以得到这些客户端的ip,所以只要有ip就可以用sendto来发送了。

 笔记:

客户端套接字使用Send()函数的时候,是将数据放入TCP系统共有的缓冲区,然后TCP系统在服务器将缓冲区的数据接收,使用的是Receive()函数。在服务器接收的过程中,客户端会一直处于阻塞状态,只有当服务器将缓冲区里面的数据全部接收完了之后,客户端才会开始下一次Send()。

CSocket与线程配合使用:

发送端:使用AfxBeginThread()生成一个新的线程,在这个线程里面Send()数据就行了。这里应该加入一个确认发送标志,即每发送完一个包之后,就进入一个死循环,当服务器处理了这个包的数据后,向客户端发送一个确认包,客户端解析这个包之后,退出死循环,再进行下一个包的处理。

接收端:先把数据全部接收到一个buf里面,注意这个buf应尽可能大一点,否则可能会出现粘包的情况。然后通过判断如果是需要的客户端端口发来数据,就进入AfxBeginThread()线程,在这个线程里面处理数据。处理完了之后,想客户端发送确认包,进行下一次的数据处理。

根据上面的原理,可以实现CSocket+线程传送文件




长风破浪会有时,直挂云帆济沧海!