此代码是学习c语言的时候写的一个demo,能简单实现sockect通信。
运行环境:基于window10的linux子系统, 基于window10的virbox虚拟机的ubuntu系统
代码中都有比较详细的注释,具体的方法如果大家感兴趣,查阅c语言手册
在linux系统下使用gcc或g++编译都行:例如
gcc server.c -o server
gcc client.c -o client
测试:
1、手动开启两个ssh界面,一个运行./server。 一个运行./client “this is a test”
服务端
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <ctype.h>
#include <arpa/inet.h>
#define SERVER_PORT 666
int main(void)
{
int sock; // 代表信箱
int i;
struct sockaddr_in server_addr;
//1.创建一个信箱
sock = socket(AF_INET, SOCK_STREAM, 0);
//2.清空标签。写上地址和端口号
bzero(&server_addr, sizeof(server_addr));
server_addr.sin_family = AF_INET; // 选择协议族,ipv4
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);// 监听本地所有的IP地址
server_addr.sin_port = htons(SERVER_PORT);//绑定端口号
//实现标签贴到收信的信箱上
bind(sock, (struct sockaddr *)&server_addr, sizeof(server_addr));
//把信箱挂在传达室,这样,保安就可以接信了
listen(sock, 64);// 同一时间端只能接受64封
// 完事具备,只能来信
printf("等待客户端的连接\n");
int done = 1;
while (done) {
struct sockaddr_in client;
int client_sock;
char client_ip[64];
char buf[256];
int len;
socklen_t client_addr_len;
client_addr_len = sizeof(client);
client_sock = accept(sock, (struct sockaddr *)&client, &client_addr_len);
// 打印客户端的IP地址和端口号
printf("client ip: %s\t port: %d\n",
inet_ntop(AF_INET, &client.sin_addr.s_addr, client_ip, sizeof(client_ip)),
ntohs(client.sin_port));
// 读取客户端发送的数据
len = read(client_sock, buf, sizeof(buf) - 1);
buf[len] = '\0';
printf("recive[%d]: %s\n", len, buf);
// 转换成大写
for(i=0; i<len; i++){
/*if (buf[i] >= 'a' && buf[i] <= 'z') {
buf[i] = buf[i] - 32;
}*/
buf[i] = toupper(buf[i]);
}
len = write(client_sock, buf, len);
printf("write finished. len = %d\n", len);
close(client_sock);
}
close(sock);
return 0;
}
客户端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define SERVER_PORT 666
#define SERVER_IP "127.0.0.1"
int main(int argc, char *argv[])
{
int sockfd;
char *message;
struct sockaddr_in servaddr;
int n;
char buf[64];
if (argc != 2) {
fputs("Usage : ./echo_client message \n" ,stderr);
exit(1);
}
message = argv[1];
printf("message : %s\n", argv[0]);
printf("message : %s\n", message);
sockfd = socket(AF_INET, SOCK_STREAM, 0);
memset(&servaddr, '\0', sizeof(struct sockaddr_in));
servaddr.sin_family = AF_INET;
inet_pton(AF_INET, SERVER_IP, &servaddr.sin_addr);
servaddr.sin_port = htons(SERVER_PORT);
connect(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));
write(sockfd, message, strlen(message));
n = read(sockfd, buf, sizeof(buf)-1);
if (n > 0) {
buf[n] = '\0';
printf("recevice : %s\n", buf);
} else {
perror("error!!!");
}
printf("finished. \n");
close(sockfd);
return 0;
}
注意事项
1、目前我们的端口是写死的,所以如果端口冲突也会导致失败
2、如果连接失败可能是端口不匹配造成,因为不用root账号去启动server的时候,他的端口号可能就匹配不到,具体原因不明。