- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我有 3 个应用程序、一个服务器、2 个客户端。我正在尝试对此进行编程,以便我可以将我的客户端的登录请求发送到服务器并请求查看有关另一个客户端的信息。
问题:我对所有 3 个应用程序使用相同的 ip 和端口号(这都是在我计算机上的 3 个 visual studio 应用程序上完成的)否则我认为它无法连接到服务器。这样可以吗?
当我尝试在服务器的 while 循环之外使用 listen 和 accept 时,它适用于一个客户端,但不适用于另一个客户端。当我尝试将 listen 和 accept 放在 while 循环中时,它对两个客户端都有效,但不会响应多个 accept。我的问题是什么?
服务器
struct Users {
int message;
char userName[50];
char ipAddress[50];
int PortNumber;
bool online;
};
int main () {
//Create users
Users client[2];
client[0].PortNumber = 20000;
client[0].online = false;
sprintf(&client[0].userName[0], "Albert");
sprintf(&client[0].ipAddress[0], "127.0.0.1");
client[1].PortNumber = 20000;
client[1].online = false;
sprintf(&client[1].userName[0], "Monique");
sprintf(&client[1].ipAddress[0], "127.0.0.1");
//Set up all the connection stuff
struct sockaddr_in SvrAddr;
SOCKET WelcomeSocket, ConnectionSocket;
int PortNumber = 20000;
int result;
char IPAddress[] = "127.0.0.1";
char RxBuffer[128];
char TxBuffer[128];
WORD wVersionRequested;
WSADATA wsaData;
wVersionRequested = MAKEWORD(2,2); //For Server
if(WSAStartup(wVersionRequested, &wsaData) != 0)
return -1;
//Setting up the welcome socket
WelcomeSocket = socket(AF_INET, SOCK_STREAM, 0);
//Setting up the sockaddr svraddr structure
SvrAddr.sin_family = AF_INET;
SvrAddr.sin_addr.s_addr = inet_addr(IPAddress);
SvrAddr.sin_port = htons(PortNumber);
//Bind
bind(WelcomeSocket, (sockaddr*)&SvrAddr, sizeof(SvrAddr));
//listening
//listen(WelcomeSocket, 5);
//temp users
Users temp;
//ConnectionSocket = accept(WelcomeSocket, NULL, NULL);
ConnectionSocket = SOCKET_ERROR;
//while loop for the connection
while (1) {
listen(WelcomeSocket, 5);
if ((ConnectionSocket = accept(WelcomeSocket, NULL, NULL)) == SOCKET_ERROR)
{
return -1;
}
else {
int n = recv(ConnectionSocket, RxBuffer, sizeof(RxBuffer), 0);
if (n == 0)
break;
memcpy(&temp, RxBuffer, sizeof(struct Users));
//cout << temp.message << temp.userName << endl << endl;
//check which message type is being sent
switch(temp.message) {
//if message type 1
case 1 :
for (int i = 0; i < 2; i++) {
//if receieved username matches with any username in the database
if (strcmp(temp.userName, client[i].userName) == 0) {
//assign the recieved users information to the matched one in database
strcpy(client[i].userName, temp.userName);
client[i].online = true;
client[i].message = 2;
cout << "Username: " << client[i].userName << endl << "Online status: " << client[i].online << endl << endl;
//send the acknowledgement packet
send(ConnectionSocket, (char *)&client[i], sizeof(struct Users), 0);
//closesocket(ConnectionSocket);
}
}
break;
//if message type 3
case 3 :
cout << "User being searched for: " << temp.userName << endl << endl;
for (int i = 0; i < 2; i++) {
//if receieved username matches with any username in the database
if (strcmp(temp.userName, client[i].userName) == 0) {
client[i].message = 4;
//send the acknowledgement packet
send(ConnectionSocket, (char *)&client[i], sizeof(struct Users), 0);
//closesocket(ConnectionSocket);
}
}
break;
default :
break;
}
}
}
closesocket(ConnectionSocket);
WSACleanup();
}
客户端(两者相同)
struct Users {
int message;
char userName[50];
char ipAddress[50];
int PortNumber;
bool online;
};
int main () {
struct sockaddr_in SvrAddr;
SOCKET ClientSocket;
int PortNumber = 20000;
char IPAddress[] = "127.0.0.1";
//char message[] = "Hello this is the client.";
char RxBuffer[128];
WORD wVersionRequested;
WSADATA wsaData;
wVersionRequested = MAKEWORD(2,3); //For Client
if(WSAStartup(wVersionRequested, &wsaData) != 0)
return -1;
ClientSocket = socket(AF_INET, SOCK_STREAM, 0);
SvrAddr.sin_family = AF_INET;
SvrAddr.sin_addr.s_addr = inet_addr(IPAddress);
SvrAddr.sin_port = htons(PortNumber);
connect(ClientSocket, (sockaddr*)&SvrAddr, sizeof(SvrAddr));
//cout << "Name: ";
//cin >> login;
//Send request to login
int log;
char * name = new char[128];
char * request = new char[128];
Users client;
Users talkto;
do {
cout << "To login press (1) to end press (2). ";
cin >> log;
flushall();
if (log == 1) {
cout << "Username : ";
cin.getline(name, 128, '\n');
cout << endl;
flushall();
//Set client login info
strcpy(client.userName, name);
client.message = 1;
send(ClientSocket, (char *)&client, sizeof(struct Users), 0);
//Recieve acknowledgement
recv(ClientSocket, RxBuffer, sizeof(RxBuffer), 0);
//create temp users
Users temp;
memcpy(&temp, RxBuffer, sizeof(struct Users));
//If logged in and received a message of type 2 (acknowledgement)
if (temp.message == 2) {
cout << "You have logged in." << endl << endl;
cout << "Enter user to request user information: ";
cin.getline(talkto.userName, 128, '\n');
flushall();
cout << endl;
talkto.message = 3;
//send request for user information packet
send(ClientSocket, (char *)&talkto, sizeof(struct Users), 0);
recv(ClientSocket, RxBuffer, sizeof(RxBuffer), 0);
memcpy(&temp, RxBuffer, sizeof(struct Users));
//if message received is of type 4 (acknowledgement of user request info)
if (temp.message == 4) {
cout << "Requested User: " << temp.userName << endl << "Online status: " << temp.online << endl << endl;
}
}
//cout << temp.userName << endl << temp.online << endl << temp.message;
}
} while (log != 2);
closesocket(ClientSocket);
WSACleanup();
}
最佳答案
您不要在循环内使用 listen()
。只有 accept()
。此外,您不使用非阻塞套接字
,但这是您的偏好。另一件事,每次完成 accept
时都必须使用 closesocket
。此外,您的 recv()
必须在循环中处理,因为与发送不同,它可能会在接收到所有数据之前返回。
服务器的基本结构:
struct sockaddr_in inAddr; //incoming address. Use it to distinguise between incoming clients
struct sockaddr_in address;
unsigned int inLen;
int recvMsgSize;
unsigned short port = 20000;
Socket socket_handle, acceptSocket;
socket_handle = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if(socket_handle == INVALID_SOCKET){
//error
}
memset(&address, 0, sizeof(address));
address.sin_family = AF_INET;
address.sin_addr.s_addr = htonl(INADDR_ANY); //All incoming addresses
address.sin_port = htons(port); //port must be unsigned short not int!
if(bind(socket_handle, (struct sockaddr *)&address, sizeof(address)) < 0){
//error
}
if(listen(socket_handle, MAXPENDING) < 0){
//error
}
while(true){
acceptSocket = accept(socket_handle, (struct sockaddr *) &inAddr, &inLen));
if(acceptSocket == INVALID_SOCKET){
//error
}
while(true){
recvMsgSize = recv(acceptSocket, received_data, strlen(received_data), 0);
if(recvMsgSize < 0){/*error*/}
else if(recvMsgSize == 0){break;}
else{/*receive your data*/}
}
(do other stuff...)
closesocket(acceptSocket);
}
closesocket(socket_handle);
WSACleanup();
瓦尔特
关于c - 在一台服务器上接收多个客户端数据包,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22122066/
#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
我是一名优秀的程序员,十分优秀!