前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >通过SQL解读财富的分配

通过SQL解读财富的分配

作者头像
jeanron100
发布2018-03-21 17:12:09
6520
发布2018-03-21 17:12:09
举报

还是继续昨天的问题,知乎上一个蛮有意思的问题,数据分析师做了一个很有意思的解读。

题目是:房间里有100个人,每人都有100元钱,他们在玩一个游戏。每轮游戏中,每个人都要拿出一元钱随机给另一个人,最后这100个人的财富分布是怎样的?

我们用SQL来解读一下,当然还可以测试的更充分一些。

首先我们创建一个表test_money,里面是存放很简单的数据,pid就是人员编号,money就是手头的财富。

create table test_money(pid number primary key,money number);

先插入100条记录。 insert into test_money select level,1 from dual connect by level <=100;

接着我使用如下的脚本来批量模拟实验,即循环pid,从1到100,依次扣减一块,依次随机添加一块。

delete from test_money; insert into test_money select level,1 from dual connect by level <=100; commit; begin for i in 1..100 loop update test_money set money=money-1 where pid=i; update test_money set money=money+1 where pid=trunc(dbms_random.value(0,100)) ; end loop; end; /

运行之后查看数据,第一次是这样的效果,可以看到,0,1,2的数据占比差不多是95% MONEY COUNT(*) ---------- ---------- 1 37 2 25 4 1 3 3 0 34 再来依次测试,数据有一定的变化,但是占比差别不大。 MONEY COUNT(*) ---------- ---------- 1 42 2 19 4 2 3 4 0 33

我们能不能继续改进一下,这么模拟测试100次,最后得到的平均数就相对更有说服力了。

为此我们创建一个表test_money_sum,插入统计数据。

create table test_money_sum(money int,money_cnt number); insert into test_money_sum select money,count(*)from test_money group by money;

按照这样的逻辑,我可以很轻松的模拟100遍这个实验。

使用如下的shell脚本来批量更新数据,得到最后的结果。

for i in {1..100} do sqlplus -s xx/xx<<EOF @test.sql EOF done sqlplus -s xx/xx<<EOF select money,sum(money_cnt)/100 from test_money_sum group by money; EOF

到了出数据的时候了,100次测试之后,财富的分布情况如下:

金钱

平均人数

0

37.36

1

36.57

2

18.12

3

6.16

4

1.43

5

0.28

6

0.07

8

0.01

如果用图表来查看会更好一些

而所占的比例用这个饼图来看更加直白。

整体来看,这个实验还是很有意蕴的,想要突破现有的思维桎梏,达到一个新的高度,这个比例还是很低的,但是无论如何还是需要花很大的功夫的。

我们再想一下,如果这个样本是1000万呢。

这就涉及几个技巧,首先是初始化数据,1000万的数据初始化可以这么来做。

SQL> insert into test_money select rownum,1 from xmltable('1 to 10000000'); 10000000 rows created. Elapsed: 00:00:29.50

半分钟即可搞定,而如何批量更新呢,可以参考这个pl/sql

declare cursor test_cur is select pid from test_money; type rec_type is table of test_cur%rowtype index by binary_integer; recs rec_type; begin open test_cur; fetch test_cur bulk collect into recs; close test_cur; forall i in 1..recs.COUNT update test_money set money=money-1 where pid=recs(i).pid; update test_money set money=money+1 where pid=trunc(dbms_random.value(0,10000000)) ; commit; end; /

数据还在运行。稍后提供。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2017-07-29,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 杨建荣的学习笔记 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
数据库
云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档