C++ 网络编程 总结

所属分类: 软件编程 / C 语言 阅读数: 89
收藏 0 赞 0 分享

第一次用C++写程序,对C++ 只是菜鸟级别的,倒是对C#很熟悉。两者有很大的相似性。但也有不同。

首先写了一个网络通讯用的小的MFC程序。发现

(1)MFC写界面真的好麻烦呀。用C#写的tab 分分钟搞定的事,用C++害得我写了两天.关键是不熟练. 还有list control 控件的图标显示.  真是很麻烦

         不过,由于最后的 detch() 函数执行后,就真正显示出来了.这些具体的小细节,一般在 书上都没有写.

(2)用C++ 写类的特征,基本上与C#是相似的.

说一下网络编程的问题吧

1\一开始并不清楚 C++ 写程序用的网络套接字,三类的不一样的地方.总以为我用的VS2013,用最高级别的套接字应该更容易一些,于是选用了CSOCKET 结果由于这个套接字是阻塞模式,结果被卡住了,卡住不知道如何做了.如果对方设备没有反应,最不能死等吧,这样不行.

于上网上搜索,发现一篇文章写到 给 CSocket 加上超时.于时照抄照搬着做了一遍,可惜失败了.  怎么查也查不出原因.  按原文章一字一句的比较,也没有找出原因来.失败换思路

2\想到低一点的 CAsyncSocket 是异步操作的.这样总可以了吧不会阻塞了吧.但是回调函数使得处理起来也不方便.在什么时候做处理,就需要消息做处理.但是也很麻烦.

3\于是,找出书来,大部分书上对于网络部分只是介绍了一个最基本的 Socket ,看起来也挺简单的. 就先试一下这个最基本的吧.

没想到 30分钟后,网络程序测试成功.  而且有超时接收,超时发送,等.正合我意.

原来最基本的,才是最好的.

总结一下:

     SOCKET的操作方法

以下是一个网络客户端的例子:

// client.cpp 
 
#include <iostream> 
#include <cstdio> 
#include <Winsock2.h> 
 
using namespace std; 
 
int main() 
{ 
// 加载socket动态链接库(dll) 
  WORD wVersionRequested; 
  WSADATA wsaData;  // 这结构是用于接收Wjndows Socket的结构信息的 
  int err; 
    
  wVersionRequested = MAKEWORD( 1, 1 );  // 请求1.1版本的WinSock库 
    
  err = WSAStartup( wVersionRequested, &wsaData ); 
  if ( err != 0 ) { 
    return -1;     // 返回值为零的时候是表示成功申请WSAStartup 
  } 
    
  if ( LOBYTE( wsaData.wVersion ) != 1 || HIBYTE( wsaData.wVersion ) != 1 ) { 
    // 检查这个低字节是不是1,高字节是不是1以确定是否我们所请求的1.1版本 
    // 否则的话,调用WSACleanup()清除信息,结束函数 
    WSACleanup( ); 
    return -1;  
  } 
   
// 创建socket操作,建立流式套接字,返回套接字号sockClient 
  // SOCKET socket(int af, int type, int protocol); 
  // 第一个参数,指定地址簇(TCP/IP只能是AF_INET,也可写成PF_INET) 
  // 第二个,选择套接字的类型(流式套接字),第三个,特定地址家族相关协议(0为自动) 
  SOCKET sockClient = socket(AF_INET, SOCK_STREAM, 0); 
 
// 将套接字sockClient与远程主机相连 
  // int connect( SOCKET s, const struct sockaddr* name, int namelen); 
  // 第一个参数:需要进行连接操作的套接字 
  // 第二个参数:设定所需要连接的地址信息 
  // 第三个参数:地址的长度 
  SOCKADDR_IN addrSrv; 
  addrSrv.sin_addr.S_un.S_addr = inet_addr("127.0.0.1");   // 本地回路地址是127.0.0.1;  
  addrSrv.sin_family = AF_INET; 
  addrSrv.sin_port = htons(6000); 
  connect(sockClient, (SOCKADDR*)&addrSrv, sizeof(SOCKADDR)); 
 
  char recvBuf[100]; 
  recv(sockClient, recvBuf, 100, 0); 
  printf("%s\n", recvBuf); 
 
  send(sockClient, "Attention: A Client has enter...\n", strlen("Attention: A Client has enter...\n")+1, 0); 
 
  printf("我们可以聊五句话"); 
  int n = 5; 
  do{ 
    printf("\n还剩%d次:", n); 
    char talk[100]; 
    printf("\nPlease enter what you want to say next(\"quit\"to exit):"); 
    gets(talk); 
    send(sockClient, talk, strlen(talk)+1, 0);     // 发送信息 
 
    char recvBuf[100]; 
    recv(sockClient, recvBuf, 100, 0); 
    printf("%s Says: %s\n", "Server", recvBuf);   // 接收信息 
  }while(--n); 
 
  printf("End linking...\n"); 
  closesocket(sockClient); 
  WSACleanup();  // 终止对套接字库的使用 
 
  printf("\n"); 
  system("pause"); 
  return 0; 
}

关于超时的处理方法

在send(),recv()过程中有时由于网络状况等原因,收发不能预期进行,而设置收发超时控制:

在Linux下需要注意的是时间的控制结构是struct timeval而并不是某一整型数,

在windows下是这样写的:

int nNetTimeout=1000;//1秒,
//设置发送超时
setsockopt(socket,SOL_SOCKET,SO_SNDTIMEO,(char *)&nNetTimeout,sizeof(int));
//设置接收超时
setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(char *)&nNetTimeout,sizeof(int));

这样做在Linux环境下是不会产生效果的,须如下定义:

struct timeval timeout = {3,0}; 
//设置发送超时
setsockopt(socket,SOL_SOCKET,SO_SNDTIMEO,(char *)&timeout,sizeof(struct timeval));

//设置接收超时
setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(char *)&timeout,sizeof(struct timeval));

有两点注意就是:

1)recv ()的第四个参数需为MSG_WAITALL(设置MSG_DONTWAIT可以不用阻塞在建立连接后在等等接收数据),在阻塞模式下不等到指定数目的数据不会返回,除非超时时间到。还要注意的是只要设置了接收超时,在没有MSG_WAITALL时也是有效的。说到底超时就是不让你的程序老在那儿等,到一定时间进行一次返回而已。

2)即使等待超时时间值未到,但对方已经关闭了socket, 则此时recv()会立即返回,并收到多少数据返回多少数据。

以上所述就是本文的全部内容了,希望大家能够喜欢。

更多精彩内容其他人还在看

用标准c++实现string与各种类型之间的转换

这个类在头文件中定义, < sstream>库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本
收藏 0 赞 0 分享

C++如何通过ostringstream实现任意类型转string

再使用整型转string的时候感觉有点棘手,因为itoa不是标准C里面的,而且即便是有itoa,其他类型转string不是很方便。后来去网上找了一下,发现有一个好方法
收藏 0 赞 0 分享

C/C++指针小结

要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区
收藏 0 赞 0 分享

C++ 类的静态成员深入解析

在C++中类的静态成员变量和静态成员函数是个容易出错的地方,本文先通过几个例子来总结静态成员变量和成员函数使用规则,再给出一个实例来加深印象
收藏 0 赞 0 分享

C++类的静态成员初始化详细讲解

通常静态数据成员在类声明中声明,在包含类方法的文件中初始化.初始化时使用作用域操作符来指出静态成员所属的类.但如果静态成员是整型或是枚举型const,则可以在类声明中初始化
收藏 0 赞 0 分享

C++类静态成员与类静态成员函数详解

静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
收藏 0 赞 0 分享

C++中的friend友元函数详细解析

友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类。友元函数的特点是能够访问类中的私有成员的非成员函数。友元函数从语法上看,它与普通函数一样,即在定义上和调用上与普通函数一样
收藏 0 赞 0 分享

static全局变量与普通的全局变量的区别详细解析

以下是对static全局变量与普通的全局变量的区别进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助
收藏 0 赞 0 分享

C++ explicit关键字的应用方法详细讲解

C++ explicit关键字用来修饰类的构造函数,表明该构造函数是显式的,既然有"显式"那么必然就有"隐式",那么什么是显示而什么又是隐式的呢?下面就让我们一起来看看这方面的知识吧
收藏 0 赞 0 分享

教你5分钟轻松搞定内存字节对齐

随便google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则
收藏 0 赞 0 分享
查看更多