Ubuntu 下 nginx-1.24.0 源码分析 - ngx_os_init 函数

news2025/2/22 10:52:33

ngx_os_init

声明在 src/os/unix/ngx_os.h

ngx_int_t ngx_os_init(ngx_log_t *log);

定义在 src\os\unix\ngx_posix_init.c

ngx_int_t
ngx_os_init(ngx_log_t *log)
{
    ngx_time_t  *tp;
    ngx_uint_t   n;
#if (NGX_HAVE_LEVEL1_DCACHE_LINESIZE)
    long         size;
#endif

#if (NGX_HAVE_OS_SPECIFIC_INIT)
    if (ngx_os_specific_init(log) != NGX_OK) {
        return NGX_ERROR;
    }
#endif

    if (ngx_init_setproctitle(log) != NGX_OK) {
        return NGX_ERROR;
    }

    ngx_pagesize = getpagesize();
    ngx_cacheline_size = NGX_CPU_CACHE_LINE;

    for (n = ngx_pagesize; n >>= 1; ngx_pagesize_shift++) { /* void */ }

#if (NGX_HAVE_SC_NPROCESSORS_ONLN)
    if (ngx_ncpu == 0) {
        ngx_ncpu = sysconf(_SC_NPROCESSORS_ONLN);
    }
#endif

    if (ngx_ncpu < 1) {
        ngx_ncpu = 1;
    }

#if (NGX_HAVE_LEVEL1_DCACHE_LINESIZE)
    size = sysconf(_SC_LEVEL1_DCACHE_LINESIZE);
    if (size > 0) {
        ngx_cacheline_size = size;
    }
#endif

    ngx_cpuinfo();

    if (getrlimit(RLIMIT_NOFILE, &rlmt) == -1) {
        ngx_log_error(NGX_LOG_ALERT, log, errno,
                      "getrlimit(RLIMIT_NOFILE) failed");
        return NGX_ERROR;
    }

    ngx_max_sockets = (ngx_int_t) rlmt.rlim_cur;

#if (NGX_HAVE_INHERITED_NONBLOCK || NGX_HAVE_ACCEPT4)
    ngx_inherited_nonblocking = 1;
#else
    ngx_inherited_nonblocking = 0;
#endif

    tp = ngx_timeofday();
    srandom(((unsigned) ngx_pid << 16) ^ tp->sec ^ tp->msec);

    return NGX_OK;
}

初始化操作系统相关的参数和配置,为 Nginx 的运行环境做好准备 

代码逻辑分析

  1. 操作系统特定初始化

    • 如果定义了 NGX_HAVE_OS_SPECIFIC_INIT,调用 ngx_os_specific_init 进行特定操作系统的初始化。
    • 如果失败,直接返回错误。
  2. 进程标题初始化

    • 调用 ngx_init_setproctitle 初始化进程标题设置功能。
    • 如果失败,直接返回错误。
  3. 系统参数初始化

    • 获取系统页面大小(getpagesize)并计算页面大小的对数(ngx_pagesize_shift)。
    • 初始化 CPU 缓存行大小(ngx_cacheline_size),如果支持 _SC_LEVEL1_DCACHE_LINESIZE,则从系统获取缓存行大小。
  4. CPU 核心数初始化

    • 如果定义了 NGX_HAVE_SC_NPROCESSORS_ONLN,通过 sysconf(_SC_NPROCESSORS_ONLN) 获取 CPU 核心数。
    • 如果核心数小于 1,则默认设置为 1。
  5. CPU 信息初始化

    • 调用 ngx_cpuinfo 获取 CPU 相关信息。
  6. 文件描述符限制

    • 使用 getrlimit 获取当前进程的最大文件描述符限制。
    • 如果获取失败,记录错误日志并返回错误。
  7. 非阻塞套接字标志

    • 根据是否支持 NGX_HAVE_INHERITED_NONBLOCKNGX_HAVE_ACCEPT4,设置 ngx_inherited_nonblocking 标志。
  8. 随机数种子初始化

    • 使用当前进程 ID、时间戳(秒和毫秒)生成随机数种子。
  9. 返回成功

    • 如果所有初始化步骤都成功,返回 NGX_OK

 

详解

函数签名

ngx_int_t
ngx_os_init(ngx_log_t *log)

参数:

ngx_log_t *log :日志对象指针,用于记录错误或调试信息。

返回值:

ngx_int_t Nginx 自定义的一个整数类型

  • NGX_OK :表示函数执行成功,所有初始化步骤均顺利完成。
  • NGX_ERROR :表示函数执行失败,某些初始化步骤未能完成

局部变量声明

    ngx_time_t  *tp;
    ngx_uint_t   n;
#if (NGX_HAVE_LEVEL1_DCACHE_LINESIZE)
    long         size;
#endif
  • tp:指向时间结构体 ngx_time_t 的指针,用于获取当前时间。
  • n:无符号整数,用于计算页面大小的对数。
  • size(条件编译):长整型变量,用于存储 CPU 缓存行大小。

NGX_HAVE_LEVEL1_DCACHE_LINESIZE

是 Nginx 源码中的一个宏,用于指示当前系统是否支持获取 CPU 一级缓存(L1 Data Cache)的缓存行大小

定义在 objs/ngx_auto_config.h 中

#ifndef NGX_HAVE_LEVEL1_DCACHE_LINESIZE
#define NGX_HAVE_LEVEL1_DCACHE_LINESIZE  1
#endif

操作系统特定初始化

#if (NGX_HAVE_OS_SPECIFIC_INIT)
    if (ngx_os_specific_init(log) != NGX_OK) {
        return NGX_ERROR;
    }
#endif
  • NGX_HAVE_OS_SPECIFIC_INIT
  • 这是一个宏,表示是否需要执行特定操作系统的初始化逻辑。
  • ngx_os_specific_init
  • 调用特定于操作系统的初始化函数。如果初始化失败,直接返回 NGX_ERROR
  • 意图 :通过条件编译,支持不同操作系统的定制化初始化逻辑,增强跨平台兼容性。

NGX_HAVE_OS_SPECIFIC_INIT

定义在 src\os\unix\ngx_linux_config.h

#define NGX_HAVE_OS_SPECIFIC_INIT    1

 使用 gcc -E 处理条件编译后,确认一下我当前 Ubuntu 环境下的实际情况

gcc -E src/os/unix/ngx_posix_init.c \
	-I src/core \
	-I src/event \
	-I src/event/modules \
	-I src/os/unix \
	-I objs \
	> ngx_posix_init_preprocessed.c

在输出文件中查找 ngx_os_init 函数

ngx_int_t
ngx_os_init(ngx_log_t *log)
{
    ngx_time_t *tp;
    ngx_uint_t n;

    long size;



    if (ngx_os_specific_init(log) != 0) {
        return -1;
    }


    if (ngx_init_setproctitle(log) != 0) {
        return -1;
    }

    ngx_pagesize = getpagesize();
    ngx_cacheline_size = 64;

    for (n = ngx_pagesize; n >>= 1; ngx_pagesize_shift++) { }


    if (ngx_ncpu == 0) {
        ngx_ncpu = sysconf(
# 60 "src/os/unix/ngx_posix_init.c" 3 4
                          _SC_NPROCESSORS_ONLN
# 60 "src/os/unix/ngx_posix_init.c"
                                              );
    }


    if (ngx_ncpu < 1) {
        ngx_ncpu = 1;
    }


    size = sysconf(
# 69 "src/os/unix/ngx_posix_init.c" 3 4
                  _SC_LEVEL1_DCACHE_LINESIZE
# 69 "src/os/unix/ngx_posix_init.c"
                                            );
    if (size > 0) {
        ngx_cacheline_size = size;
    }


    ngx_cpuinfo();

    if (getrlimit(
# 77 "src/os/unix/ngx_posix_init.c" 3 4
                 RLIMIT_NOFILE
# 77 "src/os/unix/ngx_posix_init.c"
                              , &rlmt) == -1) {
        if ((log)->log_level >= 2) ngx_log_error_core(2, log, 
# 78 "src/os/unix/ngx_posix_init.c" 3 4
       (*__errno_location ())
# 78 "src/os/unix/ngx_posix_init.c"
       , "getrlimit(RLIMIT_NOFILE) failed")
                                                        ;
        return -1;
    }

    ngx_max_sockets = (ngx_int_t) rlmt.rlim_cur;


    ngx_inherited_nonblocking = 1;




    tp = (ngx_time_t *) ngx_cached_time;
    srandom(((unsigned) ngx_pid << 16) ^ tp->sec ^ tp->msec);

    return 0;
}

ngx_os_specific_init

Ubuntu 下 nginx-1.24.0 源码分析 - ngx_os_specific_init函数-CSDN博客

 进程标题初始化

    if (ngx_init_setproctitle(log) != NGX_OK) {
        return NGX_ERROR;
    }
  • ngx_init_setproctitle :初始化进程标题设置功能,允许修改进程的命令行标题(如 ps 命令中显示的内容)
  • 意图 :方便管理员通过工具查看 Nginx 进程的状态(如主进程、工作进程等)

ngx_init_setproctitle

Ubuntu 下 nginx-1.24.0 源码分析 - ngx_init_setproctitle函数-CSDN博客

系统页面大小初始化

    ngx_pagesize = getpagesize();
    ngx_cacheline_size = NGX_CPU_CACHE_LINE;
    for (n = ngx_pagesize; n >>= 1; ngx_pagesize_shift++) { /* void */ }

 

  • getpagesize() :获取系统页面大小(单位为字节),例如 4KB。
  • ngx_pagesize :全局变量,存储页面大小。
  • ngx_cacheline_size :全局变量,存储 CPU 缓存行大小,默认值为 NGX_CPU_CACHE_LINE
  • for 循环 :计算页面大小的对数(ngx_pagesize_shift),即页面大小是 2 的多少次幂。例如,4KB 的页面大小对应 ngx_pagesize_shift = 12
  • 意图 :页面大小和缓存行大小是性能优化的重要参数,直接影响内存分配和 CPU 缓存效率。

getpagesize 

在 C 语言中,getpagesize() 函数用于获取系统内存页的大小(以字节为单位)

函数原型

int getpagesize(void);

返回值

  • 返回值是一个整数,表示系统内存页的大小(以字节为单位)
  • 例如,在许多现代系统上,返回值通常是 4096 字节(即 4KB)

需要包含以下头文件:

#include <unistd.h>

 NGX_CPU_CACHE_LINE

定义在 objs/ngx_auto_config.h

#ifndef NGX_CPU_CACHE_LINE
#define NGX_CPU_CACHE_LINE  64
#endif

CPU 核心数初始化

#if (NGX_HAVE_SC_NPROCESSORS_ONLN)
    if (ngx_ncpu == 0) {
        ngx_ncpu = sysconf(_SC_NPROCESSORS_ONLN);
    }
#endif
    if (ngx_ncpu < 1) {
        ngx_ncpu = 1;
    }

 

  • sysconf(_SC_NPROCESSORS_ONLN) :获取当前系统在线的 CPU 核心数。
  • ngx_ncpu :全局变量,存储 CPU 核心数。
  • 默认值处理 :如果核心数小于 1,则设置为 1(防止异常情况)。
  • 设计意图 :根据 CPU 核心数动态调整工作线程的数量,充分利用多核优势。

 NGX_HAVE_SC_NPROCESSORS_ONLN

定义在 objs/ngx_auto_config.h

#ifndef NGX_HAVE_SC_NPROCESSORS_ONLN
#define NGX_HAVE_SC_NPROCESSORS_ONLN  1
#endif

用于指示当前系统是否支持通过 sysconf(_SC_NPROCESSORS_ONLN) 获取在线 CPU 核心数。它的定义与否取决于目标操作系统和硬件平台的特性

不同的操作系统对 sysconf(_SC_NPROCESSORS_ONLN) 的支持情况不同:

  • Linux :完全支持 sysconf(_SC_NPROCESSORS_ONLN),因此该宏通常会被定义。
  • Windows :Windows 不支持 POSIX 标准的 sysconf 函数,因此该宏不会被定义

sysconf

函数原型

long sysconf(int name);
  • 返回值

    • 成功时,返回与 name 参数对应的系统配置值。
    • 如果发生错误(例如参数无效或功能不支持),返回 -1
  • 参数

    • name:指定要查询的系统配置选项。它是一个常量,表示不同的系统属性。
      • _SC_NPROCESSORS_ONLN 是其中一个常量,表示系统中当前在线的 CPU 核心数。

 CPU 缓存行大小初始化

#if (NGX_HAVE_LEVEL1_DCACHE_LINESIZE)
    size = sysconf(_SC_LEVEL1_DCACHE_LINESIZE);
    if (size > 0) {
        ngx_cacheline_size = size;
    }
#endif
  • sysconf(_SC_LEVEL1_DCACHE_LINESIZE) :获取一级缓存行大小。
  • 更新 ngx_cacheline_size :如果获取到的值有效,则更新全局变量。
  • 意图 :缓存行大小影响内存对齐和性能优化,动态获取可以适配不同的硬件架构。

 NGX_HAVE_LEVEL1_DCACHE_LINESIZE

定义在 objs/ngx_auto_config.h

#ifndef NGX_HAVE_LEVEL1_DCACHE_LINESIZE
#define NGX_HAVE_LEVEL1_DCACHE_LINESIZE  1
#endif

CPU 信息初始化

    ngx_cpuinfo();

 

  • ngx_cpuinfo :获取 CPU 的详细信息(如型号、特性等),并存储在全局变量中。

 ngx_cpuinfo

Ubuntu 下 nginx-1.24.0 源码分析 - ngx_cpuinfo 函数-CSDN博客

文件描述符限制初始化

    if (getrlimit(RLIMIT_NOFILE, &rlmt) == -1) {
        ngx_log_error(NGX_LOG_ALERT, log, errno,
                      "getrlimit(RLIMIT_NOFILE) failed");
        return NGX_ERROR;
    }
    ngx_max_sockets = (ngx_int_t) rlmt.rlim_cur;

 

  • getrlimit :获取当前进程的最大文件描述符限制。
  • rlmt :存储限制值的结构体。
  • ngx_max_sockets :全局变量,存储最大文件描述符数量。
  • 错误处理 :如果获取失败,记录错误日志并返回 NGX_ERROR
  • 意图 :文件描述符限制决定了 Nginx 能同时处理的最大连接数,合理设置可以避免资源耗尽。

getrlimit

getrlimit 是一个 POSIX 标准函数,用于获取当前进程的资源限制。

它允许程序查询操作系统对某种资源的硬限制(hard limit)和软限制(soft limit)

函数原型

int getrlimit(int resource, struct rlimit *rlim);
  • 返回值

    • 成功时返回 0
    • 失败时返回 -1,并设置 errno 表示错误原因。

(1) resource 参数

  • 类型 int

  • 含义 :指定要查询的资源类型。

  • 常用值

    • RLIMIT_NOFILE:表示文件描述符的最大数量(Number of Open Files)。
    • 其他可能的值包括:
      • RLIMIT_CPU:CPU 时间限制。
      • RLIMIT_DATA:数据段大小限制。
      • RLIMIT_STACK:栈大小限制。
  • 意图

    • 在这里,RLIMIT_NOFILE 被传递给 getrlimit,表示我们关心的是文件描述符的数量限制。

(2) rlim 参数

  • 类型 struct rlimit *
  • 含义 :指向一个 rlimit 结构体的指针,用于存储查询结果。
rlimit 结构体定义
struct rlimit {
    rlim_t rlim_cur;  // 软限制(Soft Limit)
    rlim_t rlim_max;  // 硬限制(Hard Limit)
};
  • rlim_cur

    • 当前生效的限制值(软限制)。
    • 进程可以动态调整软限制,但不能超过硬限制。
  • rlim_max

    • 最大允许的限制值(硬限制)。
    • 只有超级用户(root)才能修改硬限制。

意图

获取文件描述符限制

  • 文件描述符是操作系统用于管理打开文件、套接字等资源的抽象句柄。
  • 每个进程都有一个文件描述符表,其大小由 RLIMIT_NOFILE 决定。
  • 通过调用 getrlimit(RLIMIT_NOFILE, &rlmt),Nginx 获取当前进程的文件描述符限制:
    • rlmt.rlim_cur:当前允许的最大文件描述符数量(软限制)。
    • rlmt.rlim_max:理论上允许的最大文件描述符数量(硬限制)。

非阻塞套接字标志初始化

#if (NGX_HAVE_INHERITED_NONBLOCK || NGX_HAVE_ACCEPT4)
    ngx_inherited_nonblocking = 1;
#else
    ngx_inherited_nonblocking = 0;
#endif

 

  • ngx_inherited_nonblocking :全局变量,指示是否支持继承非阻塞套接字。
  • 条件编译 :根据是否支持 NGX_HAVE_INHERITED_NONBLOCKNGX_HAVE_ACCEPT4 设置标志。
  • 意图 :非阻塞套接字是高并发网络编程的基础,确保套接字行为一致。

NGX_HAVE_INHERITED_NONBLOCK

  • 含义

    • 表示当前系统是否支持继承非阻塞套接字(Inherited Non-blocking Sockets)。
    • 如果定义了该宏,则表示系统允许父进程创建的套接字在子进程中保持非阻塞状态。
  • 背景

    • 在传统的网络编程中,套接字的阻塞或非阻塞状态是由每个进程独立管理的。
    • 如果系统支持继承非阻塞套接字,则父进程设置的非阻塞状态可以直接被子进程继承,而无需额外的系统调用。
  • 优点

    • 减少了系统调用的开销,提高了性能。
    • 简化了多进程模型中的套接字管理逻辑。

NGX_HAVE_ACCEPT4

  • 含义

    • 表示当前系统是否支持 accept4 系统调用。
    • accept4 是 Linux 内核 2.6.28 引入的一个扩展版本的 accept 系统调用,允许在接收新连接时直接设置套接字选项(如非阻塞模式)。
  • 背景

    • 传统的 accept 系统调用仅返回一个新的套接字文件描述符,但无法直接设置套接字选项。
    • 使用 accept4 可以在接收连接的同时设置套接字为非阻塞模式或其他选项,从而减少额外的系统调用。
  • 优点

    • 提高了性能,减少了系统调用次数。
    • 简化了代码逻辑,避免了在 accept 后手动调用 fcntl 或其他函数来设置套接字选项。

 随机数种子初始化

    tp = ngx_timeofday();
    srandom(((unsigned) ngx_pid << 16) ^ tp->sec ^ tp->msec);
  • ngx_timeofday :获取当前时间戳。
  • srandom :设置随机数种子。
  • 种子生成公式 (进程 ID << 16) ^ 当前秒数 ^ 当前毫秒数
  • 意图 :随机数种子用于生成唯一的随机数序列,避免每次运行时生成相同的随机数。

ngx_timeofday

定义在 src/core/ngx_times.h

#define ngx_timeofday()      (ngx_time_t *) ngx_cached_time

srandom

srandom 是标准库函数,用于设置随机数生成器的种子。

种子值决定了随机数序列的初始状态。如果种子值不同,生成的随机数序列也会不同。

返回成功

    return NGX_OK;

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2303372.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Python项目】基于Python的语音数据及标注核对审核系统

【Python项目】基于Python的语音数据及标注核对审核系统 技术简介&#xff1a; 采用Python技术、MySQL数据库、Django框架等实现。 系统简介&#xff1a; 语音数据及标注核对审核系统是一个基于B/S架构的语音数据处理平台&#xff0c;旨在通过自动化的方式对语音数据进行标…

深入解析BFS算法:C++实现无权图最短路径的高效解决方案

在无权图中&#xff0c;广度优先搜索&#xff08;BFS&#xff09;是解决最短路径问题的高效算法。接下来博主从专业角度深入探讨其实现细节&#xff0c;并给出C代码示例&#xff1a; 目录 一、核心原理 二、算法步骤 三、C实现关键点 1. 数据结构 2. 边界检查 3. 路径回溯…

LeetCode刷题---二分查找---441

排列硬币 441. 排列硬币 - 力扣&#xff08;LeetCode&#xff09; 题目 你总共有 n 枚硬币&#xff0c;并计划将它们按阶梯状排列。对于一个由 k 行组成的阶梯&#xff0c;其第 i 行必须正好有 i 枚硬币。阶梯的最后一行 可能 是不完整的。 给你一个数字 n &#xff0c;计算…

Unity结合Vuforia虚拟按键实现AR机械仿真动画效果

零、最终效果 待上传 一、资源准备 1、Vuforia Vuforia版本不能高于10.17.4&#xff08;往上的版本虚拟按键功能被删除&#xff09; 2、Unity Unity版本必须要高于2022.3.x&#xff0c;不然使用Vuforia插件时会出现bug 二、主要内容 1、添加虚拟按钮 2、为虚拟按钮设置…

网络安全 linux学习计划 linux网络安全精要

&#x1f345; 点击文末小卡片 &#xff0c;免费获取网络安全全套资料&#xff0c;资料在手&#xff0c;涨薪更快 2.使用命令行 文件系统层次标准&#xff08;FHS&#xff09;是一个文件和目录在Unix和Linux操作系统上面应该如何存储的定义。 /bin 重要的二进制可执行程序/bo…

为AI聊天工具添加一个知识系统 之115 详细设计之56 知识表征 之2

本文要点 要点 知识表征的顶级范畴中最好是先将九个原语primitive T, ⊥, Independent, Relative, Mediating, Physical, Abstract, Continuant,和 Occurrent 进行分组&#xff08;分成2大组 和 4个小组&#xff09;并写出它们的满足公司&#xff0c;然后将它们和三种设计&am…

C#初级教程(1)——C# 与.NET 框架:探索微软平台编程的强大组合

图片来源&#xff1a; https://www.lvhang.site/docs/dotnettimeline 即梦AI - 一站式AI创作平台 一、历史发展脉络 在早期的微软平台编程中&#xff0c;常用的编程语言有 Visual Basic、C、C。到了 20 世纪 90 年代末&#xff0c;Win32 API、MFC&#xff08;Microsoft Found…

Mac m1 连接公司内网

1、创建VPN 1、在系统偏好设置 2、选择网络 3、进行添加 2、添加设置 1、选择VPN 2、类型选择L2TP/IPSec 3、填写服务器IP和账号 4、点击认证设置-填写密码 。然后应用 3、进行特殊配置 网上说苹果系统的问题。 1、创建命令 sudo vim /etc/ppp/options 2、添加内容-主要别…

C++:类与对象,定义类和构造函数

#define _CRT_SECURE_NO_WARNINGS 1 #include <iostream> using namespace std; //如何让定义一个类 // 封装 // 1、将数据和方法定义到一起。 // 2、把想给你看的数据给你看&#xff0c;不想给你看的封装起来。 通过访问限定符来实现 class Stack { public: //1.成…

Nginx环境安装

一、官网地址 Nginx官网&#xff1a;http://nginx.org/ Nginx中文网&#xff1a;https://nginx.p2hp.com/ 二、Nginx版本 mainline version 开发版本stableversion 稳定版本legacy version 历史版本 三、Windows系统安装Nginx 第一步&#xff1a;选择Windows版本&#xff0c;…

Spring AI + Ollama 实现调用DeepSeek-R1模型API

一、前言 随着人工智能技术的飞速发展&#xff0c;大语言模型&#xff08;LLM&#xff09;在各个领域的应用越来越广泛。DeepSeek 作为一款备受瞩目的国产大语言模型&#xff0c;凭借其强大的自然语言处理能力和丰富的知识储备&#xff0c;迅速成为业界关注的焦点。无论是文本生…

android系统SystemServer进程启动流程分析

目录 一,SystemServer整体框架 二,SystemServer启动源码分析 2.1,重要的概念 2.2,启动入口 2.3,创建对应进程的binder 三,binder驱动和binder线程池 四,SystemServer真正启动方法 4.1 SystemServer main方法里面主要做了几件事情 1)创建SystemServiceManager管理所有的…

Oracle 深入理解Lock和Latch ,解析访问数据块全流程

Oracle 锁机制介绍 根据保护对象的不同&#xff0c;单实例Oracle数据库锁可以分为以下几大类&#xff1a; DML lock&#xff08;data locks&#xff0c;数据锁&#xff09;&#xff1a;用于保护数据的完整性&#xff1b; DDL lock&#xff08;dictionary locks&#xff0c;字典…

如何基于transformers库通过训练Qwen/DeepSeek模型的传统分类能力实现文本分类任务

文章目录 模型与环境准备文档分析源码解读模型训练及推理方式进阶:CPU与显存的切换进阶:多卡数据并行训练🔑 DDP 训练过程核心步骤🚫 DDP 不适用于模型并行⚖️ DDP vs. Model Parallelism⚙️ 解决大模型训练的推荐方法🎉进入大模型应用与实战专栏 | 🚀查看更多专栏…

Unity中一个节点实现植物动态(Shader)

1 . 核心思路就操作顶点作往复运动&#xff1b; 核心代码&#xff1a; half stage1 dot(positionOS, float3(0, 1, 0)) * _Strength; half stage2 sin(dot(positionOS, float3(1, 0, 0)) * _Strength _Time.y * _Speed); half stage3 stage1 * stage2 * float3(0.001,…

PrimeTime:工具简介

相关阅读 PrimeTimehttps://blog.csdn.net/weixin_45791458/category_12900271.html?spm1001.2014.3001.5482 PrimeTime是PrimeTime Suite中的一个工具&#xff0c;能够执行全芯片级、门级的静态时序分析&#xff0c;这是芯片设计和分析流程中的一个关键部分。该工具通过检查…

【拜读】Tensor Product Attention Is All You Need姚期智团队开源兼容RoPE位置编码

姚期智团队开源新型注意力&#xff1a;张量积注意力&#xff08;Tensor Product Attention&#xff0c;TPA&#xff09;。有点像一种「动态的LoRA」&#xff0c;核心思路在于利用张量分解来压缩注意力机制中的 Q、K、V 表示&#xff0c;同时保留上下文信息&#xff0c;减少内存…

Docker-技术架构演进之路

目录 一、概述 常见概念 二、架构演进 1.单机架构 2.应用数据分离架构 3.应用服务集群架构 4.读写分离 / 主从分离架构 5.引入缓存 —— 冷热分离架构 6.垂直分库 7.业务拆分 —— 微服务 8.容器化引入——容器编排架构 三、尾声 一、概述 在进行技术学习过程中&am…

用Chrome Recorder轻松完成自动化测试脚本录制

前言 入门自动化测试,录制回放通常是小白测试首先用到的功能。而录制回放工具也一直是各大Web自动化测试必然会着重提供的一块功能。 早期WinRunner、QTP这样的工具,自动化测试可以说是围绕录制回放开展的。近年像Selenium也提供有录制工具 Selenium IDE,Playwright也包含…

python中的异常-模块-包

文章目录 异常异常的定义异常捕获语法捕获常规异常捕获指定异常捕获多个异常捕获所有异常异常else异常finally 异常传递总结 模块概念导入自定义模块及导入main方法all变量 总结 包自定义包定义pycharm中建包的基本步骤导入方式 第三方包 异常 异常的定义 当检测到一个错误时…