如果我在一段时间后发送缓冲区,我的客户端将正确地发送到服务器。但是,如果我在没有间隔的情况下发送消息,服务器会收到一个缓冲区。例如,如果我发送两个缓冲区“message1”和“message2”,并在这两个发送操作之间放置 1 秒 sleep ,服务器会收到 2 个缓冲区并打印 message1“\n”message2。但是,如果我在没有 sleep 的情况下发送这 2 个缓冲区,服务器将获得 1 个缓冲区并打印 message1message2。我不明白为什么会这样。我的服务器和客户端代码如下。
服务器.c
#include<stdio.h>
#include<string.h> //strlen
#include<stdlib.h> //strlen
#include<sys/socket.h>
#include<arpa/inet.h> //inet_addr
#include<unistd.h> //write
#include<pthread.h> //for threading , link with lpthread
void *connection_handler(void *);
int main(int argc, char *argv[])
{
int socket_desc, new_socket, c, *new_sock;
struct sockaddr_in server, client;
char *message;
//Create socket
socket_desc = socket(AF_INET, SOCK_STREAM, 0);
if (socket_desc == -1)
{
printf("Could not create socket");
}
//Prepare the sockaddr_in structure
server.sin_family = AF_INET;
server.sin_addr.s_addr = INADDR_ANY;
server.sin_port = htons(10240);
//Bind
if (bind(socket_desc, (struct sockaddr *)&server, sizeof(server)) < 0)
{
puts("bind failed");
return 1;
}
puts("bind done");
//Listen
listen(socket_desc, 3);
//Accept and incoming connection
puts("Waiting for incoming connections...");
c = sizeof(struct sockaddr_in);
while ((new_socket = accept(socket_desc, (struct sockaddr *)&client, (socklen_t*)&c)))
{
puts("Connection accepted");
//Reply to the client
// message = "Hello Client , I have received your connection. And now I will assign a handler for you\n";
// write(new_socket, message, strlen(message));
pthread_t sniffer_thread;
new_sock = malloc(1);
*new_sock = new_socket;
if (pthread_create(&sniffer_thread, NULL, connection_handler, (void*)new_sock) < 0)
{
perror("could not create thread");
return 1;
}
//Now join the thread , so that we dont terminate before the thread
//pthread_join( sniffer_thread , NULL);
puts("Handler assigned");
}
if (new_socket<0)
{
perror("accept failed");
return 1;
}
while (1);
return 0;
}
void *connection_handler(void *socket_desc)
{
//Get the socket descriptor
int sock = *(int*)socket_desc;
int read_size;
char *message, client_message[2000];
//Send some messages to the client
// message = "Greetings! I am your connection handler\n";
// write(sock, message, strlen(message));
//
// message = "Now type something and i shall repeat what you type \n";
// write(sock, message, strlen(message));
//Receive a message from client
while ((read_size = recv(sock, client_message, 2000, 0)) > 0)
{
puts(client_message);
//Send the message back to client
//write(sock, client_message, strlen(client_message));
memset(client_message, 0, sizeof(client_message));
}
if (read_size == 0)
{
puts("Client disconnected");
fflush(stdout);
}
else if (read_size == -1)
{
perror("recv failed");
}
//Free the socket pointer
free(socket_desc);
return 0;
}
客户端.c
#include<stdio.h>
#include<string.h> //strlen
#include<sys/socket.h>
#include<arpa/inet.h> //inet_addr
int main(int argc, char *argv[])
{
int socket_desc;
struct sockaddr_in server;
char *message, server_reply[2000];
//Create socket
socket_desc = socket(AF_INET, SOCK_STREAM, 0);
if (socket_desc == -1)
{
printf("Could not create socket");
}
server.sin_addr.s_addr = inet_addr("192.168.246.26");
server.sin_family = AF_INET;
server.sin_port = htons(10240);
//Connect to remote server
if (connect(socket_desc, (struct sockaddr *)&server, sizeof(server)) < 0)
{
puts("connect error");
return 1;
}
puts("Connected\n");
//Send some data
message = "nayan";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
/*sleep(1);*/
message = NULL;
message = "Mazhar vai";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
/*sleep(1);*/
message = NULL;
message = "Sarwar vai";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
/*sleep(1);*/
message = NULL;
message = "joy";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
/*sleep(1);*/
message = NULL;
message = "sumon";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
/*sleep(1);*/
message = NULL;
message = "rahim";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
/*sleep(1);*/
message = NULL;
message = "karim";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
/*sleep(1);*/
message = NULL;
message = "jamal";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
/*sleep(1);*/
message = NULL;
message = "kamal";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
/*sleep(1);*/
message = NULL;
message = "salam";
if (send(socket_desc, message, strlen(message), 0) < 0)
{
puts("Send failed");
return 1;
}
//Receive a reply from the server
// if (recv(socket_desc, server_reply, 2000, 0) < 0)
// {
// puts("recv failed");
// }
// puts("Reply received\n");
// puts(server_reply);
return 0;
}
最佳答案
TCP 是一种流 协议(protocol)(如使用SOCK_STREAM 所示)。这意味着没有数据包的开始或结束,它只是一个字节流。这也意味着单个接收调用读取的内容可能少于使用单个传输调用发送的内容,或者使用多个(包括尾随的部分)传输调用发送的内容。
要处理它,您需要在 TCP 之上提出一个更高级别的协议(protocol)。它要么需要包含正在发送的数据的长度(以固定长度格式),要么您需要有一些其他方法来知道消息的开头或至少是结尾。
关于c - 如果客户端发送多个缓冲区而没有休眠,为什么 tcp 服务器会收到一个缓冲区?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41676614/
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
我正在尝试使用ruby和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返
它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput
我可以得到Infinity和NaNn=9.0/0#=>Infinityn.class#=>Floatm=0/0.0#=>NaNm.class#=>Float但是当我想直接访问Infinity或NaN时:Infinity#=>uninitializedconstantInfinity(NameError)NaN#=>uninitializedconstantNaN(NameError)什么是Infinity和NaN?它们是对象、关键字还是其他东西? 最佳答案 您看到打印为Infinity和NaN的只是Float类的两个特殊实例的字符串
如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象