小弟我新手,麻烦赐教VC中的小疑点
我新手,麻烦赐教VC中的小问题
int CNetCon::nreceive(int milliseconds,string &rcvdata)
{
char buf[1] = " ";
int rc=0;
int total = 0;
timeval TimeOut;
TimeOut.tv_sec = milliseconds / 1000;
TimeOut.tv_usec = milliseconds % 1000 * 1000;
fd_set SocketSet;
FD_ZERO(&SocketSet);
FD_SET(m_Socket, &SocketSet);
if(this-> m_errorIndex != -1)
{
return 0;
}
for(;;)
{
if(select(0, &SocketSet, 0, 0, &TimeOut) > 0)
rc = recv(m_Socket,buf,sizeof(buf),0);
..................
代码中的
TimeOut.tv_sec = milliseconds / 1000;
TimeOut.tv_usec = milliseconds % 1000 * 1000;
是什么意思,为什么右边的时间要这个样子表示啊?
if(select(0, &SocketSet, 0, 0, &TimeOut) > 0)
select()> 0意味着什么呢?
我新手,麻烦赐教
------解决方案--------------------
楼主好厉害,新手就接触到select了,厉害啊。
int select(int numfds, fd_set *readfds, fd_set *writefds,fd_set
*exceptfds, struct timeval *timeout);
这个函数监视一系列文件描述符,特别是 readfds、writefds 和 exceptfds。如果你想知道你是否能够从标准输入和套接字描述符 sockfd 读入数据,你只要将文件描述符 0 和 sockfd 加入到集合 readfds 中。参 数 numfds 应该等于最高的文件描述符的值加1。在这个例子中,你应该 设置该值为 sockfd+1。因为它一定大于标准输入的文件描述符 (0)。 当函数 select() 返回的时候,readfds 的值修改为反映你选择的哪个 文件描述符可以读。你可以用下面讲到的宏 FD_ISSET() 来测试。 在我们继续下去之前,让我来讲讲如何对这些集合进行操作。每个集 合类型都是 fd_set。下面有一些宏来对这个类型进行操作:
FD_ZERO(fd_set *set) – 清除一个文件描述符集合
FD_SET(int fd, fd_set *set) - 添加fd到集合
FD_CLR(int fd, fd_set *set) – 从集合中移去fd
FD_ISSET(int fd, fd_set *set) – 测试fd是否在集合中
最后,是有点古怪的数据结构 struct timeval。有时你可不想永远等待别人发送数据过来。也许什么事情都没有发生的时候你也想每隔96秒在终 端上打印字符串 "Still Going... "。这个数据结构允许你设定一个时间,如果 时间到了,而 select() 还没有找到一个准备好的文件描述符,它将返回让你继续处理。
数据结构 struct timeval 是这样的:
struct timeval {
int tv_sec; /* seconds */
int tv_usec; /* microseconds */
};
只要将 tv_sec 设置为你要等待的秒数,将 tv_usec 设置为你要等待的微秒数就可以了。是的,是微秒而不是毫秒。1,000微秒等于1毫秒,1,000 毫秒等于1秒。也就是说,1秒等于1,000,000微秒。为什么用符号 "usec " 呢? 字母 "u " 很象希腊字母 Mu,而 Mu 表示 "微 " 的意思。当然,函数 返回的时候 timeout 可能是剩余的时间,之所以是可能,是因为它依赖于 你的 Unix 操作系统。
哈!我们现在有一个微秒级的定时器!别计算了,标准的 Unix 系统 的时间片是100毫秒,所以无论你如何设置你的数据结构 struct timeval, 你都要等待那么长的时间。
还有一些有趣的事情:如果你设置数据结构 struct timeval 中的数据为 0,select() 将立即超时,这样就可以有效地轮询集合中的所有的文件描述 符。如果你将参数 timeout 赋值为 NULL,那么将永远不会发生超时,即一直等到第一个文件描述符就绪。最后,如果你不是很关心等待多长时间, 那么就把它赋为 NULL 吧。
------解决方案--------------------
TimeOut.tv_sec = milliseconds / 1000; //其中的秒数
TimeOut.tv_usec = milliseconds % 1000 * 1000; //微秒数
milliseconds的单位是毫秒
if(select(0, &SocketSet, 0, 0, &TimeOut) > 0)
表示可读,不过一般select的第一个参数是socket加1
int CNetCon::nreceive(int milliseconds,string &rcvdata)
{
char buf[1] = " ";
int rc=0;
int total = 0;
timeval TimeOut;
TimeOut.tv_sec = milliseconds / 1000;
TimeOut.tv_usec = milliseconds % 1000 * 1000;
fd_set SocketSet;
FD_ZERO(&SocketSet);
FD_SET(m_Socket, &SocketSet);
if(this-> m_errorIndex != -1)
{
return 0;
}
for(;;)
{
if(select(0, &SocketSet, 0, 0, &TimeOut) > 0)
rc = recv(m_Socket,buf,sizeof(buf),0);
..................
代码中的
TimeOut.tv_sec = milliseconds / 1000;
TimeOut.tv_usec = milliseconds % 1000 * 1000;
是什么意思,为什么右边的时间要这个样子表示啊?
if(select(0, &SocketSet, 0, 0, &TimeOut) > 0)
select()> 0意味着什么呢?
我新手,麻烦赐教
------解决方案--------------------
楼主好厉害,新手就接触到select了,厉害啊。
int select(int numfds, fd_set *readfds, fd_set *writefds,fd_set
*exceptfds, struct timeval *timeout);
这个函数监视一系列文件描述符,特别是 readfds、writefds 和 exceptfds。如果你想知道你是否能够从标准输入和套接字描述符 sockfd 读入数据,你只要将文件描述符 0 和 sockfd 加入到集合 readfds 中。参 数 numfds 应该等于最高的文件描述符的值加1。在这个例子中,你应该 设置该值为 sockfd+1。因为它一定大于标准输入的文件描述符 (0)。 当函数 select() 返回的时候,readfds 的值修改为反映你选择的哪个 文件描述符可以读。你可以用下面讲到的宏 FD_ISSET() 来测试。 在我们继续下去之前,让我来讲讲如何对这些集合进行操作。每个集 合类型都是 fd_set。下面有一些宏来对这个类型进行操作:
FD_ZERO(fd_set *set) – 清除一个文件描述符集合
FD_SET(int fd, fd_set *set) - 添加fd到集合
FD_CLR(int fd, fd_set *set) – 从集合中移去fd
FD_ISSET(int fd, fd_set *set) – 测试fd是否在集合中
最后,是有点古怪的数据结构 struct timeval。有时你可不想永远等待别人发送数据过来。也许什么事情都没有发生的时候你也想每隔96秒在终 端上打印字符串 "Still Going... "。这个数据结构允许你设定一个时间,如果 时间到了,而 select() 还没有找到一个准备好的文件描述符,它将返回让你继续处理。
数据结构 struct timeval 是这样的:
struct timeval {
int tv_sec; /* seconds */
int tv_usec; /* microseconds */
};
只要将 tv_sec 设置为你要等待的秒数,将 tv_usec 设置为你要等待的微秒数就可以了。是的,是微秒而不是毫秒。1,000微秒等于1毫秒,1,000 毫秒等于1秒。也就是说,1秒等于1,000,000微秒。为什么用符号 "usec " 呢? 字母 "u " 很象希腊字母 Mu,而 Mu 表示 "微 " 的意思。当然,函数 返回的时候 timeout 可能是剩余的时间,之所以是可能,是因为它依赖于 你的 Unix 操作系统。
哈!我们现在有一个微秒级的定时器!别计算了,标准的 Unix 系统 的时间片是100毫秒,所以无论你如何设置你的数据结构 struct timeval, 你都要等待那么长的时间。
还有一些有趣的事情:如果你设置数据结构 struct timeval 中的数据为 0,select() 将立即超时,这样就可以有效地轮询集合中的所有的文件描述 符。如果你将参数 timeout 赋值为 NULL,那么将永远不会发生超时,即一直等到第一个文件描述符就绪。最后,如果你不是很关心等待多长时间, 那么就把它赋为 NULL 吧。
------解决方案--------------------
TimeOut.tv_sec = milliseconds / 1000; //其中的秒数
TimeOut.tv_usec = milliseconds % 1000 * 1000; //微秒数
milliseconds的单位是毫秒
if(select(0, &SocketSet, 0, 0, &TimeOut) > 0)
表示可读,不过一般select的第一个参数是socket加1