- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
这是与套接字编程相关的查询,服务器正在使用 select() 函数来查找准备读取的套接字。
根据 select() 函数的手册页。简而言之,“select 函数允许程序监视多个网络套接字,这些套接字已准备好读取,哪些 fd 已准备好写入”(暂时保留异常(exception))。以下两个程序,一个用于服务器程序,一个用于客户端程序。
我在 stackoverflow 中搜索了很多文章后提出了这个查询其他网站。我没有找到任何相关的解决方案。
我在 select() 中使用了下面的 readfds 代码并且工作正常。
服务器:使用readfds
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h>
#include <arpa/inet.h>
#include <signal.h>
#include <string.h>
int main(void)
{
fd_set master;
fd_set read_fds; // file descriptor list used for select()
int fdmax;
int listener;
int newfd; // newly accepted socket descriptor
struct sockaddr_storage remoteaddr; // client address
socklen_t addrlen;
char remoteIP[INET6_ADDRSTRLEN], buf[20];
int yes=1; // for setsockopt() SO_REUSEADDR, below
int i, rv, nbytes;
struct addrinfo hints, *ai, *p;
FD_ZERO(&master);
FD_ZERO(&read_fds);
memset(&hints, 0, sizeof hints);
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
hints.ai_flags = AI_PASSIVE;
if ((rv = getaddrinfo(NULL, "6000", &hints, &ai)) != 0) {
fprintf(stderr, "selectserver: %s\n", gai_strerror(rv));
return 1;
}
for(p = ai; p != NULL; p = p->ai_next) {
listener = socket(p->ai_family, p->ai_socktype, p->ai_protocol);
if (listener < 0) {
continue;
}
if (setsockopt(listener, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)) == -1) {
perror("setsockopt");
return 1;
}
if (bind(listener, p->ai_addr, p->ai_addrlen) < 0)
{
close(listener);
continue;
}
break;
}
// if we got here, it means we didn't get bound
if (p == NULL) {
printf("selectserver: failed to bind\n");
return 1;
}
freeaddrinfo(ai); // all done with this
if (listen(listener, 5) == -1) {
perror("listen");
return 1;
}
FD_SET(listener, &master);
fdmax = listener; // remembering max fd
// main loop
for(;;)
{
read_fds = master; // copying to read fds it
if (select(fdmax+1, &read_fds, NULL, NULL, NULL) == -1) {
perror("select");
return 1;
}
// run through the existing connections looking for data to read
for(i = fdmax; i >= 0; i--)
{
if (FD_ISSET(i, &read_fds))
{ // we got one!!
if (i == listener)
{
// handle new connections
addrlen = sizeof remoteaddr;
newfd = accept(listener,
(struct sockaddr *)&remoteaddr,
&addrlen);
printf("selectserver: new connection from %s on socket %d\n",
inet_ntop(remoteaddr.ss_family,
&(((struct sockaddr_in *)&remoteaddr)->sin_addr),
remoteIP, INET6_ADDRSTRLEN),
listener);
//if (nbytes = send(newfd, "something", strlen("something"), 0) == -1)
if ((nbytes = recv(newfd, buf, 20, 0)) <= 0)
printf("send failed\n");
else
{
buf[nbytes] = '\0';
printf("%s\n", buf);
}
} // END handle data from client
} // END got new incoming connection
} // END looping through file descriptors
close(newfd);
}
close(listener);
return 0;
}
客户:
#include <stdio.h>
#include <unistd.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <string.h>
#include <arpa/inet.h>
int main(int argc, char *argv[])
{
int clientSocket;
struct sockaddr_in serverAddr;
socklen_t addr_size;
char buf[20];
int nbytes;
int rv;
struct addrinfo hints, *ai, *p;
memset(&hints, 0, sizeof hints);
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
//hints.ai_flags = AI_PASSIVE;
if ((rv = getaddrinfo(NULL, "6000", &hints, &ai)) != 0) {
fprintf(stderr, "selectserver: %s\n", gai_strerror(rv));
return 1;
}
for(p = ai; p != NULL; p = p->ai_next) {
clientSocket = socket(p->ai_family, p->ai_socktype, p->ai_protocol);
if (clientSocket < 0) {
printf("error");
continue;
}
break;
}
addr_size = sizeof serverAddr;
if (connect(clientSocket, (struct sockaddr *) p->ai_addr, p->ai_addrlen) == -1)
{
printf("Connect failed\n");
return 1;
}
printf("connect success\n");
//if ((nbytes = recv(clientSocket, buf, 20, 0)) <= 0)
if ((nbytes = send(clientSocket, "something", strlen("something"), 0)) == -1)
{
printf("failed, Number of bytes: %d\n", nbytes);
return 1;
}
return 0;
}
我在需要的地方添加了评论。
在服务器中,select()函数用于readfds(当其中一个readfds被设置时它将返回),一旦返回,我将在接受套接字后从套接字读取。该程序现在正在按预期运行
在服务器程序中:
//if (nbytes = send(newfd, "something", strlen("something"), 0) == -1) <br>
if ((nbytes = recv(newfd, buf, 20, 0)) <= 0)
在客户端程序中:
//if ((nbytes = recv(clientSocket, buf, 20, 0)) <= 0)<br>
if ((nbytes = send(clientSocket, "something", strlen("something"), 0)) == -1)
我已取消注释服务器中的发送语句和客户端中的接收语句。就像下面这样,
在服务器程序中:
if (nbytes = send(newfd, "something", strlen("something"), 0) == -1)<br>
//if ((nbytes = recv(newfd, buf, 20, 0)) <= 0)
在客户端程序中:
if ((nbytes = recv(clientSocket, buf, 20, 0)) <= 0)<br>
//if ((nbytes = send(clientSocket, "something", strlen("something"), 0)) == -1)
现在程序被修改为服务器正在发送数据而客户端正在接收(请记住我在 select() 函数中使用“readfds”)。修改后的程序完全可以正常工作。由此我们可以说,select()函数中的参数'readfds'正在设置准备写入的fd,并且也在fd_set中设置这些fd准备写入。
**现在,在修改后的代码中,进行以下更改并保持其余代码不变,
if (select(fdmax+1, &read_fds, NULL, NULL, NULL) == -1) {<br>
替换为下面的行,
if (select(fdmax+1, NULL, &read_fds, NULL, NULL) == -1) {<br>
上述更改将使 select() 函数在找到准备写入的套接字时返回(从等待状态)。 (根据功能)。但是,“select()”根本不返回并等待套接字向其写入数据。并且在这里被阻止..这不符合预期。**
主要要求是,我有一台服务器在一台机器上运行,并等待在同一台机器上运行的两种类型的客户端套接字,这些套接字已准备好读取和写入。为了实现这一点,我通过指定 readfds(select() 中的第二个参数)和 writefds(select() 中的第三个参数)修改了 select 语句。哪个不起作用(即 writefds 未设置,问题已在前面解释过)。主要问题是,我无法区分套接字(准备好读写 fds),因为 select() 函数仅在 readfds 中捕获读写 sock。
这是我在 stackoverflow 上的第一篇文章。所以,我尽量解释清楚,避免数据不足。如果需要添加/修改任何内容,请评论。
最佳答案
当您选择writefds
参数时,该集合不包含任何准备写入的套接字。它不包含任何可供写入的套接字,因为您从未在 master
集中添加新的已接受套接字。
从技术上讲,您不会检查新接受的连接是否已准备好读取。
master
集中唯一的套接字是监听套接字,并且它永远不可写。
解决方案:在接受套接字后将套接字添加到 master
集中。
顺便说一句,在检查就绪套接字时,您不应该真正循环遍历所有可能的文件描述符。只需执行例如
if (select(...) < 0)
{
// Handle error
}
if (FD_ISSET(listener, &read_fds))
{
// Handle new connection
}
要了解哪些已连接的客户端已发送数据,请将已连接的客户端保留在列表中并仅循环遍历该列表。
关于c - 在 C 编程中的 select() 函数中使用 'writefds' 参数时出现问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30846829/
我的应用程序从一个有 5 个选项卡的选项卡栏 Controller 开始。一开始,第一个出现了它的名字,但其他四个没有名字,直到我点击它们。然后根据用户使用的语言显示名称。如何在选项卡栏出现之前设置选
我有嵌套数组 json 对象(第 1 层、第 2 层和第 3 层)。我的问题是数据表没有出现。任何相关的 CDN 均已导入。该表仅显示部分。我引用了很多网站,但都没有解决我的问题。 之前我使用标准表来
我正在尝试设置要显示的 Parse PFLoginViewController。这是我的一个 View Controller 的类。 import UIKit import Parse import
我遇到了这个问题,我绘制的对象没有出现在 GUI 中。我知道它正在被处理,因为数据被推送到日志文件。但是,图形没有出现。 这是我的一些代码: public static void main(Strin
我有一个树状图,其中包含出现这样的词...... TreeMap occurrence = new TreeMap (); 字符串 = 单词 整数 = 出现次数。 我如何获得最大出现次数 - 整数,
因此,我提示用户输入变量。如果变量小于 0 且大于 10。如果用户输入 10,我想要求用户再次输入数字。我问时间的时候输入4,它说你输入错误。但在第二次尝试时效果很好。例如:如果我输入 25,它会打印
我已经用 css overflow 属性做了一个例子。在这个例子中我遇到了一个溢出滚动的问题。滚动条出现了,但没有工作意味着每当将光标移动到滚动条时,在这个滚动条不活动的时间。我对此一无所知,所以请帮
我现在正在做一个元素。当您单击一个元素时,会出现以下信息,我想知道如何在您单击下一个元素而不重新单击同一元素时使其消失....例如,我的元素中有披萨,我想单击肉披萨看到浇头然后点击奶酪披萨看到浇头和肉
我有一个路由器模块,它将主题与正则表达式进行比较,并将出现的事件与一致的键掩码链接起来。 (它是一个简单的 url 路由过滤,如 symfony http://symfony.com/doc/curr
这个问题在这里已经有了答案: 9年前关闭。 Possible Duplicate: mysql_fetch_array() expects parameter 1 to be resource, bo
我在底部有一个带有工具栏的 View ,我正在使用 NavigationLink 导航到该 View 。但是当 View 出现时,工具栏显示得有点太低了。大约半秒钟后,它突然跳到位。它只会在应用程序启
我试图在我的应用程序上为背景音乐添加一个 AVAudioPlayer,我正在主屏幕上启动播放器,尝试在应用程序打开时开始播放但出现意外行为... 它播放并立即不断创建新玩家并播放这些玩家,因此同时播放
这是获取一个数字,获取其阶乘并将其加倍,但是由于基本情况,如果您输入 0,它会给出 2 作为答案,因此为了绕过它,我使用了 if 语句,但收到错误输入“if”时解析错误。如果你们能提供帮助,我真的很感
暂停期间抛出异常 android.os.DeadObjectException 在 android.os.BinderProxy.transactNative( native 方法) 在 androi
我已经为猜词游戏编写了一些代码。它从用户输入中读取字符并在单词中搜索该字符;根据字符是否在单词中,程序返回并控制一些变量。 代码如下: import java.util.Random; import
我是自动化领域的新手。这是我的简单 TestNG 登录代码,当我以 TestNG 身份运行该代码时,它会出现 java.lang.NullPointerException,双击它会突出显示我导航到 U
我是c#程序员,我习惯了c#的封装语法和其他东西。但是现在,由于某些原因,我应该用java写一些东西,我现在正在练习java一天!我要创建一个为我自己创建一个虚拟项目,以便让自己更熟悉 Java 的
我正在使用 Intellij,我的源类是 main.com.coding,我的资源文件是 main.com.testing。我将 spring.xml 文件放入资源文件中。 我的测试类位于 test.
我想要我的tests folder separate到我的应用程序代码。我的项目结构是这样的 myproject/ myproject/ myproject.py moduleon
这个问题已经有答案了: What is a NullPointerException, and how do I fix it? (12 个回答) 已关闭 6 年前。 因此,我尝试比较 2 个值,一个
我是一名优秀的程序员,十分优秀!