最近在看SRS的源码。SRS是基于协程开发的,底层使用了StateThreads。所以为了充分的理解SRS源码,需要先学习一下StateThreads。这里对StateThreads的学习做了一些总结和记录。

StateThreads是什么

StateThreads是一个用户级线程库,用于多线程编程。它提供了一种轻量级的线程模型,允许开发人员以更简单的方式编写并发程序。

StateThreads有什么用

StateThreads 的主要目标是提供一种高效的用户级线程模型,以减少线程切换和上下文切换的开销。它采用协作式调度策略,即线程在主动释放执行权之前不会被抢占。这种方式可以减少线程切换的开销,但也需要开发人员在适当的时机主动释放执行权,以避免长时间的阻塞导致程序响应性下降。

StateThreads 提供了一组简单的函数和宏,用于创建和管理线程、同步和通信等操作。它支持线程的创建、销毁、休眠、唤醒等基本操作,以及互斥锁、条件变量、信号量等同步机制。开发人员可以使用这些函数和宏来编写并发程序,而不需要直接操作操作系统提供的线程和同步原语。

总的来说,StateThreads是一个高性能、高并发、高扩展性和可读性的网络服务器架构。

StateThreads怎么用

下载

git clone -b srs https://github.com/ossrs/state-threads.git

编译

make linux-debug

编译完成后,将头文件导入需要使用到StateThreads的项目。并在编译项目时链接st库即可。

使用示例

示例一

下面是用StateThreads实现的一个简单的服务,可以监听客户端的连接。

#include <iostream>
#include <stdio.h>
#include <arpa/inet.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <st.h> #define LISTEN_PORT 8000 #define ERR_EXIT(m) \
do \
{ \
perror(m); \
exit(-1); \
} while (0) void *client_thread(void *arg)
{
st_netfd_t client_st_fd = (st_netfd_t)arg;
// 用于获取与 st_netfd_t 对象关联的文件描述符(File Descriptor)。它返回一个整数值,表示文件描述符的值。
// 将 st_netfd_t 对象转换为普通的文件描述符
int client_fd = st_netfd_fileno(client_st_fd); sockaddr_in client_addr;
socklen_t client_addr_len = sizeof(client_addr);
// 获取与套接字连接的对端的地址信息
int ret = getpeername(client_fd, (sockaddr *)&client_addr, &client_addr_len);
if (ret == -1)
{
printf("[WARN] Failed to get client ip: %s\n", strerror(ret));
} char ip_buf[INET_ADDRSTRLEN];
// 内存区域清零
memset(ip_buf, 0, sizeof(ip_buf));
inet_ntop(client_addr.sin_family, &client_addr.sin_addr, ip_buf,
sizeof(ip_buf)); while (1)
{
char buf[1024] = {0};
// 从给定的套接字中读取指定字节数的数据,并将其存储在提供的缓冲区 buf 中
ssize_t ret = st_read(client_st_fd, buf, sizeof(buf), ST_UTIME_NO_TIMEOUT);
if (ret == -1)
{
printf("client st_read error\n");
break;
}
else if (ret == 0)
{
printf("client quit, ip = %s\n", ip_buf);
break;
} printf("recv from %s, data = %s", ip_buf, buf); ret = st_write(client_st_fd, buf, ret, ST_UTIME_NO_TIMEOUT);
if (ret == -1)
{
printf("client st_write error\n");
}
}
} void *listen_thread(void *arg)
// 监听
{
while (1)
{
st_netfd_t client_st_fd =
st_accept((st_netfd_t)arg, NULL, NULL, ST_UTIME_NO_TIMEOUT);
if (client_st_fd == NULL)
{
continue;
} printf("get a new client, fd = %d\n", st_netfd_fileno(client_st_fd)); st_thread_t client_tid =
st_thread_create(client_thread, (void *)client_st_fd, 0, 0);
if (client_tid == NULL)
{
printf("Failed to st create client thread\n");
}
}
} int main()
{
// 用于设置 ST 库的事件系统。
int ret = st_set_eventsys(ST_EVENTSYS_ALT);
if (ret == -1)
{
printf("st_set_eventsys use linux epoll failed\n");
}
// st初始化
ret = st_init();
if (ret != 0)
{
printf("st_init failed. ret = %d\n", ret);
return -1;
}
// 创建套接字
int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
if (listen_fd == -1)
{
ERR_EXIT("socket");
} int reuse_socket = 1; // 设置套接字选项
ret = setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &reuse_socket,
sizeof(int));
if (ret == -1)
{
ERR_EXIT("setsockopt");
} struct sockaddr_in server_addr; // 用于表示 IPv4 地址的结构体
server_addr.sin_family = AF_INET; // 地址族,一般为 AF_INET
server_addr.sin_port = htons(LISTEN_PORT); // 端口
server_addr.sin_addr.s_addr = INADDR_ANY; // ipv4地址结构
// 将套接字与特定的 IP 地址和端口号进行绑定
ret =
bind(listen_fd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr));
if (ret == -1)
{
ERR_EXIT("bind");
} ret = listen(listen_fd, 128);
if (ret == -1)
{
ERR_EXIT("listen");
}
// st_netfd_open_socket() 是 State Threads (ST) 库中的一个函数,用于创建一个 st_netfd_t 类型的文件描述符对象,以便进行异步 I/O 操作。
st_netfd_t st_listen_fd = st_netfd_open_socket(listen_fd);
if (!st_listen_fd)
{
printf("st_netfd_open_socket open socket failed.\n");
return -1;
} // 创建线程监听来一个建立连接的请求
st_thread_t listen_tid =
st_thread_create(listen_thread, (void *)st_listen_fd, 1, 0);
if (listen_tid == NULL)
{
printf("Failed to st create listen thread\n");
} while (1)
{
st_sleep(1);
} return 0;
}

示例二

StateThreads创建多线程

#include <stdio.h>
#include <st.h>
#include <string> void *do_calc(void *arg)
{
int sleep_ms = (int)(long int)(char *)arg * 10; for (;;)
{
printf("in sthread #%dms\n", sleep_ms);
st_usleep(sleep_ms * 1000);
} return NULL;
} int main(int argc, char **argv)
{
if (argc <= 1)
{
printf("Test the concurrence of state-threads!\n"
"Usage: %s <sthread_count>\n"
"eg. %s 10000\n",
argv[0], argv[0]);
return -1;
} if (st_init() < 0)
{
printf("error!");
return -1;
} int i;
int count = std::stoi(argv[1]);
for (i = 1; i <= count; i++)
{
if (st_thread_create(do_calc, (void *)i, 0, 0) == NULL)
{
printf("error!");
return -1;
}
} st_thread_exit(NULL); return 0;
}

关于StateThreads的运行原理,可以看文章《SRS开源直播服务 - StateThreads微线程框架学习

SRS中的StateThreads

使用的源码为SRS4.0

系统架构图:



在SRS的源码中,StateThreads在srs_st_init()函数中完成初始化。具体的调用流程如下。

SRS的main函数在文件srs_main_server.cpp中。

srs_main_server.cpp

......
int main(int argc, char** argv)
{
srs_error_t err = do_main(argc, argv);
......
} srs_error_t do_main(int argc, char** argv)
{
srs_error_t err = srs_success; // Initialize global or thread-local variables.
if ((err = srs_thread_initialize()) != srs_success) {
return srs_error_wrap(err, "thread init");
}
......
}

srs_app_threads.cpp

......
srs_error_t srs_thread_initialize()
{
srs_error_t err = srs_success;
......
// Initialize ST, which depends on pps cids.
if ((err = srs_st_init()) != srs_success) {
return srs_error_wrap(err, "initialize st failed");
}
......
}
......

srs_service_st.cpp

......
srs_error_t srs_st_init()
{
......
int r0 = 0;
if((r0 = st_init()) != 0){
return srs_error_new(ERROR_ST_INITIALIZE, "st initialize failed, r0=%d", r0);
}
......

在srs_service_st.cpp中调用StateThreads库的初始化函数,完成StateThreads的初始化。

SRS之StateThreads学习的更多相关文章

  1. 开源流媒体服务器SRS学习笔记(1) - 安装、推流、拉流

    SRS(Simple RTMP Server)  是国人写的一款非常优秀的开源流媒体服务器软件,可用于直播/录播/视频客服等多种场景,其定位是运营级的互联网直播服务器集群. 一.安装 官网提供了3种安 ...

  2. 开源流媒体服务器SRS学习笔记(4) - Cluster集群方案

    单台服务器做直播,总归有单点风险,利用SRS的Forward机制 + Edge Server设计,可以很容易搭建一个大规模的高可用集群,示意图如下 源站服务器集群:origin server clus ...

  3. 开源流媒体服务器SRS学习笔记(3) - HTTPCallback实现安全认证

    按上回继续,安全论证是绝大多数应用的基本要求,如果任何人都能无限制的发布/播放视频,显然不适合.SRS中可以通过HTTPCallback机制来实现,参考下面的配置: ... vhost __defau ...

  4. 开源流媒体服务器SRS学习笔记(2) - rtmp / http-flv / hls 协议配置 及跨域问题

    对rtmp/http-flv/hls这三种协议不熟悉的同学,强烈建议先看看网友写的这篇文章科普下:理解RTMP.HttpFlv和HLS的正确姿势 .   srs可以同时支持这3种协议,只要修改conf ...

  5. 关于直播学习笔记-003-nginx-rtmp、srs、vlc、obs

    服务器 1.nginx-rtmp:https://github.com/illuspas/nginx-rtmp-win32 2.srs:https://github.com/illuspas/srs- ...

  6. 关于直播学习笔记-004-nginx-rtmp、srs、vlc、obs

    1.采集端:OBS RTMP推流地址:rtmp://192.168.198.21:1935/live 流密钥:livestream(任意-但播放地址与此一致) 2.播放端:nginx-rtmp-win ...

  7. Android学习——windows下搭建Cygwin环境

    在上一篇博文<Android学习——windows下搭建NDK_r9环境>中,我们详细的讲解了在windows下进行Android NDK开发环境的配置,我们也讲到了在NDk r7以后,我 ...

  8. ArcGIS api fo silverlight学习一(silverlight加载GeoServer发布的WMS地图)

    最好的学习资料ArcGIS api fo silverlight官网:http://help.arcgis.com/en/webapi/silverlight/samples/start.htm 一. ...

  9. cocos2dx进阶学习之屏幕适配

    背景 在学习cocos2dx时,我们在main函数中发现一句代码, #include "main.h" #include "AppDelegate.h" #in ...

  10. osgEarth学习笔记(转载)

    osgEarth学习笔记1.        通过earth文件创建图层时,可以指定多个影像数据源和多个高程数据源,数据源的顺序决定渲染顺序,在earth文件中处于最前的在渲染时处于最底层渲染:所以如果 ...

随机推荐

  1. 无法加载 DLL“xxxx.dll”: 找不到指定的模块。 (异常来自 HRESULT:0x8007007E)。

    有一台服务器在执行接口的时候遇到了这样一个问题: 其他服务器上都没有这个问题,IIS部署好的项目目录的bin文件夹下是有这个dll的,但却提示无法加载,在网上找了好多帖子,终于发现了问题. 首先用De ...

  2. socket搭建web服务端

    import socket from threading import Thread import time def html(conn): time_tag = str(time.time()) p ...

  3. iOS16新特性 | 灵动岛适配开发与到家业务场景结合的探索实践

    作者:京东零售 姜海 灵动岛是苹果在iPhone 14 Pro和iPhone 14 Pro Max上首次提出的全新UI交互形式,创新性的让虚拟软件和硬件的交互变得更为流畅.当有来电.短信等通知时,灵动 ...

  4. 扎实打牢数据结构算法根基,从此不怕算法面试系列之004 week01 02-04 使用泛型实现线性查找法

    1.算法描述 在数组中逐个查找元素,即遍历. 2.上一篇文的实现结果 在 扎实打牢数据结构算法根基,从此不怕算法面试系列之003 week01 02-03 代码实现线性查找法中,我们实现了如下代码: ...

  5. 第138篇:了解HTTP协议(TCP/IP协议,DNS域名解析,浏览器缓存)

    好家伙,发现自己的网络知识十分匮乏,赶紧补一下   这里先举个我生活中的例子 欸,作业不会写了,上网搜一下 用edge浏览器上bing必应搜一下(百度广告太多了,真不想用百度举例子)   假设这是我们 ...

  6. 从不均匀性角度浅析AB实验

    作者:京东零售 路卫强 本篇的目的是从三个不均匀性的角度,对AB实验进行一个认知的普及,最终着重讲述AB实验的一个普遍的问题,即实验准确度问题. 一.AB实验场景 在首页中,我们是用红色基调还是绿色基 ...

  7. cryptohack wp day(3)

    第二节模运算----第一题( GCD ) 在做这道题前,了解下欧几里得算法: 欧几里得算法,也叫辗转相除法,用于求解两个非负整数a和b的最大公约数(Greatest Common Divisor, G ...

  8. 【已解决】使用代理后,登陆微软账号提示0x800190001

    今天晚上想要登录Onedrive同步文件时,发现怎么都登陆不上去,报出的错误代码是0x80190001,在网上搜索了各种方法,重置网络,重置Onedrive都没什么用,甚至把Onedrive重装了一遍 ...

  9. Navicat Premium 16 安装教程

    使用数据库时经常会使用到Navicat,码一个教程 转载自https://www.bilibili.com/read/cv21586676?spm_id_from=444.41.list.card_a ...

  10. 我自己写了一个波场(Tron)本地网页版钱包

    最近由于项目需要,需要给每个用户分配一个充币地址,考虑到钱包安全性和方便管理,于是自己动手写了一个本地网页版的钱包,附上源代码跟大家交流下. Github 源代码地址 钱包和项目是分离的,项目通过鉴权 ...