zoukankan      html  css  js  c++  java
  • HBase之六:HBase的RowKey设计

    数据模型

     我们可以将一个表想象成一个大的映射关系,通过行健、行健+时间戳或行键+列(列族:列修饰符),就可以定位特定数据,Hbase是稀疏存储数据的,因此某些列可以是空白的,

    Row Key

    Time Stamp

    Column Family:c1

    Column Family:c2

    r1

    t7

    c1:1

    value1-1/1

     

     

    t6

    c1:2

    value1-1/2

     

     

    t5

    c1:3

    value1-1/3

     

     

    t4

     

     

    c2:1

    value1-2/1

    t3

     

     

    c2:2

    value1-2/2

    t2

    t2

    c1:1

    value2-1/1

     

     

    t1

     

     

    c2:1

    value2-1/1

    从上表可以看出,test表有r1和r2两行数据,并且c1和c2两个列族,在r1中,列族c1有三条数据,列族c2有两条数据;在r2中,列族c1有一条数据, 列族c2有一条数据,每一条数据对应的时间戳都用数字来表示,编号越大表示数据越旧,反而表示数据越新。

    3:物理视图

        虽然从概念视图来看每个表格是由很多行组成的,但是在物理存储上面,它是按照列来保存的。

       

    Row Key

    Time Stamp

    Column Family:c1

    r1

    t7

    c1:1

    value1-1/1

    t6

    c1:2

    value1-1/2

    t5

    c1:3

    value1-1/3

                 表:HBase数据的物理视图(1)

    Row Key

    Time Stamp

    Column Family:c2

    r1

    t4

    c2:1

    value1-2/1

    t3

    c2:2

    value1-2/2

              表:HBase数据的物理视图(2)

    需要注意的是,在概念视图上面有些列是空白的,这样的列实际上并不会被存储,当请求这些空白的单元格时,会返回null值。如果在查询的时候不

    提供时间戳,那么会返回距离现在最近的那一个版本的数据,因为在存储的时候,数据会按照时间戳来排序。

     

    通过shell操作hbase 会更清楚结构

    这里我们用一个学生成绩表作为例子,对HBase的基本操作和基本概念进行讲解:

    下面是学生的成绩表:

    name grad      course:math   course:art

    Tom    1                87                    97

    Jerry   2            100                  80

            这里grad对于表来说是一个列,course对于表来说是一个列族,这个列族由两个列组成:math和art,当然我们可以根据我们的需要在course中建立更多的列族,如computer,physics等相应的列添加入course列族.

            有了上面的想法和需求,我们就可以在HBase中建立相应的数据表啦!

    1, 建立一个表格 scores 具有两个列族grad 和courese

    hbase(main):002:0> create 'scores', 'grade', 'course'

    0 row(s) in 4.1610 seconds

    2,查看当先HBase中具有哪些表

    hbase(main):003:0> list

    scores

    1 row(s) in 0.0210 seconds

    3,查看表的构造

    hbase(main):004:0> describe 'scores'

    {NAME => 'scores', IS_ROOT => 'false', IS_META => 'false', FAMILIES => [{NAME => 'course', BLOOMFILTER => 'false', IN_MEMORY => 'false', LENGTH => '2147483647', BLOCKCACHE => 'false', VERSIONS => '3', TTL => '-1', COMPRESSION => 'NONE'}, {NAME => 'grade', BLOOMFILTER => 'false', IN_MEMORY => 'false', LENGTH => '2147483647', BLOCKCACHE => 'false', VERSIONS => '3', TTL => '-1', COMPRESSION => 'NONE'}]}

    1 row(s) in 0.0130 seconds

    4, 加入一行数据,行名称为 Tom 列族grad的列名为”” 值位1

    hbase(main):005:0> put 'scores', 'Tom', 'grade:', '1'

    0 row(s) in 0.0070 seconds

    5,给Tom这一行的数据的列族添加一列 <math,87>

    hbase(main):006:0> put 'scores', 'Tom', 'course:math', '87'

    0 row(s) in 0.0040 seconds

    6,给Tom这一行的数据的列族添加一列 <art,97>

    hbase(main):007:0> put 'scores', 'Tom', 'course:art', '97'

    0 row(s) in 0.0030 seconds

    7, 加入一行数据,行名称为 Jerry 列族grad的列名为”” 值位2

    hbase(main):008:0> put 'scores', 'Jerry', 'grade:', '2'

    0 row(s) in 0.0040 seconds

    8,给Jerry这一行的数据的列族添加一列 <math,100>

    hbase(main):009:0> put 'scores', 'Jerry', 'course:math', '100'

    0 row(s) in 0.0030 seconds

    9,给Jerry这一行的数据的列族添加一列 <art,80>

    hbase(main):010:0> put 'scores', 'Jerry', 'course:art', '80'

    0 row(s) in 0.0050 seconds

    10,查看scores表中Tom的相关数据

    hbase(main):011:0> get 'scores', 'Tom'

    COLUMN                       CELL

    course:art                  timestamp=1224726394286, value=97

    course:math                 timestamp=1224726377027, value=87

    grade:                      timestamp=1224726360727, value=1

    3 row(s) in 0.0070 seconds

    11,查看scores表中所有数据

    hbase(main):012:0> scan 'scores'

    ROW                          COLUMN+CELL

    Tom                         column=course:art, timestamp=1224726394286, value=97

    Tom                         column=course:math, timestamp=1224726377027, value=87

    Tom                         column=grade:, timestamp=1224726360727, value=1

    Jerry                        column=course:art, timestamp=1224726424967, value=80

    Jerry                        column=course:math, timestamp=1224726416145, value=100

    Jerry                        column=grade:, timestamp=1224726404965, value=2

    6 row(s) in 0.0410 seconds

    12,查看scores表中所有数据courses列族的所有数据

    hbase(main):013:0> scan 'scores', ['course:']

    ROW                          COLUMN+CELL

    Tom                         column=course:art, timestamp=1224726394286, value=97

    Tom                         column=course:math, timestamp=1224726377027, value=87

    Jerry                        column=course:art, timestamp=1224726424967, value=80

    Jerry                        column=course:math, timestamp=1224726416145, value=100

    4 row(s) in 0.0200 seconds

            上面就是HBase的基本shell操作的一个例子,可以看出,hbase的shell还是比较简单易用的,从中也可以看出HBase shell缺少很多传统sql中的一些类似于like等相关操作,当然,HBase作为BigTable的一个开源实现,而BigTable是作为 google业务的支持模型,很多sql语句中的一些东西可能还真的不需要.

    1 概述

    HBase是一个分布式的、面向列的数据库,它和一般关系型数据库的最大区别是:HBase很适合于存储非结构化的数据,还有就是它基于列的而不是基于行的模式。

    既然HBase是采用KeyValue的列存储,那Rowkey就是KeyValue的Key了,表示唯一一行。Rowkey也是一段二进制码流,最大长度为64KB,内容可以由使用的用户自定义。数据加载时,一般也是根据Rowkey的二进制序由小到大进行的。

    HBase是根据Rowkey来进行检索的,系统通过找到某个Rowkey (或者某个 Rowkey 范围)所在的Region,然后将查询数据的请求路由到该Region获取数据。HBase的检索支持3种方式:

    (1) 通过单个Rowkey访问,即按照某个Rowkey键值进行get操作,这样获取唯一一条记录;

    (2) 通过Rowkey的range进行scan,即通过设置startRowKey和endRowKey,在这个范围内进行扫描。这样可以按指定的条件获取一批记录;

    (3) 全表扫描,即直接扫描整张表中所有行记录。

    HBASE按单个Rowkey检索的效率是很高的,耗时在1毫秒以下,每秒钟可获取1000~2000条记录,不过非key列的查询很慢。

    Table中Family和Qualifier的关系与区别

    就像用MySQL一样,我们要做的是表设计,MySQL中的表,行,列的在HBase已经有所区别了,在HBase中主要是TableFamilyQualifier,这三个概念。Table可以直接理解为表,而Family和Qualifier其实都可以理解为列,一个Family下面可以有多个Qualifier,所以可以简单的理解为,HBase中的列是二级列,也就是说Family是第一级列,Qualifier是第二级列。两个是父子关系。

    测试发现:

    在实际应用场景中,对于单column qualifier和多column qualifier两种情况,

    如果value长度越长,row key长度越短,字段数(column qualifier数)越少,前者和后者在实际传输数据量上会相差小些;反之则相差较大。

    如果采用多column qualifier的方式存储,且客户端采取批量写入的方式,则可以根据实际情况,适当增大客户端的write buffer大小,以便能够提高客户端的写入吞吐量。

    从性能的角度谈table中family和qualifier的设置
      对于传统关系型数据库中的一张table,在业务转换到hbase上建模时,从性能的角度应该如何设置family和qualifier呢?
      最极端的,①每一列都设置成一个family,②一个表仅有一个family,所有列都是其中的一个qualifier,那么有什么区别呢?


      从读的方面考虑:
      family越多,那么获取每一个cell数据的优势越明显,因为io和网络都减少了。
      如果只有一个family,那么每一次读都会读取当前rowkey的所有数据,网络和io上会有一些损失。
      当然如果要获取的是固定的几列数据,那么把这几列写到一个family中比分别设置family要更好,因为只需一次请求就能拿回所有数据。

      从写的角度考虑:
      首先,内存方面来说,对于一个Region,会为每一个表的每一个Family分配一个Store,而每一个Store,都会分配一个MemStore,所以更多的family会消耗更多的内存。
      其次,从flush和compaction方面说,目前版本的hbase,在flush和compaction都是以region为单位的,也就是说当一个family达到flush条件时,该region的所有family所属的memstore都会flush一次,即使memstore中只有很少的数据也会触发flush而生成小文件。这样就增加了compaction发生的机率,而compaction也是以region为单位的,这样就很容易发生compaction风暴从而降低系统的整体吞吐量。
      第三,从split方面考虑,由于hfile是以family为单位的,因此对于多个family来说,数据被分散到了更多的hfile中,减小了split发生的机率。这是把双刃剑。更少的split会导致该region的体积比较大,由于balance是以region的数目而不是大小为单位来进行的,因此可能会导致balance失效。而从好的方面来说,更少的split会让系统提供更加稳定的在线服务。而坏处我们可以通过在请求的低谷时间进行人工的split和balance来避免掉。
         因此对于写比较多的系统,如果是离线应该,我们尽量只用一个family好了,但如果是在线应用,那还是应该根据应用的情况合理地分配family。

    首先,不同的family是在同一个region下面。而每一个family都会分配一个memstore,所以更多的family会消耗更多的内存。
    其次,目前版本的hbase,在flush和compaction都是以region为单位的,也就是说当一个family达到flush条件时,该region的所有family所属的memstore都会flush一次,即使memstore中只有很少的数据也会触发flush而生成小文件。这样就增加了compaction发生的机率,而compaction也是以region为单位的,这样就很容易发生compaction风暴从而降低系统的整体吞吐量。
    第三,由于hfile是以family为单位的,因此对于多个family来说,数据被分散到了更多的hfile中,减小了split发生的机率。这是把双刃剑。更少的split会导致该region的体积比较大,由于balance是以region的数目而不是大小为单位来进行的,因此可能会导致balance失效。而从好的方面来说,更少的split会让系统提供更加稳定的在线服务。
    上述第三点的好处对于在线应用来说是明显的,而坏处我们可以通过在请求的低谷时间进行人工的split和balance来避免掉。
    因此对于写比较多的系统,如果是离线应该,我们尽量只用一个family好了,但如果是在线应用,那还是应该根据应用的情况合理地分配family。

    2 HBase的RowKey设计

    2.1 设计原则

    2.1.1 Rowkey长度原则

    Rowkey是一个二进制码流,Rowkey的长度被很多开发者建议说设计在10~100个字节,不过建议是越短越好,不要超过16个字节。

    原因如下:

    (1)数据的持久化文件HFile中是按照KeyValue存储的,如果Rowkey过长比如100个字节,1000万列数据光Rowkey就要占用100*1000万=10亿个字节,将近1G数据,这会极大影响HFile的存储效率;

    (2)MemStore将缓存部分数据到内存,如果Rowkey字段过长内存的有效利用率会降低,系统将无法缓存更多的数据,这会降低检索效率。因此Rowkey的字节长度越短越好。

    (3)目前操作系统是都是64位系统,内存8字节对齐。控制在16个字节,8字节的整数倍利用操作系统的最佳特性。

    2.1.2 Rowkey散列原则

    如果Rowkey是按时间戳的方式递增,不要将时间放在二进制码的前面,建议将Rowkey的高位作为散列字段,由程序循环生成,低位放时间字段,这样将提高数据均衡分布在每个Regionserver实现负载均衡的几率。如果没有散列字段,首字段直接是时间信息将产生所有新数据都在一个 RegionServer上堆积的热点现象,这样在做数据检索的时候负载将会集中在个别RegionServer,降低查询效率。

    2.1.3 Rowkey唯一原则

    必须在设计上保证其唯一性。

    2.2 应用场景

    基于Rowkey的上述3个原则,应对不同应用场景有不同的Rowkey设计建议。

    2.2.1 针对事务数据Rowkey设计

    事务数据是带时间属性的,建议将时间信息存入到Rowkey中,这有助于提示查询检索速度。对于事务数据建议缺省就按天为数据建表,这样设计的好处是多方面的。按天分表后,时间信息就可以去掉日期部分只保留小时分钟毫秒,这样4个字节即可搞定。加上散列字段2个字节一共6个字节即可组成唯一 Rowkey。如下图所示:

    事务数据Rowkey设计
    第0字节 第1字节 第2字节 第3字节 第4字节 第5字节
    散列字段 时间字段(毫秒) 扩展字段
    0~65535(0x0000~0xFFFF) 0~86399999(0x00000000~0x05265BFF)  

    这样的设计从操作系统内存管理层面无法节省开销,因为64位操作系统是必须8字节对齐。但是对于持久化存储中Rowkey部分可以节省25%的开销。也许有人要问为什么不将时间字段以主机字节序保存,这样它也可以作为散列字段了。这是因为时间范围内的数据还是尽量保证连续,相同时间范围内的数据查找的概率很大,对查询检索有好的效果,因此使用独立的散列字段效果更好,对于某些应用,我们可以考虑利用散列字段全部或者部分来存储某些数据的字段信息,只要保证相同散列值在同一时间(毫秒)唯一。

    2.2.2 针对统计数据的Rowkey设计

    统计数据也是带时间属性的,统计数据最小单位只会到分钟(到秒预统计就没意义了)。同时对于统计数据我们也缺省采用按天数据分表,这样设计的好处无需多说。按天分表后,时间信息只需要保留小时分钟,那么0~1400只需占用两个字节即可保存时间信息。由于统计数据某些维度数量非常庞大,因此需要4个字节作为序列字段,因此将散列字段同时作为序列字段使用也是6个字节组成唯一Rowkey。如下图所示:

    统计数据Rowkey设计
    第0字节 第1字节 第2字节 第3字节 第4字节 第5字节
    散列字段(序列字段) 时间字段(分钟) 扩展字段
    0x00000000~0xFFFFFFFF) 0~1439(0x0000~0x059F)  

    同样这样的设计从操作系统内存管理层面无法节省开销,因为64位操作系统是必须8字节对齐。但是对于持久化存储中Rowkey部分可以节省25%的开销。预统计数据可能涉及到多次反复的重计算要求,需确保作废的数据能有效删除,同时不能影响散列的均衡效果,因此要特殊处理。

    2.2.3 针对通用数据的Rowkey设计

    通用数据采用自增序列作为唯一主键,用户可以选择按天建分表也可以选择单表模式。这种模式需要确保同时多个入库加载模块运行时散列字段(序列字段)的唯一性。可以考虑给不同的加载模块赋予唯一因子区别。设计结构如下图所示。

    通用数据Rowkey设计
    第0字节 第1字节 第2字节 第3字节
    散列字段(序列字段) 扩展字段(控制在12字节内)
    0x00000000~0xFFFFFFFF) 可由多个用户字段组成
    2.2.4 支持多条件查询的RowKey设计

    HBase按指定的条件获取一批记录时,使用的就是scan方法。 scan方法有以下特点:

    (1)scan可以通过setCaching与setBatch方法提高速度(以空间换时间);

    (2)scan可以通过setStartRow与setEndRow来限定范围。范围越小,性能越高。

    通过巧妙的RowKey设计使我们批量获取记录集合中的元素挨在一起(应该在同一个Region下),可以在遍历结果时获得很好的性能。

    (3)scan可以通过setFilter方法添加过滤器,这也是分页、多条件查询的基础。

    在满足长度、三列、唯一原则后,我们需要考虑如何通过巧妙设计RowKey以利用scan方法的范围功能,使得获取一批记录的查询速度能提高。下例就描述如何将多个列组合成一个RowKey,使用scan的range来达到较快查询速度。

    例子:

    我们在表中存储的是文件信息,每个文件有5个属性:文件id(long,全局唯一)、创建时间(long)、文件名(String)、分类名(String)、所有者(User)。

    我们可以输入的查询条件:文件创建时间区间(比如从20120901到20120914期间创建的文件),文件名(“中国好声音”),分类(“综艺”),所有者(“浙江卫视”)。

    假设当前我们一共有如下文件:

    ID CreateTime Name Category UserID
    1 20120902 中国好声音第1期 综艺 1
    2 20120904 中国好声音第2期 综艺 1
    3 20120906 中国好声音外卡赛 综艺 1
    4 20120908 中国好声音第3期 综艺 1
    5 20120910 中国好声音第4期 综艺 1
    6 20120912 中国好声音选手采访 综艺花絮 2
    7 20120914 中国好声音第5期 综艺 1
    8 20120916 中国好声音录制花絮 综艺花絮 2
    9 20120918 张玮独家专访 花絮 3
    10 20120920 加多宝凉茶广告 综艺广告 4

    这里UserID应该对应另一张User表,暂不列出。我们只需知道UserID的含义:

    1代表 浙江卫视; 2代表 好声音剧组; 3代表 XX微博; 4代表赞助商。调用查询接口的时候将上述5个条件同时输入find(20120901,20121001,”中国好声音”,”综艺”,”浙江卫视”)。此时我们应该得到记录应该有第1、2、3、4、5、7条。第6条由于不属于“浙江卫视”应该不被选中。我们在设计RowKey时可以这样做:采用 UserID + CreateTime + FileID组成RowKey,这样既能满足多条件查询,又能有很快的查询速度。

    需要注意以下几点:

    (1)每条记录的RowKey,每个字段都需要填充到相同长度。假如预期我们最多有10万量级的用户,则userID应该统一填充至6位,如000001,000002…

    (2)结尾添加全局唯一的FileID的用意也是使每个文件对应的记录全局唯一。避免当UserID与CreateTime相同时的两个不同文件记录相互覆盖。

    按照这种RowKey存储上述文件记录,在HBase表中是下面的结构:

    rowKey(userID 6 + time 8 + fileID 6) name category ….

    00000120120902000001

    00000120120904000002

    00000120120906000003

    00000120120908000004

    00000120120910000005

    00000120120914000007

    00000220120912000006

    00000220120916000008

    00000320120918000009

    00000420120920000010

    怎样用这张表?

    在建立一个scan对象后,我们setStartRow(00000120120901),setEndRow(00000120120914)。

    这样,scan时只扫描userID=1的数据,且时间范围限定在这个指定的时间段内,满足了按用户以及按时间范围对结果的筛选。并且由于记录集中存储,性能很好。

    然后使用 SingleColumnValueFilter(org.apache.hadoop.hbase.filter.SingleColumnValueFilter),共4个,分别约束name的上下限,与category的上下限。满足按同时按文件名以及分类名的前缀匹配。

    (注意:使用SingleColumnValueFilter会影响查询性能,在真正处理海量数据时会消耗很大的资源,且需要较长的时间)

    如果需要分页还可以再加一个PageFilter限制返回记录的个数。

    以上,我们完成了高性能的支持多条件查询的HBase表结构设计。

    HBase的rowkey的设计原则

    HBase是三维有序存储的,通过rowkey(行键),column key(column family和qualifier)和TimeStamp(时间戳)这个三个维度可以对HBase中的数据进行快速定位。
    HBase中rowkey可以唯一标识一行记录,在HBase查询的时候,有两种方式:
    1、通过get方式,指定rowkey获取唯一一条记录 
    2、通过scan方式,设置startRow和stopRow参数进行范围匹配 
    3、全表扫描,即直接扫描整张表中所有行记录
    rowkey长度原则:
    rowkey是一个二进制码流,可以是任意字符串,最大长度64kb,实际应用中一般为10-100bytes,以byte[]形式保存,一般设计成定长。建议越短越好,不要超过16个字节,原因如下:
    数据的持久化文件HFile中是按照KeyValue存储的,如果rowkey过长,比如超过100字节,1000w行数据,光rowkey就要占用100*1000w=10亿个字节,将近1G数据,这样会极大影响HFile的存储效率; 
    MemStore将缓存部分数据到内存,如果rowkey字段过长,内存的有效利用率就会降低,系统不能缓存更多的数据,这样会降低检索效率。 
    目前操作系统都是64位系统,内存8字节对齐,控制在16个字节,8字节的整数倍利用了操作系统的最佳特性。
     
    rowkey散列原则:
    如果rowkey按照时间戳的方式递增,不要将时间放在二进制码的前面,建议将rowkey的高位作为散列字段,由程序随机生成,低位放时间字段,这样将提高数据均衡分布在每个RegionServer,以实现负载均衡的几率。如果没有散列字段,首字段直接是时间信息,所有的数据都会集中在一个RegionServer上,这样在数据检索的时候负载会集中在个别的RegionServer上,造成热点问题,会降低查询效率。
     
    rowkey唯一原则:
    必须在设计上保证其唯一性,rowkey是按照字典顺序排序存储的,因此,设计rowkey的时候,要充分利用这个排序的特点,将经常读取的数据存储到一块,将最近可能会被访问的数据放到一块。
     
    什么是热点:
    HBase中的行是按照rowkey的字典顺序排序的,这种设计优化了scan操作,可以将相关的行以及会被一起读取的行存取在临近位置,便于scan。然而糟糕的rowkey设计是热点的源头。热点发生在大量的client直接访问集群的一个或极少数个节点(访问可能是读,写或者其他操作)。大量访问会使热点region所在的单个机器超出自身承受能力,引起性能下降甚至region不可用,这也会影响同一个RegionServer上的其他region,由于主机无法服务其他region的请求。设计良好的数据访问模式以使集群被充分,均衡的利用。
    为了避免写热点,设计rowkey使得不同行在同一个region,但是在更多数据情况下,数据应该被写入集群的多个region,而不是一个。
     
    下面是一些常见的避免热点的方法以及它们的优缺点:
    • 加盐
    这里所说的加盐不是密码学中的加盐,而是在rowkey的前面增加随机数,具体就是给rowkey分配一个随机前缀以使得它和之前的rowkey的开头不同。分配的前缀种类数量应该和你想使用数据分散到不同的region的数量一致。加盐之后的rowkey就会根据随机生成的前缀分散到各个region上,以避免热点。
    • 哈希
    哈希会使同一行永远用一个前缀加盐。哈希也可以使负载分散到整个集群,但是读却是可以预测的。使用确定的哈希可以让客户端重构完整的rowkey,可以使用get操作准确获取某一个行数据
    • 反转

    第三种防止热点的方法时反转固定长度或者数字格式的rowkey。这样可以使得rowkey中经常改变的部分(最没有意义的部分)放在前面。这样可以有效的随机rowkey,但是牺牲了rowkey的有序性。

    反转rowkey的例子 
    以手机号为rowkey,可以将手机号反转后的字符串作为rowkey,这样的就避免了以手机号那样比较固定开头导致热点问题
    • 时间戳反转
    一个常见的数据处理问题是快速获取数据的最近版本,使用反转的时间戳作为rowkey的一部分对这个问题十分有用,可以用Long.Max_Value - timestamp追加到key的末尾,例如[key][reverse_timestamp],[key]的最新值可以通过scan [key]获得[key]的第一条记录,因为HBase中rowkey是有序的,第一条记录是最后录入的数据。
    比如需要保存一个用户的操作记录,按照操作时间倒序排序,在设计rowkey的时候,可以这样设计 
    [userId反转][Long.Max_Value - timestamp],在查询用户的所有操作记录数据的时候,直接指定反转后的userId,startRow是[userId反转][000000000000],stopRow是[userId反转][Long.Max_Value - timestamp] 
    如果需要查询某段时间的操作记录,startRow是[user反转][Long.Max_Value - 起始时间],stopRow是[userId反转][Long.Max_Value - 结束时间]
  • 相关阅读:
    PHP学习笔记:APACHE配置虚拟目录、一个站点使用多域名配置方式
    转载:分页原理+分页代码+分页类制作
    PHP学习笔记:数据库学习心得
    PHP学习笔记:用mysqli连接数据库
    PHP学习笔记:MySQL数据库的操纵
    PHP学习笔记:利用时间和mt_rand函数获取随机名字
    PHP学习笔记:等比例缩放图片
    前端学习(一) html介绍和head标签
    Python 协程
    Python 线程----线程方法,线程事件,线程队列,线程池,GIL锁,协程,Greenlet
  • 原文地址:https://www.cnblogs.com/duanxz/p/4660784.html
Copyright © 2011-2022 走看看