`

PostgreSQL启动过程中的那些事七:初始化共享内存和信号九:shmem中初始化Predicate锁

阅读更多

 

       这一节 pg 初始化 predicate 锁,支持可序列化事务隔离。通过 InitPredicateLocks 例程实现,主要是干了下面这么几件事:

A 创建了哈希表"PREDICATELOCKTARGET hash"

B 在上面的哈希表里增加了 ScratchTargetTag 结构的索引

C 创建了哈希表"PREDICATELOCK hash"

D 初始化了 "PredXactList" 相关结构

E 创建了哈希表"SERIALIZABLEXID hash"

F 初始化了RWConflictPool 相关结构

H 初始化了 "FinishedSerializableTransactions" 相关结构

I 初始化了 "OldSerXid SLRU Ctl" 相关结构

J 初始化了 "OldSerXidControlData" 相关结构

 

1 先上个图,看一下函数调用过程梗概,中间略过部分细节

 


初始化 PredicateLocks 方法调用流程图

 

2 初始化 PredicateLocks 相关结构

话说 main()-> ->PostmasterMain()-> ->reset_shared() -> CreateSharedMemoryAndSemaphores()> ->InitPredicateLocks() 做了下面相关结构的内存分配和初始化:

A 创建了哈希表"PREDICATELOCKTARGET hash"

B 在上面的哈希表里增加了 ScratchTargetTag 结构的索引

C 创建了哈希表"PREDICATELOCK hash"

D 初始化了 "PredXactList" 相关结构

E 创建了哈希表"SERIALIZABLEXID hash"

F 初始化了RWConflictPool 相关结构,用于处理读写冲突

H 初始化了 "FinishedSerializableTransactions" 相关结构

I 初始化了 "OldSerXid SLRU Ctl" 相关结构

J 初始化了 "OldSerXidControlData" 相关结构

初始化上面这些结构都会在共享内存 /shmem 哈希表索引 shmemIndex 中增加索引项,下面把初始化这些结构后的 shmemIndex 图放到下边,就不加一个索引一个图了,减小篇幅。不过这样就看不出先后顺序了,好在这些消失的信息对理解这些过程没什么影响。图中黄色的索引项就是本节新增加的索引项。


初始化完 PredicateLocks 相关结构 的共享内存结构图

       为了精简上图,把创建 shmem 的哈希表索引 "ShmemIndex" 时创建的 HCTL 结构删掉了,这个结构的作用是记录创建可扩展哈希表的相关信息,不过这个结构在 "ShmemIndex" 创建完成后也会由于出了对象作用域而消失。增加了左边灰色底的部分,描述 共享内存 /shmem 里各变量物理布局概览,由下往上,由低地址到高地址。 图中黄色的索引项就是本节新增加的索引项。

 

A 创建了哈希表"PREDICATELOCKTARGET hash"

InitPredicateLocks()->ShmemInitHash ()->ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "PREDICATELOCKTARGET hash" ,如果没有,就在 shmemIndex 中给 "PREDICATELOCKTARGET hash" 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "PREDICATELOCKTARGET hash" 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "PREDICATELOCKTARGET hash" 相关结构(见下面 "PREDICATELOCKTARGET hash" 相关结构图 )分配空间,设置 entry (在这儿即ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小,然后返回 ShmemInitHash() ,调用 hash_create() ,创建哈希表 "PREDICATELOCKTARGET hash" ,最后返回 ShmemInitHash() ,让 HTAB * 类型静态 全局变量 PredicateLockTargetHash 指向 哈希表 "PREDICATELOCKTARGET hash"

相关结构定义见下面:

typedef struct PREDICATELOCKTARGETTAG

{

    uint32      locktag_field1 ; /* a 32-bit ID field */

    uint32      locktag_field2 ; /* a 32-bit ID field */

    uint32      locktag_field3 ; /* a 32-bit ID field */

    uint32      locktag_field4 ; /* a 32-bit ID field */

    uint32      locktag_field5 ; /* a 32-bit ID field */

} PREDICATELOCKTARGETTAG ;

typedef struct PREDICATELOCKTARGET

{

    /* hash key */

    PREDICATELOCKTARGETTAG tag ; /* unique identifier of lockable object */

 

    /* data */

    SHM_QUEUE   predicateLocks ; /* list of PREDICATELOCK objects assoc . with

                              * predicate lock target */

} PREDICATELOCKTARGET ;

 


"PREDICATELOCKTARGET hash" 相关结构图

 

B 在上面的哈希表里增加了 ScratchTargetTag 结构的索引

InitPredicateLocks() 调用 ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "PREDICATELOCKTARGET hash" 中查找 ScratchTargetTag (是 PREDICATELOCKTARGET 类型全局静态变量 ),如果没有,就把 ScratchTargetTag 作为哈希表 "PREDICATELOCKTARGET hash" (该哈希表索引的类型是 PREDICATELOCKTARGET )的第一个 索引项(参见 "PREDICATELOCKTARGET hash" 相关结构图 )。

C 创建了哈希表"PREDICATELOCK hash"

接着 InitPredicateLocks()->ShmemInitHash ()->ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "PREDICATELOCK hash" ,如果没有,就在 shmemIndex 中给 "PREDICATELOCK hash" 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "PREDICATELOCK hash" 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "PREDICATELOCK hash" 相关结构(见下面 "PREDICATELOCK hash" 相关结构图 )分配空间,设置 entry (在这儿即ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小,然后返回 ShmemInitHash() ,调用 hash_create() ,创建哈希表 "PREDICATELOCK hash" ,最后返回 ShmemInitHash() ,让 HTAB * 类型静态 全局变量 PredicateLockHash 指向 哈希表 "PREDICATELOCK hash"

相关结构定义见下面:

typedef struct PREDICATELOCKTAG

{

    PREDICATELOCKTARGET * myTarget ;

    SERIALIZABLEXACT * myXact ;

} PREDICATELOCKTAG ;

typedef struct PREDICATELOCK

{

    /* hash key */

    PREDICATELOCKTAG tag ;       /* unique identifier of lock */

 

    /* data */

    SHM_QUEUE   targetLink ;       /* list link in PREDICATELOCKTARGET's list of

                              * predicate locks */

    SHM_QUEUE   xactLink ;     /* list link in SERIALIZABLEXACT's list of

                               * predicate locks */

    SerCommitSeqNo commitSeqNo ; /* only used for summarized predicate locks */

} PREDICATELOCK ;

 


"PREDICATELOCK hash" 相关结构图

 

D 初始化了 "PredXactList" 相关结构

InitPredicateLocks() 调用 ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "PredXactList " ,如果没有,就在 shmemIndex 中给 "PredXactList " 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "PredXactList " 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "PredXactList " 相关结构(见下面“ PredXactList 相关结构图” )分配空间,设置 entry (在这儿及ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小 最后返回 InitPredicateLocks () ,让 PredXactList * 类型 全局变量 PredXact 指向 所分配内存 ,设置 PredXactList 结构类型的成员值。

相关结构定义见下面:

typedef struct PredXactListData

{

    SHM_QUEUE   availableList ;

    SHM_QUEUE   activeList ;

 

    /*

      * These global variables are maintained when registering and cleaning up

      * serializable transactions.  They must be global across all backends ,

      * but are not needed outside the predicate.c source file. Protected by

      * SerializableXactHashLock.

      */

    TransactionId SxactGlobalXmin ;     /* global xmin for active serializable

                                     * transactions */

    int         SxactGlobalXminCount ;    /* how many active serializable

                                     * transactions have this xmin */

    int         WritableSxactCount ;      /* how many non-read-only serializable

                                     * transactions are active */

    SerCommitSeqNo LastSxactCommitSeqNo ;      /* a strictly monotonically

                                            * increasing number for

                                            * commits of serializable

                                            * transactions */

    /* Protected by SerializableXactHashLock. */

    SerCommitSeqNo CanPartialClearThrough ;    /* can clear predicate locks

                                            * and inConflicts for

                                            * committed transactions

                                            * through this seq no */

    /* Protected by SerializableFinishedListLock. */

    SerCommitSeqNo HavePartialClearedThrough ; /* have cleared through this

                                            * seq no */

    SERIALIZABLEXACT * OldCommittedSxact ;      /* shared copy of dummy sxact */

 

    PredXactListElement element ;

}   PredXactListData ;

 

typedef struct PredXactListData * PredXactList ;

 

typedef struct SERIALIZABLEXACT

{

    VirtualTransactionId vxid ;  /* The executing process always has one of

                              * these. */

 

    /*

      * We use two numbers to track the order that transactions commit. Before

      * commit, a transaction is marked as prepared, and prepareSeqNo is set.

      * Shortly after commit, it's marked as committed, and commitSeqNo is set.

      * This doesn't give a strict commit order, but these two values together

      * are good enough for us, as we can always err on the safe side and

      * assume that there's a conflict, if we can't be sure of the exact

      * ordering of two commits.

      *

      * Note that a transaction is marked as prepared for a short period during

      * commit processing, even if two-phase commit is not used. But with

      * two-phase commit, a transaction can stay in prepared state for some

      * time.

      */

    SerCommitSeqNo prepareSeqNo ;

    SerCommitSeqNo commitSeqNo ;

 

    /* these values are not both interesting at the same time */

    union

    {

       SerCommitSeqNo earliestOutConflictCommit ;     /* when committed with

                                                   * conflict out */

       SerCommitSeqNo lastCommitBeforeSnapshot ;      /* when not committed or

                                                   * no conflict out */

    }          SeqNo ;

    SHM_QUEUE   outConflicts ; /* list of write transactions whose data we

                              * couldn't read. */

    SHM_QUEUE   inConflicts ;  /* list of read transactions which couldn't

                              * see our write. */

    SHM_QUEUE   predicateLocks ; /* list of associated PREDICATELOCK objects */

    SHM_QUEUE   finishedLink ; /* list link in

                              * FinishedSerializableTransactions */

 

    /*

      * for r/o transactions: list of concurrent r/w transactions that we could

      * potentially have conflicts with, and vice versa for r/w transactions

      */

    SHM_QUEUE   possibleUnsafeConflicts ;

 

    TransactionId topXid ;       /* top level xid for the transaction, if one

                              * exists; else invalid */

    TransactionId finishedBefore ;      /* invalid means still running; else

                                     * the struct expires when no

                                     * serializable xids are before this. */

    TransactionId xmin ;         /* the transaction's snapshot xmin */

    uint32      flags ;        /* OR'd combination of values defined below */

    int         pid ;          /* pid of associated process */

} SERIALIZABLEXACT ;

 


PredXactList 相关结构图

 

E 创建了哈希表"SERIALIZABLEXID hash"

接着 InitPredicateLocks() 调用 ShmemInitHash ()->ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "SERIALIZABLEXID hash" ,如果没有,就在 shmemIndex 中给 "SERIALIZABLEXID hash" 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "SERIALIZABLEXID hash" 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "SERIALIZABLEXID hash" 相关结构(见下面 "SERIALIZABLEXID hash" 相关结构图 )分配空间,设置 entry (在这儿即ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小,然后返回 ShmemInitHash() ,调用 hash_create() ,创建哈希表 "SERIALIZABLEXID hash" ,最后返回 ShmemInitHash() ,让 HTAB * 类型静态 全局变量 SerializableXidHash 指向 哈希表 "SERIALIZABLEXID hash"

相关结构定义见下面:

typedef struct SERIALIZABLEXIDTAG

{

    TransactionId xid ;

} SERIALIZABLEXIDTAG ;

typedef struct SERIALIZABLEXID

{

    /* hash key */

    SERIALIZABLEXIDTAG tag ;

 

    /* data */

    SERIALIZABLEXACT * myXact ;   /* pointer to the top level transaction data */

} SERIALIZABLEXID ;

 

 



 

"SERIALIZABLEXID hash" 相关结构图

 

F 初始化了RWConflictPool 相关结构,用于处理读写冲突

InitPredicateLocks() 调用 ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "RWConflictPool " ,如果没有,就在 shmemIndex 中给 "RWConflictPool " 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "RWConflictPool " 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "RWConflictPool " 相关结构(见下面“ RWConflictPool FinishedSerializableTransactions 相关结构图相关结构图” )分配空间,设置 entry (在这儿及ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小 最后返回 InitPredicateLocks () ,让 RWConflictPoolHeaderData * 类型 全局变量 RWConflictPool 指向 所分内存 ,设置其中 RWConflictPoolHeaderData 结构类型的成员值。

相关结构定义见下面:

typedef struct RWConflictData

{

    SHM_QUEUE   outLink ;      /* link for list of conflicts out from a sxact */

    SHM_QUEUE   inLink ;           /* link for list of conflicts in to a sxact */

    SERIALIZABLEXACT * sxactOut ;

    SERIALIZABLEXACT * sxactIn ;

}   RWConflictData ;

typedef struct RWConflictPoolHeaderData

{

    SHM_QUEUE   availableList ;

    RWConflict element ;

}   RWConflictPoolHeaderData ;

 


RWConflictPool FinishedSerializableTransactions 相关结构图

 

H 初始化了 "FinishedSerializableTransactions" 相关结构

InitPredicateLocks() 调用 ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "FinishedSerializableTransactions " ,如果没有,就在 shmemIndex 中给 "FinishedSerializableTransactions " 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "FinishedSerializableTransactions " 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "FinishedSerializableTransactions " 相关结构(见下面“ RWConflictPool FinishedSerializableTransactions 相关结构图相关结构图” )分配空间,设置 entry (在这儿及ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小 最后返回 InitPredicateLocks () ,让 SHM_QUEUE * 类型 全局变量 FinishedSerializableTransactions 指向 所分 内存,设置其中SubTransCtlData 结构类型的成员值。

 

I 初始化了 "OldSerXid SLRU Ctl" 相关结构

InitPredicateLocks()->OldSerXidInit()->SimpleLruInit()->ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "OldSerXid SLRU Ctl " ,如果没有,就在 shmemIndex 中给 "OldSerXid SLRU Ctl " 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "OldSerXid SLRU Ctl " 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "OldSerXid SLRU Ctl " 相关结构(见下面“ OldSerXid SLRU Ctl OldSerXidControlData 相关结构图” )分配空间,设置 entry (在这儿及ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小 最后返回 SimpleLruInit () ,让 SlruCtlData * 类型 全局变量 OldSerXidSlruCtl 指向给 "OldSerXid SLRU Ctl " 相关结构分配的内存起始地址,设置其中 SlruCtlData 结构类型的成员值。

 

J 初始化了 "OldSerXidControlData" 相关结构

InitPredicateLocks()->OldSerXidInit() 调用 ShmemInitStruct() 在其中 调用 hash_search() 在哈希表索引 "ShmemIndex" 中查找 "OldSerXidControlData " ,如果没有,就在 shmemIndex 中给 "OldSerXidControlData " 分一个 HashElement ShmemIndexEnt entry ,在其中的 Entry 中写上 "OldSerXidControlData " 。返回 ShmemInitStruct() ,再调用 ShmemAlloc() 在共享内存上给 "OldSerXidControlData " 相关结构(见下面“ OldSerXid SLRU Ctl OldSerXidControlData 相关结构图” )分配空间,设置 entry (在这儿及ShmemIndexEnt 类型变量)的成员 location 指向该空间, size 成员记录该空间大小 最后返回 OldSerXidInit () ,让 OldSerXidControlData * 类型 全局变量 oldSerXidControl 指向给 "OldSerXidControlData " 相关结构分配的内存地址,设置其中 OldSerXidControlDat 结构类型的成员值。

相关结构定义见下面:

typedef struct OldSerXidControlData

{

    int         headPage ;     /* newest initialized page */

    TransactionId headXid ;      /* newest valid Xid in the SLRU */

    TransactionId tailXid ;      /* oldest xmin we might be interested in */

    bool        warningIssued ;    /* have we issued SLRU wrap-around warning? */

}   OldSerXidControlData ;

 


OldSerXid SLRU Ctl OldSerXidControlData 相关结构图



 

 

  • 大小: 142.3 KB
  • 大小: 423.4 KB
  • 大小: 144.6 KB
  • 大小: 112.2 KB
  • 大小: 146.4 KB
  • 大小: 52.3 KB
  • 大小: 104.9 KB
  • 大小: 61.7 KB
1
0
分享到:
评论

相关推荐

    nacos-2.0.1 postgresql初始化脚本

    nacos-2.0.1 postgresql初始化脚本

    quartz-2.2.3版本的quartz初始化sql语句

    在Quartz 2.2.3版本中,初始化数据库是使用Quartz的关键步骤,因为Quartz依赖于一个持久化存储来保存作业和触发器的信息。这个过程通常涉及执行一系列SQL语句来创建必要的表结构。 Quartz的初始化SQL语句主要用于...

    关于PostGreSQL中的存储过程

    其中,存储过程是一个非常重要的概念,本文将对 PostgreSQL 中的存储过程进行详细的介绍和解释。 什么是存储过程 存储过程是一组为了完成特定任务而编写的 SQL 语句集合。它可以将复杂的操作封装起来,以便于重复...

    Postgresql存储过程

    Postgresql存储过程详解 Postgresql存储过程是指在Postgresql数据库中定义的一组SQL语句的...Postgresql存储过程是Postgresql数据库中的一种强大工具,可以实现复杂的操作,提高数据库服务器的性能和应用程序的性能。

    PostgreSQL中文手册9.2

    一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: 一、服务器进程的启动和关闭: . 50 PostgreSQL PostgreSQL...

    8基础 5:初始化 MySQL 数据库并建立连接(3).md

    在本节内容中,我们将深入了解如何使用Go语言和GORM库初始化MySQL数据库,并建立与该数据库的连接。GORM是一个流行的Go语言ORM(对象关系映射)库,它允许开发者通过编程方式与数据库交互,而无需编写大量的SQL代码...

    linux配置postgresql

    在 Linux 系统中,可以通过以下命令启动 PostgreSQL: ``` # su – postgres pg$ /usr/local/pgsql/bin/postmaster ``` 这将启动 PostgreSQL 数据库服务器,并允许用户访问数据库。 配置 PostgreSQL 数据库需要多个...

    在windows下手动初始化PostgreSQL数据库教程

    在初始化过程中,initdb会生成一系列的配置文件和数据库模板,并设置默认的最大连接数、共享缓冲区等参数。完成初始化之后,会得到一系列的成功消息,表明数据库系统已经准备就绪。 最后,使用pg_ctl工具来启动...

    postgresql-42.3.1-API文档-中文版.zip

    赠送jar包:postgresql-42.3.1.jar; 赠送原API文档:postgresql-42.3.1-javadoc.jar; 赠送源代码:postgresql-42.3.1-sources.jar;...人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请放心使用。

    PostgreSQL修炼之道 从小工到专家.pptx

    * PostgreSQL配置:包括设置数据库参数、内存管理和日志记录等方面。 数据类型与表达式 * 数据类型:包括整数、字符串、日期时间、布尔值等基本数据类型。 * 表达式:包括算术运算符、比较运算符、逻辑运算符和...

    Linux下PostgreSQL安装与开机启动

    初始化数据库是设置PostgreSQL的重要步骤之一,它创建数据库集群。 ##### 操作命令: ```bash su postgres /usr/local/pgsql/bin/initdb -D /usr/local/pgsql/data ``` **解释:** - `su postgres`:切换到`...

    windows环境下新版12.2postgreSQL的安装+初始化配置+启动

    总之,安装和配置Windows上的PostgreSQL 12.2涉及到下载安装文件、初始化数据库、设置服务启动和配置连接参数等步骤。遵循这些步骤,你就可以顺利地在Windows环境中搭建起一个功能齐全的PostgreSQL数据库系统了。 ...

    Ubuntu 下源码安装Postgresql

    七、初始化数据库目录 使用 initdb 命令初始化数据库目录: ./initdb -D PGDATA/postgres 八、启动数据库 使用以下命令启动数据库: ./postgres -D PGDATA/postgres 九、配置开机启动 为了使数据库在开机时...

    postgresql-42.2.5-API文档-中英对照版.zip

    赠送jar包:postgresql-42.2.5.jar; 赠送原API文档:postgresql-42.2.5-javadoc.jar;...人性化翻译,文档中的代码和结构保持不变,注释和说明精准翻译,请放心使用。 双语对照,边学技术、边学英语。

    postgresql8.2.3 中文文档

    1. **安装与配置**:这部分内容会指导用户如何在不同操作系统上安装 PostgreSQL,包括设置数据目录、初始化数据库集群、配置服务器参数以及启动和停止服务。 2. **SQL语言**:文档详细介绍了SQL的使用,包括数据...

    Postgresql-10安装包

    10. **阅读官方文档**:PostgreSQL 提供详细的官方文档,包括安装、配置和使用指南,强烈建议在安装和使用过程中参考。 **四、安装脚本的使用** 如果你从博主那里获取了安装脚本,通常这是一个包含上述步骤的自动...

    MySQL和PostgreSQL的比较

    一个实例同样可以管理多个数据库,但这些数据库被组织成一个集群,存储在一个初始化时设定的磁盘区域中,该区域由一个目录构成,存储着所有数据。首次数据库创建则通过`initdb`命令完成。值得注意的是,PostgreSQL也...

    PostGreSQL安装部署系列:Centos 7.9 安装指定PostGreSQL-15版本数据库

    初始化数据库是PostgreSQL安装过程中必不可少的一步,可以通过执行以下命令完成: ```bash sudo /usr/pgsql-15/bin/postgresql-15-setup initdb ``` 初始化完成后,将会创建数据库目录并设置默认权限。 ##### 3.4...

    框架使用到的初始化脚本

    在IT行业中,数据库初始化脚本是项目启动前的关键步骤,特别是在使用像Spring MVC这样的MVC框架构建应用时。本文将详细解析标题“框架使用到的初始化脚本”以及描述中涉及的知识点,主要关注与MySQL数据库相关的初始...

    新建SQL数据库并初始化

    这些文件对于理解数据库的创建和初始化过程具有重要价值。 总之,新建和初始化SQL数据库是一项关键的任务,涉及数据库管理系统的选择、安装、设计表结构、设置权限等多个环节。掌握这些技能对于任何IT专业人员来说...

Global site tag (gtag.js) - Google Analytics