第三章:Process & Event IO 实践

转载地址

一.进程通信

1.共享内存

共享内存就是允许两个不相关的进程访问同一个逻辑内存。共享内存是在两个正在运行的进程之间共享和传递数据的一种非常有效的方式。不同进程之间共享的内存通常安排为同一段物理内存。进程可以将同一段共享内存连接到它们自己的地址空间中,所有进程都可以访问共享内存中的地址,就好像它们是由用C语言函数malloc分配的内存一样。而如果某个进程向共享内存写入数据,所做的改动将立即影响到可以访问同一段共享内存的任何其他进程。

特别提醒:共享内存并未提供同步机制,也就是说,在第一个进程结束对共享内存的写操作之前,并无自动机制可以阻止第二个进程开始对它进行读取。所以我们通常需要用其他的机制来同步对共享内存的访问,例如前面说到的信号量。有关信号量的更多内容,可以查阅我的另一篇文章:Linux进程间通信——使用信号量。Linux进程间通信——使用共享内存

2.管道

  • 管道是一组(2个)特殊的描述符
  • 管道需要在FORK函数前调用
  • 如果某一段主动关闭管道,另一端的读写操作会直接返回0

3.消息队列

通过制定Key创建一个消息队列
在消息队列中传递的数据有大小限制(65535)
消息队列会一直保留知道被主动关闭

二.IO多路复用

1.epoll模型

epoll

epoll并不是异步的是阻塞的
epoll函数会监听注册你在自己名下所有的socket描述符
当有socket监听的事件发生时,epoll函数才会响应,并返回事件发生的socket集合
epoll本质是阻塞的,他的优点在于同事处理大量的socket连接

2.Event_loop

这里写图片描述
- event_loop是一个Reactor线程,其中运行了一个epoll实例
- 可以通过接口添加socket描述符到epoll监听中,并指定事件响应的回调函数
- Eventloop不可用于FPM模式下

3. Event_loop常见问题

  • 为什么开启了event_loop的程序会一直运行不停止?
    开启Event_loop后,程序内会启动一个线程并一直阻塞在epoll的监听上,因此不会退出。
  • 如何关闭Event_loop?
    调用swoole_event_exit函数即可关闭时间循环 (注意,swoole_server程序中此函数无效)

#

C语言本身并不直接支持`text/event-stream`这样的HTTP响应格式,因为这通常是用于Web实时应用程序(如WebSocket)的服务器端处理。然而,如果你想要在C语言环境中处理这种流式数据,你可以选择使用一些库或者第三方组件,它们可能会提供事件驱动的IO模型来帮助解析。 例如,使用`libcurl`库可以方便地发送HTTP请求并接收响应。对于`text/event-stream`格式,你需要解析接收到的HTTP头信息,识别Content-Type字段是否包含`text/event-stream`,然后逐行读取数据,每行代表一个事件。 以下是一个简单的示例,展示如何使用`libcurl`处理`text/event-stream`数据: ```c #include <curl/curl.h> // ... CURL *curl; CURLcode res; // 初始化CURL curl_global_init(CURL_GLOBAL_DEFAULT); // 创建CURL会话 curl = curl_easy_init(); if(curl) { // 设置请求的URL curl_easy_setopt(curl, CURLOPT_URL, "your_url"); // 检查是否启用事件流处理 if(curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, event_stream_write_callback) == CURLE_OK) { struct WriteData data; data.buffer = malloc(BUFFER_SIZE); // 定义缓冲区 data.offset = 0; // 初始化偏移量 // 连接到服务器并开始接收数据 res = curl_easy_perform(curl); if(res != CURLE_OK) fprintf(stderr, "curl_easy_perform() failed: %s\n", curl_easy_strerror(res)); // 清理资源 free(data.buffer); curl_easy_cleanup(curl); } curl_global_cleanup(); // 关闭全局CURL上下文 } void event_stream_write_callback(char *data, size_t size, CURL *handle, void *userp) { WriteData *write_data = (WriteData *)userp; char *new_buffer = realloc(write_data->buffer, write_data->offset + size + 1); // 扩展缓冲区 if (!new_buffer) { perror("realloc"); return; } memcpy(new_buffer + write_data->offset, data, size); new_buffer[write_data->offset + size] = '\0'; // 添加新行结束符 write_data->offset += size; process_event(new_buffer); // 根据需要解析并处理每个事件 } ``` 这里假设你有一个`process_event`函数来处理每个独立的事件。请注意,实际应用中,你还需要处理错误情况、关闭连接等细节。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值