在Linux环境下使用C语言实现一对一通信,通常可以采用以下几种常见的方法:
一、基于套接字(Socket)的TCP通信
socket
函数创建套接字。对于一对一通信,TCP协议比较合适,因为它提供可靠的、面向连接的字节流服务。bind
函数将套接字绑定到一个特定的IP地址和端口上,然后使用listen
函数监听连接请求。客户端使用connect
函数向服务器发起连接请求,服务器使用accept
函数接受连接,之后双方就可以通过套接字进行数据传输。#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#define PORT 8888
#define BUFFER_SIZE 1024
int main() {
int server_fd, new_socket;
struct sockaddr_in address;
int addrlen = sizeof(address);
char buffer[BUFFER_SIZE] = {0};
// 创建套接字
if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) {
perror("socket failed");
exit(EXIT_FAILURE);
}
// 设置服务器地址结构
address.sin_family = AF_INET;
address.sin_addr.s_addr = INADDR_ANY;
address.sin_port = htons(PORT);
// 绑定套接字到地址
if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < 0) {
perror("bind failed");
close(server_fd);
exit(EXIT_FAILURE);
}
// 监听连接
if (listen(server_fd, 3) < 0) {
perror("listen");
close(server_fd);
exit(EXIT_FAILURE);
}
printf("Server is listening on port %d
", PORT);
// 接受连接
if ((new_socket = accept(server_fd, (struct sockaddr *)&address, (socklen_t*)&addrlen)) < 0) {
perror("accept");
close(server_fd);
exit(EXIT_FAILURE);
}
// 接收数据
int valread = read(new_socket, buffer, BUFFER_SIZE);
printf("Received: %s
", buffer);
// 发送响应
const char *response = "Hello from server";
send(new_socket, response, strlen(response), 0);
close(new_socket);
close(server_fd);
return 0;
}
示例代码(客户端)
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#define PORT 8888
#define BUFFER_SIZE 1024
int main() {
int sock = 0;
struct sockaddr_in serv_addr;
char *message = "Hello from client";
char buffer[BUFFER_SIZE] = {0};
// 创建套接字
if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
printf("
Socket creation error
");
return -1;
}
serv_addr.sin_family = AF_INET;
serv_addr.sin_port = htons(PORT);
// 将IPv4地址从文本转换为二进制形式
if (inet_pton(AF_INET, "127.0.0.1", &serv_addr.sin_addr)<=0) {
printf("
Invalid address/ Address not supported
");
return -1;
}
// 连接服务器
if (connect(sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < 0) {
printf("
Connection Failed
");
return -1;
}
// 发送数据
send(sock, message, strlen(message), 0);
printf("Sent: %s
", message);
// 接收响应
int valread = read(sock, buffer, BUFFER_SIZE);
printf("Received: %s
", buffer);
close(sock);
return 0;
}
二、基于命名管道(FIFO)的一对一通信
mkfifo
函数创建命名管道。// 创建命名管道(在服务器端或先运行的进程)
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#define FIFO_NAME "my_fifo"
int main() {
if (mkfifo(FIFO_NAME, 0666) < 0 && errno!= EEXIST) {
perror("mkfifo");
return 1;
}
return 0;
}
// 写入数据到命名管道(客户端)
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#define FIFO_NAME "my_fifo"
int main() {
int fd = open(FIFO_NAME, O_WRONLY);
if (fd < 0) {
perror("open");
return 1;
}
const char *message = "Hello from writer";
write(fd, message, strlen(message));
close(fd);
return 0;
}
// 从命名管道读取数据(服务器端)
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#define FIFO_NAME "my_fifo"
int main() {
int fd = open(FIFO_NAME, O_RDONLY);
if (fd < 0) {
perror("open");
return 1;
}
char buffer[1024];
int n = read(fd, buffer, sizeof(buffer));
printf("Received: %s
", buffer);
close(fd);
return 0;
}
三、可能遇到的问题及解决方法
bind
函数中的地址和端口设置,确保端口未被其他进程占用(可以使用netstat -tlnp
命令查看端口使用情况)。对于客户端,检查连接的IP地址是否为服务器的正确地址,端口是否正确。fcntl
函数设置O_NONBLOCK
标志),但需要注意处理非阻塞模式下可能出现的特殊情况,如没有数据可读或者没有空间可写时的情况。领取专属 10元无门槛券
手把手带您无忧上云