我正在尝试学习Berkeley Socket编程,但我真的被困在这里: 我的服务器和客户端程序都有10038错误,根据MSDN的意思:“尝试对不是套接字的东西”,但我的套接字是“套接字”。这是服务器/客户端代码:
我的客户:
#include <winsock.h>
#include <stdio.h>
#include <string.h>
#define BUFFSIZE 32
void DieWithError(char *errorMessage);
int main(int argc, char* argv[]){
int sock, bytesLen;
struct sockaddr_in servAddr;
unsigned short servPort =13;
char * servIP;
char buff[BUFFSIZE+1];
WSADATA wsaData;
if(argc == 2){
servIP = argv[1];
}
if(WSAStartup(MAKEWORD(2,0), &wsaData) != 0){
puts("ERROR WSA");
}
if(sock=socket(AF_INET, SOCK_STREAM, 0) < 0){
puts("ERROR socket");
}
memset(&servAddr, 0, sizeof(servAddr));
servAddr.sin_family = AF_INET;
servAddr.sin_port = htons(servPort);
servAddr.sin_addr.s_addr = inet_addr(&servIP);
if(connect(sock, (struct sockaddr *)&servAddr, sizeof(servAddr)<0)){
DieWithError("connect Error");
}
while(bytesLen=read(sock, buff, BUFFSIZE)){
buff[bytesLen]= '\n';
}
printf("Heure : %s",buff);
closesocket(sock);
WSACleanup();
return 0;
}
void DieWithError(char *errorMessage)
{
fprintf(stderr,"%s: %d\n",
errorMessage, WSAGetLastError());
exit(1);
}
和我的服务器:
#include <winsock.h>
#include <stdio.h>
#include <string.h>
#include <time.h>
#include <stdlib.h>
#define MAXLINE 32
void DieWithError(char *errorMessage);
int main(){
int sock, sockc;
struct sockaddr_in servAddr;
unsigned short servPort =13;
time_t ticks;
WSADATA wsaData;
char buff[MAXLINE];
char * servIP;
if(WSAStartup(MAKEWORD(2,0), &wsaData) != 0){
puts("ERROR WSA");
}
if(sock=socket(AF_INET, SOCK_STREAM, IPPROTO_TCP) < 0){
puts("ERROR socket");
}
servIP = "10.1.1.3";
memset(&servAddr, 0, sizeof(servAddr));
servAddr.sin_family = AF_INET;
servAddr.sin_port = htons(servPort);
servAddr.sin_addr.s_addr = htonl(INADDR_ANY);
if(bind(sock,(struct sockaddr *)&servAddr, sizeof(servAddr) < 0)){
DieWithError("bind)()");
}
if(listen(sock,MAXQUEUE) < 0){
DieWithError("listen ()");
}
while(1){
sockc = accept(sock, (struct sockaddr *)NULL, NULL);
ticks = time(NULL);
snprintf(buff,sizeof(buff),"%s",ctime(&ticks));
write(buff, sizeof(buff), MAXLINE);
}
return 0;
}
void DieWithError(char *errorMessage)
{
fprintf(stderr,"%s: %d\n",
errorMessage, WSAGetLastError());
exit(1);
}
每次开始我在客户端的connect()
函数上有10038,在服务器上的bind()
函数上有10038。
感谢您的帮助。
答案 0 :(得分:1)
与其他平台不同,Windows不使用文件描述符来表示套接字。它使用实际的内核对象。其他平台使用int
来表示打开的文件描述符,从而表示套接字。 Windows使用SOCKET
类型,它是UINT_PTR
的typedef(32位为unsigned int
,64位为unsigned __int64
),因此它可以容纳对象句柄。 Windows上的socket()
函数在出错时返回INVALID_SOCKET
((SOCKET)(~0)
),而其他平台则返回(int)-1
。您需要确保对此进行说明,这样就不会在Windows上切片/截断合法的套接字句柄。
当您的客户呼叫inet_addr()
时,它会传递char**
,其中char*
是预期的。您的编译器应该报告错误。
由于套接字在Windows上未表示为文件描述符,因此您无法在Windows套接字中使用read()
和write()
函数。您必须使用recv()
/ WSARecv()
和send()
/ WSASend()
功能。并且您需要处理函数在出错时返回SOCKET_ERROR
( - 1)的情况。
尝试更像这样的东西:
客户:
#include <winsock.h>
#include <stdio.h>
#include <string.h>
#define BUFFSIZE 32
void DieWithErrorCode(char *errorMessage, int errCode);
void DieWithError(char *errorMessage, int *errCode = NULL);
int main(int argc, char* argv[])
{
SOCKET sock;
int ret;
struct sockaddr_in servAddr;
unsigned short servPort = 13;
char *servIP;
char buff[BUFFSIZE];
WSADATA wsaData;
if (argc != 2) {
DieWithErrorCode("ERROR argc", WSAEINVAL);
}
servIP = argv[1];
ret = WSAStartup(MAKEWORD(2,0), &wsaData);
if (ret != 0) {
DieWithErrorCode("ERROR WSAStartup", ret);
}
sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (sock == INVALID_SOCKET) {
DieWithError("ERROR socket");
}
memset(&servAddr, 0, sizeof(servAddr));
servAddr.sin_family = AF_INET;
servAddr.sin_port = htons(servPort);
servAddr.sin_addr.s_addr = inet_addr(servIP);
if (servAddr.sin_addr.s_addr == INET_NONE) {
DieWithErrorCode("ERROR inet_addr", WSAEINVAL);
}
if (connect(sock, (struct sockaddr *)&servAddr, sizeof(servAddr)) == SOCKET_ERROR) {
DieWithError("ERROR connect");
}
while(1) {
ret = recv(sock, buff, BUFFSIZE, 0);
if (ret == SOCKET_ERROR) {
DieWithError("ERROR recv");
}
if (ret == 0) {
fprintf(stderr, "disconnected\n");
break;
}
printf("Heure : %.*s\n", ret, buff);
}
closesocket(sock);
WSACleanup();
return 0;
}
void DieWithErrorCode(char *errorMessage, int errCode)
{
fprintf(stderr, "%s: %d\n", errorMessage, errCode);
exit(1);
}
void DieWithError(char *errorMessage)
{
DieWithErrorCode(errorMessage, WSAGetLastError());
}
服务器:
#include <winsock.h>
#include <stdio.h>
#include <string.h>
#include <time.h>
#include <stdlib.h>
#define MAXLINE 32
void DieWithErrorCode(char *errorMessage, int errCode);
void DieWithError(char *errorMessage);
int main()
{
SOCKET sock, sockc;
struct sockaddr_in servAddr;
unsigned short servPort = 13;
time_t ticks;
WSADATA wsaData;
char buff[MAXLINE];
int ret, buflen;
ret = WSAStartup(MAKEWORD(2,0), &wsaData);
if (ret != 0) {
DieWithErrorCode("ERROR WSA", ret);
}
sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (sock == INVALID_SOCKET) {
DieWithError("ERROR socket");
}
memset(&servAddr, 0, sizeof(servAddr));
servAddr.sin_family = AF_INET;
servAddr.sin_port = htons(servPort);
servAddr.sin_addr.s_addr = htonl(INADDR_ANY);
if (bind(sock, (struct sockaddr *)&servAddr, sizeof(servAddr)) == SOCKET_ERROR) {
DieWithError("ERROR bind");
}
if (listen(sock, MAXQUEUE) == SOCKET_ERROR) {
DieWithError("ERROR listen");
}
while(1) {
sockc = accept(sock, NULL, NULL);
if (sockc == INVALID_SOCKET) {
DieWithError("ERROR accept");
}
ticks = time(NULL);
buflen = snprintf(buff, sizeof(buff), "%s", ctime(&ticks));
if (send(sockc, buff, buflen, 0) == SOCKET_ERROR) {
fprintf(stderr, "ERROR send: %d\n", WSAGetLastError());
}
closesocket(sockc);
}
closesocket(sock);
WSACleanup();
return 0;
}
void DieWithErrorCode(char *errorMessage, int errCode)
{
fprintf(stderr,"%s: %d\n", errorMessage, errCode);
exit(1);
}
void DieWithError(char *errorMessage)
{
DieWithErrorCode(char *errorMessage, WSAGetLastError());
}