「学习笔记」解决 Linux C Socket 异常问题 (linux c socket 异常)

学习笔记:解决 Linux C Socket 异常问题

在 Linux C 开发中,Socket 是最常用的网络编程接口之一。然而,在 Socket 编程中,我们经常会遇到各种各样的异常问题。本篇学习笔记将介绍在 Linux C Socket 编程中,如何解决相关异常问题。

错误处理

在 Socket 编程中,错误处理是必不可少的一步。Socket 接口提供了一个名为 errno 的全局变量,它可以告诉我们最近一次 Socket 函数调用失败的原因。

errno 的值是一个整数,其定义在 errno.h 头文件中。一般情况下,errno 的值为0表示没有错误,其他值表示错误发生。例如,当调用 socket 函数失败时,errno 的值可能为 EAFNOSUPPORT 表示地址族不支持。

errno 的值在每次函数调用之前必须重置为0,以便确保在函数调用失败时判断 errno 的值是否为0。如果不重置 errno,那么errno 的值可能是错误发生的函数调用之前的其他函数的错误值。

下面是一个简单的例子:

“`c

#include

#include

#include

int mn() {

int sockfd;

sockfd = socket(AF_INET, SOCK_STREAM, 0);

if (sockfd

printf(“socket error: %d\n”, errno);

}

return 0;

}

“`

上述代码中,我们尝试创建一个 TCP socket。如果 socket 函数调用失败,那么 errno 的值将不为0,并输出错误信息。

阻塞与非阻塞模式

Socket 可以在阻塞或非阻塞模式下运行。默认情况下,Socket 是阻塞的。

在阻塞模式下,调用 read 和 write 函数时,系统将一直等待数据准备就绪或数据发送完成。这意味着,当调用 read 函数时,进程会一直被阻塞,直到有数据可读。同样地,当调用 write 函数时,进程会一直被阻塞,直到所有数据都被发送。

在非阻塞模式下,当调用 read 或 write 函数时,进程将立即返回,而不管数据是否准备就绪或是否已发送全部数据。在非阻塞模式下,read 和 write 函数的返回值可能是负数,表示函数调用遇到了错误。常见的错误包括 EAGN 和 EWOULDBLOCK,这两个错误指示进程需要稍后重新尝试。

下面是一个简单的例子:

“`c

#include

#include

#include

#include

#include

#include

#define PORT 8080

int mn() {

int sockfd, connfd, flags;

struct sockaddr_in server_addr, client_addr;

char buffer[1024];

sockfd = socket(AF_INET, SOCK_STREAM, 0);

if (sockfd

printf(“socket error: %d\n”, errno);

return 1;

}

bzero(&server_addr, sizeof(server_addr));

server_addr.sin_family = AF_INET;

server_addr.sin_addr.s_addr = htonl(INADDR_ANY);

server_addr.sin_port = htons(PORT);

if (bind(sockfd, (struct sockaddr*)&server_addr, sizeof(server_addr))

printf(“bind error: %d\n”, errno);

return 1;

}

if (listen(sockfd, 10)

printf(“listen error: %d\n”, errno);

return 1;

}

flags = fcntl(sockfd, F_GETFL, 0);

if (flags

printf(“fcntl F_GETFL error: %d\n”, errno);

return 1;

}

flags |= O_NONBLOCK;

if (fcntl(sockfd, F_SETFL, flags)

printf(“fcntl F_SETFL O_NONBLOCK error: %d\n”, errno);

return 1;

}

while (1) {

socklen_t len = sizeof(client_addr);

connfd = accept(sockfd, (struct sockaddr*)&client_addr, &len);

if (connfd

if (errno == EAGN || errno == EWOULDBLOCK) {

usleep(100);

continue;

} else {

printf(“accept error: %d\n”, errno);

break;

}

}

bzero(buffer, 1024);

if (read(connfd, buffer, 1024)

if (errno == EAGN || errno == EWOULDBLOCK) {

usleep(100);

} else {

printf(“read error: %d\n”, errno);

break;

}

}

printf(“message from client : %s\n”, buffer);

close(connfd);

}

close(sockfd);

return 0;

}

“`

上述代码中,我们创建了一个 TCP 服务器,并将其设置为非阻塞模式。在主循环中,我们不断等待客户端连接。当有客户端连接到达时,我们使用非阻塞模式读取数据。如果 read 函数返回 EAGN 或 EWOULDBLOCK,我们需要稍后重新尝试。这意味着我们可能需要在下一次循环中再次调用 read 函数。否则,我们将打印来自客户端的消息并关闭连接。

内存泄漏

内存泄漏是 Socket 编程中常见的错误之一。在 C 语言中,我们必须手动管理内存,包括分配、释放和复制内存。如果我们忘记释放内存,那么可能会导致内存泄漏。内存泄漏可能会导致内存不足、崩溃或其他严重问题。

下面是一个例子:

“`c

#include

#include

#include

#include

#include

#define PORT 8080

int mn() {

int sockfd;

struct sockaddr_in server_addr;

sockfd = socket(AF_INET, SOCK_STREAM, 0);

if (sockfd

printf(“socket error: %d\n”, errno);

return 1;

}

bzero(&server_addr, sizeof(server_addr));

server_addr.sin_family = AF_INET;

server_addr.sin_addr.s_addr = htonl(INADDR_ANY);

server_addr.sin_port = htons(PORT);

if (bind(sockfd, (struct sockaddr*)&server_addr, sizeof(server_addr))

printf(“bind error: %d\n”, errno);

return 1;

}

if (listen(sockfd, 10)

printf(“listen error: %d\n”, errno);

return 1;

}

while (1) {

int connfd;

struct sockaddr_in client_addr;

socklen_t len = sizeof(client_addr);

connfd = accept(sockfd, (struct sockaddr*)&client_addr, &len);

if (connfd

printf(“accept error: %d\n”, errno);

return 1;

}

char* buffer = (char*)malloc(1024);

if (read(connfd, buffer, 1024)

printf(“read error: %d\n”, errno);

return 1;

}

printf(“message from client : %s\n”, buffer);

free(buffer);

close(connfd);

}

close(sockfd);

return 0;

}

“`

上述代码中,我们将从客户端读取的数据存储在 buffer 变量中。然而,我们忘记了释放 buffer。这可能会导致内存泄漏,并最终导致内存不足。

结论

相关问题拓展阅读:

  • linux c 读取socket问题
  • linux下socket编译时出现绑定错误

linux c 读取socket问题

这个,我说下,你那含早个read的函旁老衫数那个地方有问题,你可以用一个while循环 来接收数据 ,

while(read(sockfd,buf,1900) != 0)

{

printf(“%s”,buf);

}

但是这样的话没法保存,你看看再弄个buf来保存一下

数据在网络中舆不运腔是一次就传完 ,多次接收才能正常p

linux下socket编译时出现绑定错误

教你个调试方法,你把printf(“bind error”);换成printf(“bind error: %s\n”, strerror(errno)); 这样可改蠢乱以看出哪里出错了.

我没猜错的话错误信息应该是”Address already in use.” ,如果是这个错误的话,你再等档蠢一会核档从新运行server就可以了.

去红旗Linux论坛去找找!

关于linux c socket 异常的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。

香港服务器首选树叶云,2H2G首月10元开通。
树叶云(shuyeidc.com)提供简单好用,价格厚道的香港/美国云服务器和独立服务器。IDC+ISP+ICP资质。ARIN和APNIC会员。成熟技术团队15年行业经验。

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/192753.html<

(0)
运维的头像运维
上一篇2025-04-01 17:53
下一篇 2025-04-01 17:54

相关推荐

  • 个人主题怎么制作?

    制作个人主题是一个将个人风格、兴趣或专业领域转化为视觉化或结构化内容的过程,无论是用于个人博客、作品集、社交媒体账号还是品牌形象,核心都是围绕“个人特色”展开,以下从定位、内容规划、视觉设计、技术实现四个维度,详细拆解制作个人主题的完整流程,明确主题定位:找到个人特色的核心主题定位是所有工作的起点,需要先回答……

    2025-11-20
    0
  • 社群营销管理关键是什么?

    社群营销的核心在于通过建立有温度、有价值、有归属感的社群,实现用户留存、转化和品牌传播,其管理需贯穿“目标定位-内容运营-用户互动-数据驱动-风险控制”全流程,以下从五个维度展开详细说明:明确社群定位与目标社群管理的首要任务是精准定位,需明确社群的核心价值(如行业交流、产品使用指导、兴趣分享等)、目标用户画像……

    2025-11-20
    0
  • 香港公司网站备案需要什么材料?

    香港公司进行网站备案是一个涉及多部门协调、流程相对严谨的过程,尤其需兼顾中国内地与香港两地的监管要求,由于香港公司注册地与中国内地不同,其网站若主要服务内地用户或使用内地服务器,需根据服务器位置、网站内容性质等,选择对应的备案路径(如工信部ICP备案或公安备案),以下从备案主体资格、流程步骤、材料准备、注意事项……

    2025-11-20
    0
  • 如何企业上云推广

    企业上云已成为数字化转型的核心战略,但推广过程中需结合行业特性、企业痛点与市场需求,构建系统性、多维度的推广体系,以下从市场定位、策略设计、执行落地及效果优化四个维度,详细拆解企业上云推广的实践路径,精准定位:明确目标企业与核心价值企业上云并非“一刀切”的方案,需先锁定目标客户群体,提炼差异化价值主张,客户分层……

    2025-11-20
    0
  • PS设计搜索框的实用技巧有哪些?

    在PS中设计一个美观且功能性的搜索框需要结合创意构思、视觉设计和用户体验考量,以下从设计思路、制作步骤、细节优化及交互预览等方面详细说明,帮助打造符合需求的搜索框,设计前的规划明确使用场景:根据网站或APP的整体风格确定搜索框的调性,例如极简风适合细线条和纯色,科技感适合渐变和发光效果,电商类则可能需要突出搜索……

    2025-11-20
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注