大数据基础技能入门指南

cf7944cdf234c66117d3b6e17df898c6.gif

本文介绍了数据工作中数据基础和复杂数据查询两个基础技能。

5792f368cbb64867a0f9200d5d877741.png

背景

当下,不管是业务升级迭代项目,还是体验优化项目,对于数据的需求都越来越大。数据需求主要集中在以下几个方面:

  1. 项目数据看板搭建:特别是一些AB实验的看板,能直观呈现项目的核心数据变化

  2. 数据分析:项目启动前的探索挖掘以及项目后的效果分析
    但是,眼下存在的一个普遍矛盾是:日益增长的数据需求和落后的数据生产力之前的矛盾

俗话说,求人不如求己,掌握基础的数据技能对于技术同学(尤其是开发岗位的同学)并不是一件难事,只是缺少一个合适的入门指南。本文旨在让想学习数据处理的同学能快速入门。

a2b8b40a9ddb44c598ce9653f46af32b.png

基础技能

  数据基础

这一部分先介绍MaxCompute(原odps)平台进行数据处理开发的基础知识。

  • 表基础

表(table)是数据处理的起点和终点,因此能看懂别人的表,会创建自己的表是数据处理技能中最最基础的一环。

表的创建和修改

创建临时表:

-- 临时表命名建议以“tmp_”开头,odps会知道该表是临时表
-- 临时表的生命周期建议按需设置,不要设置太长,避免资源浪费
CREATE TABLE tmp_ut_cart_clk LIFECYCLE 7 AS 
SELECT  user_id
FROM    <用户浏览数据表>

创建正式表:

-- analytics_dw是odps的空间名,后面的是表名
-- 空间名.表名 才能确定唯一的数据表
-- 以下是一个实际案例
CREATE TABLE IF NOT EXISTS analytics_dw.ads_tb_biz_request_opt_1d
(bucket_id                       STRING COMMENT '分桶',os                             STRING COMMENT '系统',uv                             BIGINT COMMENT '分桶用户数',pv                             BIGINT COMMENT '页面访问pv',page_stay_time                 BIGINT COMMRNT '页面停留时间(ms)'...
)
PARTITIONED BY       -- 分区
(ds                              STRING COMMENT '日期'
)
LIFECYCLE 30
;

注意 PARTITIONED BY 这是指定分区字段。分区在odps的数据处理中很重要,合理的选择分区会让数据查询速度快非常多。

简单理解 分区 就是文件目录的概念,目录信息越精确,需要查询的原始数据就越少,查询效率自然越高。

c415d47f549684a764a652529f7fb090.png

表的命名

如果我们计划要做一张需要定期更新并供他人使用的表,那么表的命名必须要符合一定规范。简单提炼下,我们日常数据处理经常遇到的表大概有4种:

类型

命名前缀

说明

维表

dim_xxxx

提供一些维度信息,一般是让其他表关联来补足部分信息

明细表

dwd_xxxx

经过清洗,过滤,字段处理过的表。仅仅是对客观行为的描述。是数据处理分析的基础

轻度汇总表

dws_xxxx

为了方面后续的数据分析,对部分维度进行聚合计算。讲大白话就是对dwd的部分维度进行group by聚合,轻度聚合表会丢失部分不那么重要的信息,能为后续的分析提供便利

应用层表

ads_xxxx

为上层应用提供数据的表。到了这一层,表中的数据基本不具备继续加工处理的能力。这些表中的数据一般都是用来配置报表,或者用来辅助决策的制定

对于开发同学来说,在需要我们自己创建表的情况下一般都是ads类型的表,表的命名可以按照下面的格式:

<空间名>.ads_<业务><二级业务/如有><功能/实验><数据统计周期/1d/7d/30d等等>

  • 基础查询

基础查询是数据处理的基础,这一步的主要工作包括数据的清洗和过滤,字段的加工拓展,为后续的数据处理打好坚实基础

下面是一个非常非常基础的查询sql:

SET odps.sql.mapper.split.size=2048; -- 默认是256(单位M)SELECT  user_id,page ,time_stamp...
FROM    <App用户使用明细表>
WHERE   ds = '${bizdate}'
AND     product = '<App名称>'
AND     event_type = '<事件类型:浏览\点击>'
AND     page = '<页面标识>'
;

如果查询的数据量巨大,那么查询时可能会遇到下面这个错误:

FAILED: ODPS-0130071:[0,0] Semantic analysis exception - physical plan generation failed: java.lang.RuntimeException: com.aliyun.odps.lot.cbo.plan.splitting.disruptor.InstExceedLimitException: task:M1 instance count exceeds limit 99999

这是因为odps是根据 数据存储大小/splitSize 来确定需要的实例数,但有个99999的上限,超限了就会报错。这时适当将splitSize调大一点即可(可以每次*2的幅度来调整)

字符串处理

查询过程中常见的字符串处理方法:

-- 单个条件
SELECT  IF(page = 'Page_XXX', 'y', 'n') AS is_page_xxx -- 多个条件,CASE WHEN hh <= 12 THEN '上午'WHEN hh > 12 AND hh <= 18 THEN '下午'ELSE '晚上'END AS 时间段-- 超级有用:提取args中的kv,KEYVALUE(args, ',', '=', 'itemid') AS item_id-- 分割字符串(value为“a_b_c”这种有规律的字符串可以使用),SPLIT(value, '_') AS value_list  -- 这个是数组,可以使用索引 value_list[0]-- 去除空值(使用a,b,c中第一个不为NULL的值,否则用最后的空字符串),COALESCE(a, b, c, '') AS xxx-- 版本比较,超级实用,IF(bi_udf:bi_yt_compare_version(app_version, '10.24.10') >= 0, 'y', 'n') AS is_target_version-- 解析JSON,提取目标信息,GET_JSON_OBJECT(json_str, '$.section.item.name') AS item_name-- 类型转换,CAST(user_id AS BIGINT) AS user_id-- 大小写转换,TOUPPER(os) , TOLOWER(os)
日期处理

字符串处理中有关于日期时间的处理也比较常见,比如“查询最近7天的数据”,关于日期的常用函数如下:

-- 日期格式描述yyyy    年,4位MM      月,2位dd    日,2位hh/HH    12小时制/24小时制,2位mi    分钟,2位ss    秒,2位SSS    毫秒,3位
-- 通过上面这些格式就能在转化具体的日期时描述日期的格式:
20230807             yyyyMMdd
2023-08-07          yyyy-MM-dd
20230806 13:22:00            yyyyMMdd HH:mi:si-- 单纯查询某个日期之前或者之后的数据
ds >= '20230807'TO_DATE('20230807', 'yyyyMMdd')       -- 将日期字符串转为 datetime 实例,日期处理的基础TO_CHAR(datetime, 'yyyyMMdd')          -- 将日期函数处理得到各种datetime转换为字符串FROM_UNIXTIME(123456789)         -- 将unix时间戳转换成datetime对象      -- 日期加减,自动处理进位关系
DATEADD(TO_DATE('20230807', 'yyyyMMdd'), 7, 'dd')    -- 20230814
DATEADD(TO_DATE('20230807', 'yyyyMMdd'), -7, 'dd')   -- 20230731-- 2个日期间隔(第一个日期-第二个日期,结果可为负)
DATEDIFF(TO_DATE('20230807', 'yyyyMMdd'), TO_DATE('20230806', 'yyyyMMdd'), 'dd') -- 1-- 提取指定时间
-- 在希望分小时段统计的场景下很实用
DATEPART(TO_DATE('2023-08-07 12:13:22', 'yyyy-MM-dd hh:mi:ss'), 'hh') -- 12
  • 关联查询

很多时候单一表的数据无法满足我们的需求,需要通过其他表来补充一些信息,这时就需要关联数据。在sql上表现为有Join操作。

常用的关联操作有 LEFT JOIN、RIGHT JOIN、INNER JOIN

-- 基本的join语法如下
SELECT  a.user_id ,a.arg1,a.args,b.bucket_id
FROM    (SELECT  user_id,arg1,args FROM  <用户手淘行为表>
) a
LEFT JOIN (SELECT  user_d,bucket_id FROM  <AB实验分流表>
) b
ON    a.user_id = b.user_id
;-- left join、right join、inner join差别left join:会保留左表的所有数据(在上面这个例子中左表就是 a,join左边的表),右表中没有匹配的数据将会丢失
right join:和left join相反会保留右表(b)的所有数据,左表中没有匹配的数据会丢失
inner join:最终只有两个表的交集部分会被保留下来

Join操作很容易出错,导致查询结果出错,而且这种错误有时非常隐蔽难以发现。主要原因就是匹配条件遗漏或者关联字段有重复值,出现多对多的情况,导致数据膨胀,进而影响了统计结果。为了避免出现问题,有几个建议:

  1. a、b表关联前先进行必要的数据清洗和去重,而不是先关联后处理

  2. 如果a、b表都是数据量很大的表,建议先随机抽取小样本数据生成临时表a'、b',然后对比最终表数据量和a'、b'的数据量大小是否符合预期

特别地,在小表关联大表的情况下,可以使用MapJoin提升效率,比如在一个每日成交表中有商品的类目信息,现在需要关联到对应的行业信息,而类目和行业的映射关系是一个很小的表,这种情况下就可以使用MapJoin提升任务的执行效率。

SELECT  /* + mapjoin(J2) */J1.*,J2.industry
FROM    <订单表> J1
LEFT JOIN  
(SELECT  cate_level1_id,industryFROM    <行业维表>WHERE   ds = '${bizdate}'
) J2
ON J1.cate_level1_id = J2.cate_level1_id
;
  • 聚合查询

聚合就是针对数据中的某些维度(系统、版本等)执行一系列计算返回单一值。一般在sql上体现为有Group By操作。一般我们数据处理(指标计算)的最后几步都离不开聚合操作。

-- 常见聚合函数
AVG(age) AS avg_age )    -- 平均值SUM(cnt) AS total_cnt    -- 求和MIN(age) AS min_age      -- 最小值MAX(age) AS max_age      -- 最大值COUNT(*) / COUNT(item_id)  -- 计数 count(*)不会忽略null,count(xx)会忽略nullCOUNT(DISTINCT utdid)      -- 去重计数COLLECT_SET(item_id)       -- 将去重后的item_id存在一个数组中COLLECT_ARRAY(item_id)     -- 将item_id存在一个数组中(不去重)PERCENTILE(duration, 0.95)  -- 求分位数

通常,我们在执行聚合时可能会有一些特殊的需求,比如我们想查询每日成交中每个省份的GMV同时还想查询所有省份的整体GMV。正常可能需要这么写:

SELECT  province  ,SUM(amount) AS gmv
FROM  <每日成交表>
GROUP BY provinceUNION ALLSELECT  '整体' AS province  ,SUM(amount) AS gmv
FROM  <每日成交表>

维度少的时候这么写没问题,但考虑下这个需求,我们想看每个省下面的每个城市的gmv,同时也想看这个省整体的gmv,同时也想看所有省份的gmv,这时再用上面的写法就会很繁琐。这时可以考虑使用CUBE或者GROUPING SETS来简化查询逻辑:

SELECT  IF(GROUPING(province) == 0, province, 'all') AS province,IF(GROUPING(city) == 0, city, 'all') AS city,SUM(amount) AS gmv
FROM  <每日成交表>
GROUP BY GROUPING SETS((), (province), (province, city))-- 下面是CUBE的示例SELECT  IF(GROUPING(province) == 0, province, 'all') AS province,IF(GROUPING(city) == 0, city, 'all') AS city,SUM(amount) AS gmv
FROM  <每日成交表>
GROUP BY CUBE(province, city)-- 说明:
-- GROUPING SETS:按照制定维度组合来做聚合
-- CUBE:按照相关维度的全排列来做聚合

最后再强调一句:数据查询时 一定要指定分区 一定要指定分区 一定要指定分区

  复杂数据查询

很多同学其实是具备sql的基本知识的,但是一旦数据查询稍微变复杂一点,就有点束手无策。这个很正常,因为复杂sql的可读性、可维护性本来就很。和开发思路类似解决这个问题的方法就是将复杂的逻辑的拆解为简单的过程,减少查询的套娃。个人推荐的方法主要有:临时表、odps script、cte表达式三种方式。

  • 临时表(临时查询使用)

将复杂过程的查询过程拆解,每个过程的查询结果保存为一张临时表,直至最终完成整个查询逻辑。这个方法在做数据分析时特别好用。

一般我们都是以天为单位来分析数据,可以按照下面的模板来做:

-- 步骤1 甚至可以写注释方面以后理解
-- 建议将关心的原始数据先清洗处理保存为临时表,方便后面做各种分析使用,提升效率
DROP TABLE IF EXISTS tmp_step1_${bizdate};
CREATE TABLE tmp_step1_${bizdate} LIFECYCLE 3 AS  -- 临时表生命周期不要设置太久,避免无意义的资源浪费
SELECT  a,b,c
FROM  <数据表1>
WHERE  <筛选条件>
;-- 步骤2
DROP TABLE IF EXISTS tmp_step2_${bizdate};
CREATE TABLE tmp_step2_${bizdate} LIFECYCLE 3 AS 
SELECT  a,b,c
FROM  tmp_step1_${bizdate} 
;-- ....-- 关注的结果
SELECT  *
FROM  tmp_stepN_${bizdate}
GROUP BY xxx
;

说明:

  1. 临时表查询最好能写成无脑一键执行就能获得最终结果,这能节省大量的时间

  2. ${xxx} odps的参数写法,可以在执行sql前制定对应参数的值,然后替换掉整个sql中的对应参数,本质是字符串替换,一个sql中可以出现多个参数

  3. 为了让sql能反复执行,建表前需要确保相应的表没有被创建过(DROP TABLE)

  • ODPS SCRIPT(有局限性,不推荐)

有cte表达式后,不推荐该方法。简单示例如下:

@step1 :=
SELECT   XX 
FROM   XXXX;@step2 :=
SELECT   YY 
FROM   @step1;
....SELECT  *
FROM  @stepN;
  • CTE表达式(强力推荐,用过都说好)

CTE能让我们将复杂任务拆解,提升SQL的可读性、可维护性。此外CTE不仅可以用于临时查询,也能将任务发布为周期任务。日常的数据处理可以使用下面的模板:

WITH
step1 AS
(SELECT  XXFROM    XXXX
),step2 AS 
(SELECT  YYFROM    step1
),....stepN AS 
(SELECT  ...
)                   -- 最后的这括号后面不要加 , INSERT OVERWRITE TABLE <存储表名> PARTITION (ds = '${bizdate}')  -- 一个WITH只支持一个INSERT
SELECT  *
FROM  stepN

15e10fbd1f34929c495441cf5fa806f4.png

写在最后

数据处理并不是神秘、难以掌握的技能。每个技术同学、产品同学都是可以学会基本的数据处理技能的。希望本文能帮助有需要的同学叩开数据处理的大门。

99c8488eb210d0495d4d6153060d15a6.png

团队介绍

我们是大淘宝技术「基础交易终端团队」,主要负责电商核心交易链路业务和平台的研发,包含:淘宝购物车、下单、订单、物流、逆向等电商核心基础能力及创新型业务。这里有世界一流的技术产品,有丰富的业务场景,服务于十亿级的消费者,这里有巨大的挑战等你来。作为阿里的一支明星团队,负责阿里电商平台的核心交易主链路,是阿里移动技术的基石,每年双十一核心链路保障。

现招聘移动端(Android/iOS)开发工程师,有前端开发经验者优先,有意者可以投递简历到:guzhan.pc@taobao.com

¤ 拓展阅读 ¤

3DXR技术 | 终端技术 | 音视频技术

服务端技术 | 技术质量 | 数据算法

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MIT 6.S081 Operating System/Fall 2020 macOS搭建risc-v与xv6开发调试环境

文章目录 本机配置安装环境Homebrew执行安装脚本查看安装是否成功 RISC-V tools执行brew的安装脚本 QEMUXV6 测试有用的参考链接&#xff08;感谢前辈&#xff09;写在结尾 本机配置 电脑型号&#xff1a;Apple M2 Pro 2023 操作系统&#xff1a;macOS Ventura 13.4 所以我的电…

谷歌浏览器在新的浏览器窗口中打开所选的每条搜索结果在哪设置? 谷歌的搜索设置在哪设置??

1、进入谷歌搜索页面&#xff0c; 2、点击快捷设置&#xff0c;点击高级搜索 3、点击自定义搜索设置 4、点击其他设置 5、启用在新窗口中打开搜索结果

如何保护我们的网络安全

保护网络安全是至关重要的&#xff0c;尤其是在今天的数字化时代。以下是一些保护网络安全的基本步骤&#xff1a; 1、使用强密码&#xff1a;使用包含字母、数字和特殊字符的复杂密码。不要在多个网站上重复使用相同的密码。定期更改密码。 2、启用双因素认证 (2FA)&#xff…

数据分析基础:数据可视化+数据分析报告

数据分析是指通过对大量数据进行收集、整理、处理和分析&#xff0c;以发现其中的模式、趋势和关联&#xff0c;并从中提取有价值的信息和知识。 数据可视化和数据分析报告是数据分析过程中非常重要的两个环节&#xff0c;它们帮助将数据转化为易于理解和传达的形式&#xff0…

CRM系统:快速实现外勤出差人员远程访问企业提升工作效率!

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏:《速学数据结构》 《C语言进阶篇》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 文章目录 快速实现外勤出差人员远程访问企业CRM系统前言1. 无需公网IP&#xff0c;高效低成本实现CRM系统远程访问1.1 下…

单片机外围电路设计常出现的问题

网友&#xff1a;单片机外围电路有哪些注意事项呀&#xff1f; 工程师&#xff1a;单片机外围电路设计需要注意的点有很多&#xff0c;今天我简单说几个吧 一、单片机上拉电阻的选择&#xff0c;复位电路中电阻R110k时RST是高电平&#xff0c;而当R150时RST为低电平&#xff0…

月老脱单交友盲盒小程序源码系统 带完整搭建教程

大家好啊&#xff0c;知道大家想要什么就给大家分享什么。今天要给大家分享的是月老脱单交友盲盒小程序源码系统&#xff0c;在当下可是十分火热&#xff0c;搭建起来也比较简单&#xff0c;一起来看看吧。 系统特色功能一览&#xff1a; 实现随机生成虚拟商品盒子&#xff0c…

并发、并行、同步、异步、阻塞、非阻塞

一、多核、多cpu &#xff08;一&#xff09;多核 Multicore 核是CPU最重要的部分。负责运算。核包括控制单元、运算单元、寄存器等单元。 多核就是指单个CPU中有多个核。 &#xff08;二&#xff09;多cpu Multiprocessor 多cpu就是一个系统拥有多个CPU。每个CPU可能有单个核…

AI:38-基于深度学习的抽烟行为检测

随着人工智能的迅猛发展,它在各个领域展现出了广泛的应用潜力。其中,基于深度学习的抽烟行为检测技术引起了人们的极大兴趣。这项技术利用计算机视觉和深度学习算法,能够自动检测和监测人们的抽烟行为,为烟草控制和公共卫生提供了新的手段。本文将深入探讨基于深度学习的抽…

SELECT COUNT(*)会不会导致全表扫描引起慢查询

SELECT COUNT(*)会不会导致全表扫描引起慢查询呢&#xff1f; SELECT COUNT(*) FROM SomeTable 网上有一种说法&#xff0c;针对无 where_clause 的 COUNT(*)&#xff0c;MySQL 是有优化的&#xff0c;优化器会选择成本最小的辅助索引查询计数&#xff0c;其实反而性能最高&…

【C++】:内存管理

朋友们、伙计们&#xff0c;我们又见面了&#xff0c;本期来给大家解读一下有关Linux的基础知识点&#xff0c;如果看完之后对你有一定的启发&#xff0c;那么请留下你的三连&#xff0c;祝大家心想事成&#xff01; C 语 言 专 栏&#xff1a;C语言&#xff1a;从入门到精通 数…

1.MySQL库的操作

个人主页&#xff1a;Lei宝啊 愿所有美好如期而遇 1.创建数据库&#xff1a; 语法&#xff1a;create database if not exists db_name charset字符集 collate校验规则 ; if not exists &#xff0c;charset &#xff0c;collate可以不加&#xff0c;直接create database 数…

关于如何进行ChatGPT模型微调的新手指南

微调是指在预训练的模型基础上&#xff0c;通过进一步的训练来调整模型以适应特定任务或领域。预训练的模型在大规模的文本数据上进行了广泛的学习&#xff0c;从中获得了一定的知识和语言理解能力。然而&#xff0c;由于预训练并不针对具体任务&#xff0c;因此需要微调来使模…

【TensorFlow2 之011】TF 如何使用数据增强提高模型性能?

一、说明 亮点&#xff1a;在这篇文章中&#xff0c;我们将展示数据增强技术作为提高模型性能的一种方式的好处。当我们没有足够的数据可供使用时&#xff0c;这种方法将非常有益。 教程概述&#xff1a; 无需数据增强的训练什么是数据增强&#xff1f;使用数据增强进行训练可视…

隐马尔可夫模型(一)Evaluation

前提 import torch import torch.nn.functional as F N 3 # 离散隐变量可以取到的值的个数 M 2 # 可观测变量个数 pi F.softmax(torch.randn((N,1),dtypetorch.float32),dim0) # 初始状态概率矩阵 A F.softmax(torch.randn((N,N),dtypetorch.float32),dim-1) # 转移…

Linux8yum安装mysql5.7版本流程

Linux8yum安装mysql Linux8yum安装报错解决 yum安装流程 首先下载mysql的yum配置 wget http://dev.mysql.com/get/mysql57-community-release-el7-11.noarch.rpm安装mysql源 yum -y install mysql57-community-release-el7-11.noarch.rpm安装mysql yum -y install mysql-s…

【目标检测】大图包括标签切分,并转换成txt格式

前言 遥感图像比较大&#xff0c;通常需要切分成小块再进行训练&#xff0c;之前写过一篇关于大图裁切和拼接的文章【目标检测】图像裁剪/标签可视化/图像拼接处理脚本&#xff0c;不过当时的工作流是先将大图切分成小图&#xff0c;再在小图上进行标注&#xff0c;于是就不考…

Elasticsearch:使用 Langchain 和 OpenAI 进行问答

这款交互式 jupyter notebook 使用 Langchain 将虚构的工作场所文档拆分为段落 (chunks)&#xff0c;并使用 OpenAI 将这些段落转换为嵌入并将其存储到 Elasticsearch 中。然后&#xff0c;当我们提出问题时&#xff0c;我们从向量存储中检索相关段落&#xff0c;并使用 langch…

Google SGE 正在添加人工智能图像生成器,现已推出:从搜索中的生成式 AI 中获取灵感的新方法

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

1688拍立淘接口,按图搜索1688商品接口,图片搜索商品接口,1688API接口

按图搜索1688商品的方法如下&#xff1a; 打开1688平台&#xff0c;点击首页右上角的搜索框&#xff0c;进入搜索页面。 点击搜索框右侧的相机图标&#xff0c;选择“拍照”或“相册”&#xff0c;上传你想要搜索的图片。 等待图片上传完成&#xff0c;系统会自动识别图片中的…