pig实战 pig常用语法总结,教你快速入门——算法篇

  本文主要来源于google资料的自我总结,对常用的pig语法即“算法”进行了总结

基础概念:
  relation bag tuple field data
  关系(relation) = 包(bag)
  一个包是一个元组(tuple)的集合,在pig中用{}扩起来表示一个包
  一个元组(tuple)是若干个字段的有序集合(order set),在pig的数据结构中,用()扩起来标识一个元组
  一个字段(field)是列数据(data)的标识;

  和数据库的对应关系:
    pig database
    relation/bag table
    tuple one record
    field field(one volume)

  但是pig中tuple的字段数是随意的,这点和数据库不同


运行模式以及注释:
  1、 运行模式
    a) 本地
      i. pig –x local
    b) 集群
      i. pig –x mapreduce
      ii. 或者pig
    c) 批处理命令
      i. 将命令批量写入xx.pig
      ii. 用本地或者集群模式运行,如pig xx.pig

  2、注释
    a) 段落注释:/**/
    b) 行注释: --


基本用法框架:
  a) 输入
    i. A = Load ‘输入路径’ USING PigStorage(“\t”) AS (name:chararray, age:int, gpa:float);
    ii. 逻辑:用pig自带的PigStorage读入输入路径的数据,对每一行用”\t”分割,并用字符串类型的name、int类型的age、float类型的gpq作为每行数据的字段名
  b) 中间处理
    i. B = FOREACH A GENERATE name;
  c) 输出
    i. DUMP B; --输出到控制台
    ii. 或者STORE B INTO ‘输出路径’ USING PigStorage();

基本语法实例:
    1、计算多维度组合下的平均值 的实际例子
      需求:求文件中第2、3、4组合的第4、5列的平均值
      脚本解释:
        A = Load '1.txt' USING PigStorage(' ') AS(col1:chararray, col2:int, col3:int, col4:int, col5:double, col6:double);
        将1.txt中的每行数据用" "分割,然后安装(col1:chararray, col2:int, col3:int, col4:int, col5:double, col6:double)解析数据,并装载到每个bag/tuple/filed
        col1是每列的别名,如果不指定,在后面可以用$0,$n来索引,但是可读性较差

        A的结构是{col1:chararray, col2:int, col3:int, col4:int, col5:double, col6:double)}

        B = GROUP A BY (col2, col3, col4);
        对A用col2/col3/col4的组合进行分组,然后按组将每条tuple汇集成一个bag,
        B的数据结构是B:{group:(col2,col3,col4),A:bag{:tuple,tuple}}

        C = FOREACH B GENERATE group, AVG(col5), AVG(col6);
        FOREACH是遍历每个组中汇集的tuple,并对其用组合函数处理字段
        C的结构是C:{group:(col2,col3,col4),double,double}

        DUMP C;//单机运行打印调试信息
        STORE C INTO “output”;//存储结果

    2、统计行数
       A = LOAD '1.txt' USING PigStorage (' ‘) AS(col1:chararray, col2:int, col3:int, col4:int, col5:double, col6:double);
       B = GROUP A all;
       C = FOREACH B GENERATE COUNT(col2);
       DUMP C;
       注意:如果col2这一列有NULL,则这一行将不会被统计。
        GROUP A all;没有by关键字。
        If you want to include NULL values in the count computation, use COUNT_STAR.

    3、FLATTEN
      同1中的例子,但是C不同
      C = FOREACH B GENERATE FLATTEN(group), AVG(col5), AVG(col6);
      1中的例子得到的结果结构是(col2, col3, col4),avg1,avg2
      但是对group作用FLATTEN后结构变为col2,col3,col4,avg1,avg2;可以看出FLATTEN的作用是解嵌套(un-nest)包或者元组tuple,

    4、GROUP的两点说明:
      a)用于GROUP的key如果多于一个字段(正如本文前面的例子),则GROUP之后的数据的key是一个元组(tuple),否则它就是与用于GROUP的key相同类型的东西
      b)GROUP后得到的结果是一个包或者是relation,其包含的每一个tuple中包含两个字段,第一个字段被命名为'group',第二个字段是一个包,是含有group对应值的所有tuple的set

    5、把数据作为tuple来加载
      语法:A = LOAD '1.txt' AS (T: tuple(col1:chararray, col2:int, col3:int,col4:int, col5:double, col6:double);
      用describe A;可以查看A的结构。
      注意:输入的数据必须是(xx,xx,xx...xx)的结构才能以元组的形式被加载,适用于中间结果的进一步处理

    6、计算多维组合中不重复记录的条数
      C = FOREACH B GENERATE (D = DISTINCT col5; group, COUNT(D));
      说明:distinct后跟一个字段即可,如果这个字段不同,则肯定不重复,而如果重复,此字段肯定相同,而且重命名D要用等号,后面跟;

    7、将relation转为表量scalar

    8、在pig中使用shell脚本进行辅助处理
      A = LOAD ‘2.txt' AS (col1:int, col2:int, col3:int, col4:chararray, col5:int);
      B = STREAM A THROUGH `awk '{if($4 == "=") print $1"\t"$2"\t"$3"\t"999"\t"$5; else print $0}'`;
      DUMP B;
      作用:将第4列中的"="换成999

    9、向pig脚本传参:
      语法:pig -param output_dir="/xxx" xxx.pig;
      而在xxx.pig中STORE D INTO ‘$output_dir’;

    10、COGROUP的作用:
      对两个关系中的包,分别按制定的字段进行分组
      A = LOAD '3.txt' AS (acol1:chararray, acol2:int, acol3:int);
      B = LOAD '4.txt' AS (bcol1:int, bcol2:chararray, bcol3:int);
      C = COGROUP A BY A.acol1, B BY B.bcol2;
      DUMP C;
      结果的结构{key,{A's one tuple or null},{B's ont tuple or null}};
      按A的第一行和B的第二行的值,分别对A、B进行分组组合,如果A中的tuple含有此key则显示在结果中,如果没有则为{},B也一样

    11、piggybank
      自定义函数的名称为piggybank

    12、UDF的构造函数会被调用多次,所以在其中做一次工作时要特别小心

    13、LOAD多个目录下的数据
      LOAD '/data/201{1,2}'
      load/data/2011 /data/2012两个目录下的数据

    14、GROUP 函数不会忽略NULL值,而COUNT会

    15、统计某些字段组合的种数:
      A = LOAD '15.txt' AS (col1:int, col2:int);
      B = GROUP A BY (col1, col2);
      C = GROUP B ALL;
      D = FOREACH C GENERATE COUNT(B);
      DUMP D;
      思路:先根据需求对数据进行分组,然后对分组COUNT;
      注意:
        a)D中COUNT的是B,因为通过C将所有的数据组为一个新的关系,只包含一个tuple,第一个字段为ALL,第二个字段是B的所有tuple组成的一个bag。
        b)COUNT作用的对象必须是一个bag,所以在统计字段前,要用GROUP X ALL,将X中所有的tuple组成一个bag

    16、两个整数相除,如何得到一个float
      A = LOAD '16.txt' AS (col1:int, col2:int);
      B = FOREACH A GENERATE (float)col1/col2;
      DUMP B;
      注意先转型在计算,而不是(float)(col1/col2);

    17、UNION两个数据进行统计
      A = LOAD '1.txt' AS (col1:int, col2:int);
      B = LOAD '2.txt' AS (col3:int, col4:int);
      C = UNION A, B;
      D = GROUP C BY $0;
      E = FOREACH D GENERATE group, SUM(C.$1);
      #E = FOREACH D GENERATE FLATTEN(group), SUM(C.$1);
      DUMP E;
      注意:
        UNION操作是给关系添加新的tuple,而且UNION后注意观测filed的引用方法,见结构篇
        多个关系时,对列的操作要指定关系名

    18、正则表达式过滤
      过滤出符合*//*.qq.com/*的url
      A = LOAD '18.txt' AS (col1:int, col2:chararray);
      B = FILTER A BY col2 matches '.*//.*\\.qq\\.com/.*';
      说明:.*标识至少一个字符
      而匹配.字符要进行转义\.,而在''内转义要用两个\\. ;

    19、截取字符串:
      SUBSTRING(col1, 0, 4):0为startIndex,4为stopIndex,不包含stopIndex;

      A = LOAD '19.txt' AS (dataStr:chararray, col2:chararray);
      B = FOREACH A GENERATE SUBSTRING(dataStr,0,4);
      C = DISTINCT B;
      DUMP C;

    20、连接字符串:
      A = LOAD ‘20.txt' AS (col1:chararray, col2:int);
      B = FOEACH A GENERATE CONCAT(col1, (chararray)col2);
      DUMP B;
      注意:连接的字段类型必须是chararray,如果不是要转型
        嵌套使用:CONCAT(A,CONCAT(B,C);

    21、用JOIN求两个数据集的交集&不同数据类型交集会失败
      A = LOAD '211.txt' AS (a:int);
      B = LOAD '212.txt' AS (b:int);
      C = JOIN A BY a, B BY b;
      D = GROUP C ALL;
      E = FOREACH D GENERATE COUNT(C);
      DUMP E;
      JOIN后C的结构为:C:{A::a:int, B::b:int}

      去重:
        A = LOAD '211.txt' AS (a:int);
        B = LOAD '212.txt' AS (b:int);
        C = JOIN A BY a, B BY b;
        uniq_C = DISTINCT(C);
        D = GROUP uniq_C ALL;
        E = FOREACH D GENERATE COUNT(C);
        DUMP E;

    22、三目运算符使用必须使用():用来替换空值
      B = FOREACH A GENERATE ((col1 is null) ? -1 : col1)

    23、如何在得到计算结果后,补齐空白
      A = LOAD ‘23.txt' AS (col1:int, b:tuple(col2:int, col3:int);
      B = FOREACH A GENERATE col1, FLATTEN(b);
      C = GROUP B BY B.col1;
      D = FOREACH C GENERATE group, SUM(B.col1) AS sum;
      E = FOREACH D GENERATE group, ((sum is null) ? 0 : sum);
      DUMP E;

    24、DISTINCT操作用于去重,正因为它要把数据集合到一起,才知道哪些数据是重复的,因此,它会产生reduce过程。同时,在map阶段,它也会利用combiner来先去除一部分重复数据以加快处理速度。

    25、提高Pig job的优先级:set job.priority HIGH;提高Pig job的运行速度

    26、“Scalars can be only used with projections”错误和嵌套/inner FOREACH
      在第一列的每种组合中,第二列为3/6的数据分别有多少条

      A = LOAD ’26.txt' AS (col1:chararray, col2:int);
      B = GROUP A BY col1;
      C = FOREACH B {
        D = FILTER A BY col2 == 3;
        E = FILTER A BY col2 == 6;
      GENERATE group, COUNT(D), COUNT(E);};
      DUMP C;

    27、在grunt模式下按Ctrl+A 和 Ctrl+E 代替 HOME 和 END,就可以跳到行首和行末了

    28、同一个关系进行JOIN连接必须导入两次,做连接,否则出错

    29、外链接JOIN
      LEFT:左边的数据全量显示

      A = LOAD '291.txt' AS (col1:int, col2:chararray);
      B = LOAD '292.txt' AS (col1:int, col2:chararray);
      C = JOIN A BY col1 LEFT, B BY col1;
      DESCRIBE C;
      DUMP C;
      这个和数据库的左右连接和内链接一致

    30、pig中支持过滤中文,但是在交互模式下不行

    31、统计 tuple 中的 field 数,bag 中的 tuple 数,map 中的 key/value 组数用SIZE函数

    32、字符此为null 用col is null来判断,但是不能过滤" "," " 过滤要用SIZE(xx)>0
      FILTER A BY (col1 is not null AND (SIZE(col2) > 0));

    33、Pig中的各operator(操作符),哪些会触发reduce过程
      GROUP:由于GROUP操作会将所有具有相同key的记录收集到一起,所以数据如果正在map中处理的话,就会触发shuffle→reduce的过程。
      ORDER:由于需要将所有相等的记录收集到一起(才能排序),所以ORDER会触发reduce过程。同时,除了你写的那个Pig job之外,Pig还会添加一个额外的M-R job到你的数据流程中,因为Pig需要对你的数据集做采样,以确定数据的分布情况,从而解决数据分布严重不均的情况下job效率过于低下的问题。
      DISTINCT:由于需要将记录收集到一起,才能确定它们是不是重复的,因此DISTINCT会触发reduce过程。当然,DISTINCT也会利用combiner在map阶段就把重复的记录移除。
      JOIN:JOIN用于求重合,由于求重合的时候,需要将具有相同key的记录收集到一起,因此,JOIN会触发reduce过程。
      LIMIT:由于需要将记录收集到一起,才能统计出它返回的条数,因此,LIMIT会触发reduce过程。
      COGROUP:与GROUP类似(参看本文前面的部分),因此它会触发reduce过程。
      CROSS:计算两个或多个关系的叉积。

    34、如何统计一个字符串中包含的指定字符数
      shell 脚本 awk -F ":" '{print NF - 1}'
      以:分割字符串,然后打印总的volume数

posted @ 2013-02-19 17:31  成金之路  阅读(7276)  评论(0编辑  收藏  举报