- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试用C语言实现一个客户端-服务器程序。我希望客户端在特定端口5555上连接到服务器。服务器接受连接后,我希望服务器将任意端口发送回客户端。为了将客户端重定向到客户端可以发送信息的端口。我的问题是客户端收到新端口但无法通过该端口连接到服务器
客户端源代码如下(graham.c):
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netdb.h>
#include <netinet/in.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#define PORT 5555
#define MAX_USERNAME_SIZE 16
#define MAX_FILENAME_SIZE 255
#define BUFFSIZE 1024
char *extract_filename_from_path(char *pathname);
char *extract_filename_from_path(char *pathname)
{
char *last = strrchr(pathname, '/');
return (last != NULL) ? last + 1 : pathname;
}
int main(int argc, char *argv[])
{
int sockfd1, sockfd2, numbytes;
struct sockaddr_in their_addr;
struct hostent *he;
FILE *photo;
char buff[BUFFSIZE];
short status = 1;
// verifier le nombre d'arguments
if (argc != 7)
{
fprintf(stderr, "Compléter sa commande de la façon suivante:\n"
"graham ip_serveur nom_utilisateur_dans_le_pool"
" /chemin/vers/la/photo année mois jour");
return EXIT_FAILURE;
}
if ((he = gethostbyname(argv[1])) == NULL)
{
perror("Client: gethostbyname");
return EXIT_FAILURE;
}
// initialiser le socket 1
if ((sockfd1 = socket(PF_INET, SOCK_STREAM, 0)) == -1)
{
perror("Client: socket");
return EXIT_FAILURE;
}
their_addr.sin_family = AF_INET;
their_addr.sin_port = htons(PORT);
their_addr.sin_addr = *((struct in_addr*)he->h_addr);
memset(&(their_addr.sin_zero), '\0', 8);
if (connect(sockfd1, (struct sockaddr *)&their_addr,
sizeof(struct sockaddr)) == -1)
{
perror("Client: connect 1");
return EXIT_FAILURE;
}
// get port
short int port;
if (recv(sockfd1, &port, sizeof(short int), 0) == -1)
{
perror("Client: recv port");
return EXIT_FAILURE;
}
printf("recieved port: %d\n", ntohs(port));
close(sockfd1);
// changer de port
// initialiser le socket 2
if ((sockfd2 = socket(PF_INET, SOCK_STREAM, 0)) == -1)
{
perror("Client: socket");
return EXIT_FAILURE;
}
their_addr.sin_family = AF_INET;
their_addr.sin_port = port;
their_addr.sin_addr = *((struct in_addr*)he->h_addr);
memset(&(their_addr.sin_zero), '\0', 8);
if (connect(sockfd2, (struct sockaddr *)&their_addr,
sizeof(struct sockaddr)) == -1)
{
perror("Client: connect 2");
return EXIT_FAILURE;
}
// 2) envoyer le nom d'utilisateur
if (send(sockfd2, argv[2], MAX_USERNAME_SIZE, 0) == -1)
{
perror("Client: send username");
return EXIT_FAILURE;
}
// 3) envoyer la date de prise
if (send(sockfd2, argv[4], 5, 0) == -1)
{
perror("Client: send year");
return EXIT_FAILURE;
}
if (send(sockfd2, argv[5], 3, 0) == -1)
{
perror("Client: send month");
return EXIT_FAILURE;
}
if (send(sockfd2, argv[6], 3, 0) == -1)
{
perror("Client: send day");
return EXIT_FAILURE;
}
// 4) envoyer le nom de l'image
if (send(sockfd2, extract_filename_from_path(argv[3]), MAX_FILENAME_SIZE, 0) == -1)
{
perror("Client: send filename");
return EXIT_SUCCESS;
}
// 5) envoyer l'image au client
if ((photo = fopen(argv[3], "r")) == NULL)
{
perror("Client: fopen");
return EXIT_FAILURE;
}
do
{
if ((numbytes = write(sockfd2, buff, BUFFSIZE)) > 0)
{
for (int i = 0; i < numbytes; i++)
{
buff[i] = getc(photo);
}
}
else
{
break;
}
}while(1);
fclose(photo);
// 6) recevoir et traiter le code de retour
if ((numbytes = recv(sockfd2, &status, sizeof(short int), 0)) == -1)
{
perror("Client: recv return code");
return EXIT_FAILURE;
}
if ((status = ntohs(status)))
{
fprintf(stderr, "Echec de la transaction");
}
close(sockfd2);
return EXIT_SUCCESS;
}
服务器源代码是(spbx.c):
#include "spbx.h"
int my_read(FILE *target, const int source_fd)
{
char buff[BUFFSIZE];
memset(buff, '\0', BUFFSIZE);
int numbytes;
int status = 0;
if ((numbytes = read(source_fd, buff, BUFFSIZE)) > 0)
{
for (int i = 0; i < numbytes; i++)
{
putc(buff[i], target);
}
}
else
{
status = -1;
}
return status;
}
int my_mkdir(const char *path, mode_t mode)
{
struct stat s;
int status = 0;
if (stat(path, &s) == -1)
{
if (mkdir(path, mode) == -1 && errno != EEXIST)
{
status = -1;
}
}
else
{
errno = ENOTDIR;
status = -1;
}
return status;
}
int init_socket(int *sockfd, const short int port)
{
struct sockaddr_in my_addr;
int yes = 1;
if ((*sockfd = socket(PF_INET, SOCK_STREAM, 0)) == -1)
{
perror("Serveur: socket");
return -1;
}
if (setsockopt(*sockfd, SOL_SOCKET,
SO_REUSEADDR, &yes, sizeof(int)) == -1)
{
perror("Serveur: setsockopt");
return -1;
}
my_addr.sin_family = AF_INET;
my_addr.sin_port = htons(port);
my_addr.sin_addr.s_addr = INADDR_ANY;
memset(&(my_addr.sin_zero), '\0', 8);
if (bind(*sockfd, (struct sockaddr *)&my_addr,
sizeof(struct sockaddr)) == -1)
{
perror("Serveur: bind");
return -1;
}
if (listen(*sockfd, BACKLOG) == -1)
{
perror("Serveur: listen");
return -1;
}
return 0;
}
int recv_date(int new_fd, char **year, char **month, char **day)
{
int numbytes;
if ((numbytes = recv(new_fd, year, 5, 0)) == -1)
{
perror("Serveur: recv year");
return -1;
}
if ((numbytes = recv(new_fd, month, 3, 0)) == -1)
{
perror("Serveur: recv month");
return -1;
}
if ((numbytes = recv(new_fd, day, 3, 0)) == -1)
{
perror("Serveur: recv day");
return -1;
}
return 0;
}
int main(int argc, char *argv[])
{
// ecouter sur sockfd et etablir la nouvelle
// connexion sur new_fd
int sockfd1, sockfd2, new_fd, numbytes, photo_tmp_fd;
struct sockaddr_in their_addr;
unsigned int sin_size;
char username[MAX_USERNAME_SIZE];
char filename[MAX_FILENAME_SIZE];
char path_tmp[MAX_FILEPATH_SIZE];
char path[MAX_FILEPATH_SIZE];
memset(path, '\0', MAX_FILEPATH_SIZE);
char year[5], month[3], day[3];
FILE *photo_tmp;
FILE *photo;
short int status = 1;
fd_set readfds;
int max;
// verifier le nombre d'arguments
if (argc != 2)
{
fprintf(stderr, "Donner le chemin vers le pool 2\n");
return EXIT_FAILURE;
}
// initialiser les socket
if(init_socket(&sockfd1, MY_PORT) == -1) return EXIT_FAILURE;
if(init_socket(&sockfd2, 0) == -1) return EXIT_FAILURE;
struct sockaddr_in my_addr;
if (getsockname(sockfd2, (struct sockaddr*)&my_addr, &sin_size) == -1)
{
perror("Serveur: getsockname");
return EXIT_FAILURE;
}
sin_size = sizeof(struct sockaddr_in);
while (1)
{
//initialisation du fd_set
FD_ZERO(&readfds);
FD_SET(sockfd1, &readfds);
FD_SET(sockfd2, &readfds);
max = (sockfd1 < sockfd2)? sockfd2 : sockfd1;
select(max + 1, &readfds, NULL, NULL, NULL);
if (FD_ISSET(sockfd1, &readfds))
{
// on envoie le port
if ((new_fd = accept(sockfd1,
(struct sockaddr *)&their_addr,
&sin_size)) == -1)
{
perror("Serveur: accept 1");
return EXIT_FAILURE;
}
printf("Serveur: connexion reçue du client %s\n",
inet_ntoa(their_addr.sin_addr));
if (fork() == 0)
{
close(sockfd1);
if (send(new_fd, &my_addr.sin_port, sizeof(short int), 0) == -1)
{
perror("Serveur: send port");
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
close(new_fd);
}
else if (FD_ISSET(sockfd2, &readfds))
{
if ((new_fd = accept(sockfd2,
(struct sockaddr *)&their_addr, &sin_size)) == -1)
{
perror("Serveur: accept 2");
return EXIT_FAILURE;
}
printf("Serveur: connexion reçue du client %s redirigé vers le port %d\n",
inet_ntoa(their_addr.sin_addr), ntohs(my_addr.sin_port));
if (fork() == 0)
{
close(sockfd1);
close(sockfd2);
// 2) recevoir l'utilisateur du pool sous forme
// de chaîne de caractères
if ((numbytes = recv(new_fd, &username, MAX_USERNAME_SIZE, 0)) == -1)
{
perror("Serveur: recv username");
return EXIT_FAILURE;
}
// 3) récevoir la date de prise de vue
if(recv_date(new_fd, (char **) &year, (char **)&month, (char **) &day) == -1)
return EXIT_FAILURE;
// 4) recevoir le nom du fichier
if ((numbytes = recv(new_fd, &filename, MAX_FILENAME_SIZE, 0)) == -1)
{
perror("Serveur: recv filename");
return EXIT_FAILURE;
}
// 5) recevoir l'image téléversée par le client
sprintf(path_tmp, "/tmp/%s", filename);
if ((photo_tmp = fopen(path_tmp, "w")) == NULL)
{
perror("Serveur: fopen");
return EXIT_FAILURE;
}
do
{
if (my_read(photo_tmp, sockfd2) == -1) break;
}while(1);
fclose(photo_tmp);
// 6) creer l'arborescence
sprintf(path, "%s/%s", argv[1], username);
my_mkdir(path, MODE);
sprintf(path, "%s/%s/%s", argv[1], username, year);
my_mkdir(path, MODE);
sprintf(path, "%s/%s/%s/%s", argv[1], username, year, month);
my_mkdir(path, MODE);
sprintf(path, "%s/%s/%s/%s/%s", argv[1], username, year, month, day);
my_mkdir(path, MODE);
sprintf(path, "%s/%s/%s/%s/%s/%s", argv[1], username, year, month, day, filename);
// 7) Enregistrement de la photographie
if ((photo = fopen(path, "w")) == NULL
|| (photo_tmp_fd = open(path_tmp, O_WRONLY)) == -1)
{
perror("Serveur: fopen");
return EXIT_FAILURE;
}
do
{
if (my_read(photo, photo_tmp_fd) == -1) break;
}while(1);
close(photo_tmp_fd);
fclose(photo);
// 8) enoyer le code 0 en cas de success
status = htons(0);
if (send(new_fd, &status, sizeof(short int), 0) == -1)
{
perror("Serveur: send code");
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
close(new_fd);//9) deconnexion
}
else
{
fprintf(stderr, "erreur de select\n");
}
}
return EXIT_SUCCESS;
}
当我运行服务器并尝试使用客户端程序连接它两次时,得到以下结果:
服务器端:
Serveur: connexion reçue du client 127.0.0.1
Serveur: connexion reçue du client 127.0.0.1
客户端:
recieved port: 4645
Client: connect 2: Connection refused
recieved port: 4645
Client: connect 2: Connection refused
我唯一注意到的是,对于服务器的同一次执行和客户端的多次执行,端口保持不变。但我不知道我做错了什么。提前致谢。
最佳答案
我应该在客户端连接后创建新的套接字,如所提到的@克里斯特纳。所以我移动了以下几行:
if(init_socket(&sockfd2, 0) == -1) return EXIT_FAILURE;
if (getsockname(sockfd2, (struct sockaddr*)&my_addr, &sin_size) == -1)
{
perror("Serveur: getsockname");
return EXIT_FAILURE;
}
在服务器程序中的第一个 accept()
之后和 fork()
之前。
关于c - 如何在 C 中将客户端程序重定向到新的任意端口,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53868780/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!