我只想编写一个带有命名管道的简单多客户服务器。我有服务器端的问题。当我只阅读时,一切都很棒。现在,我添加了服务器向客户端发送消息的可能性 - 现在它根本不起作用......
继承我的服务器代码(注意,当我删除注释行/ * * /时,它从所有客户端读取,它可以正常工作但只读 - 只读)。如何强制服务器也写入所有客户端? (概念是客户端第一次写入发送到服务器客户端fifoname。服务器创建并打开这样的fifo并且可以写入但是写入部分仍然不想工作...:/)
int main (int argc, char *argv[])
{
int fd_in, fd_out, j, result;
char buffer[4096];
const char *myfifo = "./fifo";
mkfifo(myfifo, 0666 );
if ((fd_in = open(myfifo, O_RDONLY | O_NONBLOCK)) < 0)
perror(myfifo);
printf("Server reads from: %s\n", myfifo);
for (;;) {
fd_set fds_r;
fd_set master;
FD_ZERO (&fds_r);
FD_ZERO (&master);
FD_SET (fd_in, &fds_r);
master = fds_r;
if ((result = select(fd_in + 1, &fds_r, NULL, NULL, NULL)) < 0)
perror ("select()");
if (! FD_ISSET(fd_in, &fds_r))
continue;
result = read(fd_in, buffer, 4096);
printf("FROM CLIENT: %s\n", buffer);
/*if(startsWith(buffer, "#"))
{
// remove # from message from client
string login = removePrefix(buffer);
string fifoname = "./fifo";
fifoname += login;
printf("REGISTERED: %s\n", fifoname.c_str());
mkfifo(fifoname.c_str(), 0666);
if ((fd_out = open(fifoname.c_str(), O_WRONLY)) < 0)
perror(fifoname.c_str());
FD_SET (fd_out, &master);
}
/* for(j = 0; j <= fd_in; j++) {
// send to everyone!
if (FD_ISSET(j, &master)) {
// except the listener and ourselves
if (j != fd_out) {
if (write(j, buffer, sizeof(buffer)) == -1) {
perror("write");
}
}
}
}*/
memset(buffer, 0, sizeof(buffer));
}
fprintf (stderr, "Got EOF!\n");
close (fd_in);
return 0;
}
client.cpp
#include <stdio.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <string>
#include <string.h>
using namespace std;
int main(int argc, char **argv)
{
int client_to_server;
const char *myfifo = "./fifo";
int server_to_client;
string f = "./fifo";
string u(argv[1]);
f += u;
const char *myfifo2 = f.c_str();
char str[BUFSIZ];
/* write str to the FIFO */
client_to_server = open(myfifo, O_WRONLY);
server_to_client = open(myfifo2, O_RDONLY);
// register message #username
string reg = "#";
reg += u;
printf("Client writes to: %s\n", myfifo);
printf("Client reads from: %s\n", myfifo2);
// first write to server, he can now make a fifo called fifo+username
if(write(client_to_server, reg.c_str(), strlen(reg.c_str())) == -1)
perror("write:");
while(1)
{
printf("Input message to serwer: ");
scanf("%s", str);
if(write(client_to_server, str, sizeof(str))==-1)
perror("Write:"); //Very crude error check
if(read(server_to_client,str,sizeof(str))==-1)
perror("Read:"); // Very crude error check
printf("...received from the server: %s\n",str);
}
close(client_to_server);
close(server_to_client);
/* remove the FIFO */
return 0;
}
我的输出:
服务器端:
$ ./SERVER
Server reads from: ./fifo
FROM CLIENT: #username
FROM CLIENT: hello
FROM CLIENT:
FROM CLIENT: ?
FROM CLIENT:
FROM CLIENT: huh
FROM CLIENT:
客户端:
$ ./CLIENT username
Client writes to: ./fifo
Client reads from: ./fifousername
Input message to serwer: hello
Read:: Bad file descriptor
...received from the server: hello
Input message to serwer: ?
Read:: Bad file descriptor
...received from the server: ?
Input message to serwer: huh
Read:: Bad file descriptor
...received from the server: huh
Input message to serwer:
答案 0 :(得分:4)
首先,检查open
电话的错误。这很可能解释了你所问的问题。
我不清楚应该是什么“信息”。我认为它应该以{{1}}开头,但服务器如何知道它的结束位置?它总是4,096字节吗?
但是你有三个主要问题:
您的代码不应该在任何地方阻止,但它不会将描述符设置为非阻塞。
您的代码需要4,096字节的消息。但是如果它读取更少的字节,它不会等到它已经读取其余部分,而是会破坏消息。 (或者它所期望的任何消息,它显然可以超过管道可以原子处理的字节数。你需要以某种方式找到消息边界。)
您的代码无法处理部分写入。如果任何客户端读取速度不够快,它将会挂起(由于上面的问题1),但是一旦你解决了这个问题,它就会破坏数据。
您需要每个连接应用程序接收缓冲区。从连接读取数据时,将其读入应用程序缓冲区。如果您收到了整条消息,那么请处理该消息。否则,等到你读完剩下的时间。如果您不使用固定长度的消息,则必须处理读取一条消息的结尾和另一条消息的开头的情况。
您有两种写策略选择:
使用每个连接的应用程序写入缓冲区。将数据发送到连接时,如果所有字节都不立即发送,请将剩余部分保存在应用程序缓冲区中。开始选择该描述符以进行写入和读取。如果你得到一个写入命中,请尝试从应用程序缓冲区写入。
仅使用内核缓冲区。如果进行部分写入,则内核缓冲区已满。断开客户端,因为它无法跟上。