金慶的專欄
C++博客
::
首頁
::
新隨筆
::
聯系
::
聚合
::
管理
::
423 隨筆 :: 0 文章 :: 454 評論 :: 0 Trackbacks
公告
我的隨筆
我的評論
我參與的隨筆
留言簿
(12)
給我留言
查看公開留言
查看私人留言
隨筆分類
(502)
1. C/C++(166)
(rss)
2. 網游開發(fā)(108)
(rss)
3. Golang(20)
(rss)
4. Linux/Unix(30)
(rss)
5. 軟工與管理(44)
(rss)
6. Python(23)
(rss)
7. Erlang(18)
(rss)
8. Rust(16)
(rss)
9. 其它(77)
(rss)
隨筆檔案
(423)
2023年1月 (1)
2022年11月 (1)
2022年10月 (2)
2022年9月 (1)
2022年4月 (6)
2022年1月 (2)
2021年12月 (4)
2021年11月 (6)
2021年10月 (2)
2021年9月 (2)
2021年8月 (7)
2021年7月 (2)
2021年5月 (2)
2021年3月 (1)
2021年2月 (2)
2021年1月 (1)
2020年12月 (1)
2020年10月 (1)
2020年9月 (5)
2020年8月 (1)
2020年7月 (1)
2020年6月 (1)
2020年4月 (2)
2020年3月 (3)
2020年2月 (3)
2020年1月 (1)
2019年12月 (1)
2019年9月 (2)
2019年4月 (2)
2019年1月 (1)
2018年12月 (1)
2018年11月 (3)
2018年10月 (1)
2018年9月 (3)
2018年8月 (3)
2018年7月 (2)
2018年6月 (4)
2018年5月 (4)
2018年4月 (4)
2018年3月 (1)
2018年1月 (2)
2017年12月 (2)
2017年11月 (3)
2017年10月 (3)
2017年8月 (7)
2017年7月 (1)
2017年6月 (1)
2017年5月 (3)
2017年4月 (3)
2017年3月 (3)
2017年2月 (2)
2017年1月 (2)
2016年12月 (5)
2016年11月 (2)
2016年10月 (2)
2016年9月 (1)
2016年8月 (6)
2016年7月 (3)
2016年6月 (2)
2016年5月 (4)
2016年4月 (2)
2016年3月 (2)
2016年1月 (3)
2015年12月 (2)
2015年11月 (2)
2015年10月 (1)
2015年8月 (2)
2015年7月 (1)
2015年6月 (1)
2015年5月 (4)
2015年4月 (3)
2015年3月 (4)
2015年2月 (5)
2015年1月 (4)
2014年12月 (3)
2014年11月 (3)
2014年10月 (2)
2014年9月 (3)
2014年8月 (1)
2014年4月 (4)
2014年3月 (1)
2014年2月 (4)
2014年1月 (5)
2013年12月 (5)
2013年11月 (5)
2013年9月 (2)
2013年8月 (2)
2013年7月 (2)
2013年6月 (2)
2013年5月 (1)
2013年1月 (2)
2012年12月 (1)
2012年11月 (1)
2012年9月 (1)
2012年8月 (3)
2012年7月 (2)
2012年6月 (1)
2012年4月 (3)
2012年3月 (2)
2012年2月 (3)
2012年1月 (2)
2011年11月 (2)
2011年10月 (3)
2011年9月 (2)
2011年8月 (2)
2011年7月 (3)
2011年6月 (2)
2011年5月 (3)
2011年1月 (2)
2010年12月 (1)
2010年11月 (2)
2010年10月 (2)
2010年9月 (3)
2010年8月 (2)
2010年7月 (3)
2010年6月 (1)
2010年5月 (3)
2010年4月 (3)
2010年3月 (5)
2010年2月 (4)
2010年1月 (4)
2009年12月 (2)
2009年11月 (3)
2009年10月 (4)
2009年9月 (3)
2009年8月 (2)
2009年7月 (4)
2009年6月 (1)
2009年5月 (3)
2009年4月 (4)
2009年3月 (2)
2009年2月 (5)
2009年1月 (1)
2008年12月 (7)
2008年11月 (4)
2008年10月 (1)
2008年9月 (3)
2008年8月 (4)
2008年7月 (3)
2008年6月 (4)
2008年5月 (6)
2008年4月 (7)
2008年3月 (6)
2008年1月 (5)
2007年12月 (7)
2007年11月 (4)
2007年10月 (5)
2007年9月 (6)
2007年8月 (8)
2007年7月 (5)
相冊
公告照片
搜索
積分與排名
積分 - 656585
排名 - 25
最新評論
1.?re: boost::asio::spawn 將一統(tǒng)C++網絡庫
asio 成為C++首選網絡庫
--linda
2.?re: log4cxx中文輸出錯誤補丁
評論內容較長,點擊標題查看
--金慶
3.?re: mingw編譯OrzNet
能發(fā)送一個mingw編譯好的OrzNet庫給我嗎? liuweiqcxy@163.com
謝謝!
--劉威
4.?re: log4cxx中文輸出錯誤補丁
評論內容較長,點擊標題查看
--bigbad
5.?re: log4cxx中文輸出錯誤補丁
評論內容較長,點擊標題查看
--bigbad
閱讀排行榜
1.?"multiple definition of" 錯誤(11032)
2.?SVN中邪惡的replace(10955)
3.?VS2005編譯libevent(10413)
4.?混音算法的學習與研究(10203)
5.?C調用lua腳本的效率測試(9007)
評論排行榜
1.?VC6正在被拋棄(35)
2.?VS2005編譯libevent(21)
3.?"multiple definition of" 錯誤(18)
4.?C++引用優(yōu)于指針(17)
5.?ACE與ASIO之間關于Socket編程的比較(16)
MongoDb 用 mapreduce 統(tǒng)計留存率
MongoDb 用 mapreduce 統(tǒng)計留存率
(金慶的專欄)
留存的定義采用的是
新增賬號第X日:某日新增的賬號中,在新增日后第X日有登錄行為記為留存
輸出如下:(類同友盟的留存率顯示)
留存用戶
注冊時間 新增用戶 留存率
1天后 2天后 3天后 4天后 5天后 6天后 7天后 14天后 30天后
2015-09-17 2300 20.7 % 15.6 % 13 % 11.3 % 9.9 %
2015-09-18 2694 21.8 % 14.8 % 11.5 % 10.5 %
2015-09-19 3325 19 % 11.4 % 10.3 %
2015-09-20 3093 16.2 % 11.9 %
2015-09-21 2303 20.5 %
服務器記錄新建帳號到 retention.register 集合,
每日記錄帳號登錄到 retention.login 集合,
每日運行統(tǒng)計腳本,統(tǒng)計前一天的留存率。
以下為 mongoDB 留存率相關的集合,
除了 retention.register 和 retention.login 由服務器代碼寫入,
其他集合都是由統(tǒng)計腳本生成。
retention.register
========================
留存率統(tǒng)計用,新建帳號。
記錄新建帳號的創(chuàng)建日期。
有以下字段:
platform, 平臺名
account_id, 帳號
date, 注冊日期,字符串,格式:“2015-01-01”
例如: {platform: "baidu", account_id: "jinqing", date: "2015-09-20"}
索引 (platform, account_id), (date)
用于統(tǒng)計每日新增帳號數。
retention.login
==================
留存率統(tǒng)計用,帳號登錄記錄。
有以下字段:
date, 登錄日期
platform, 平臺名
account_id, 帳號
register_date, 帳號注冊日期
例如:{date: "2015-09-23", platform: "baidu", account_id: "jinqing", register_date: "2015-09-20"}
索引 (date, platform, account_id).
retention.result
===================
留存率結果。例如:
{date : "2015-09-01", register : 3344, 1 : 91.1, 2 : 82.2, 3 : 73.3, 4 : 64.4, 5 : 55.5, 6 : 46.6, 7 : 37.7, 14 : 14.0, 30 : 3.33}
{date : "2015-09-02", register : 3344, 1 : 91.1, 2 : 82.2, 3 : 73.3, 4 : 64.4, 5 : 55.5, 6 : 46.6, 7 : 37.7, 14 : 14.0, 30 : 3.33}
可用 mongoexport 導出為 csv 表格文件。
例如:
D:\mongodb\bin>mongoexport -h localhost -d mydb -c retention.result -f date,register,1,2,3,4,5,6,7,14,30 --csv -o d:\temp\retention.csv
其中
date: 注冊日期
register: 新注冊個數
1,2,...7,14,30: 第1日,2日,... 7日,14日,30日留存百分率
留存率統(tǒng)計腳本
--------------
linux下用crontab,
windows下用定時任務,
每日凌晨 00:30 運行統(tǒng)計腳本。
允許隔了幾天沒運行,運行時將從上次運行處一直統(tǒng)計到當天。
如果是首次運行,則從 retention.register 集合的最早日期開始統(tǒng)計。
一天運行多次也不會影響結果。
但是不能同時運行多個實例。
需 mongo 客戶端。
可在 mongo 主機上運行。
mongo my.mongo.host retention.js
生成結果在 mydb.retention.result 集合中,可用 mongoexport 導出為 csv 文件。
#
!/
bin
/
sh
# retention.sh
# 每日凌晨定時執(zhí)行,統(tǒng)計留存率。
# 需 mongo 客戶端。
# 以下需更改為實際目錄, 將在該目錄下運行。
cd
/
home
/
jinq
/
retention
/
# 以下地址應該改為 mongod 服務器地址。
MONGODB
=
192.168
.
8.9
mongo ${MONGODB} retention.js
>>
log.txt
echo Mongo export retention result
mongoexport
-
h ${MONGODB}
-
d mydb
-
c retention.result \
--
sort
'
{"value.date" : 1}
'
\
-
f value.date,value.register,value.
1
,value.
2
,value.
3
,value.
4
,value.
5
,value.
6
,value.
7
,value.
14
,value.
30
\
--
type
=
csv
-
o retention_tmp.csv
DATE
=
`date
+%
Y
%
m
%
d`
FILE
=
retention_${DATE}.csv
# csv替換列頭
echo 日期,注冊數,1日,2日,3日,4日,5日,6日,7日,14日,30日
>
${FILE}
tail
-
n
+
2
retention_tmp.csv
>>
${FILE}
echo Done ${FILE}
!
//
留存率統(tǒng)計腳本
//
參考文檔:留存率統(tǒng)計.txt
//
Usage:
//
mongo my.mongo.host retention.js
print(Date());
db
=
db.getSisterDB(
"
mydb
"
);
//
use mydb
var
startDate
=
getStartDate();
var
endDate
=
formatDate(
new
Date());
print(
"
Calculating retention rate of [
"
+
startDate
+
"
,
"
+
endDate
+
"
)
"
);
if
(startDate
<
endDate) {
insertDefaultResult(startDate);
calcRegisterCount(startDate);
calcRetention(startDate);
print(Date());
print(
"
Done.
"
);
}
else
{
print(
"
Do nothing.
"
);
}
//
Internal functions.
//
獲取統(tǒng)計開始日期,之前的已經統(tǒng)計完成,無需重做。
//
返回字符串,格式:"2015-01-01"
//
獲取 retention.result 的最大 date + 1天, 僅須處理該天及以后的數據。
//
如果是初次運行,retention.result 為空,須讀取 retention.register 的最早日期作為開始。
function
getStartDate() {
var
lastResultDate
=
getLastResultDate();
if
(
null
==
lastResultDate) {
return
getFirstRegisterDate();
}
//
加一天
return
getNextDate(lastResultDate);
}
//
獲取最早的 retention.register 日期。
function
getFirstRegisterDate() {
var
cursor
=
db.retention.register.find(
{date : {$gt :
"
2015-09-01
"
}},
//
除去 null
{_id :
0
, date :
1
}
).sort({date :
1
}).limit(
1
);
if
(cursor.hasNext()) {
return
cursor.next().date;
}
return
formatDate(
new
Date());
}
//
獲取 retention.result 中最后的 date 字段。
//
無date字段則返回null。
//
正常返回如:"2015-01-01"
function
getLastResultDate() {
//
_id 為日期串
var
cursor
=
db.retention.result.find(
{}, {_id :
1
}).sort({_id :
-
1
}).limit(
1
);
if
(cursor.hasNext()) {
return
cursor.next()._id;
}
return
null
;
}
function
add0(m) {
return
m
<
10
?
'
0
'
+
m : m;
}
//
Return likes: "2015-01-02"
function
formatDate(date)
{
var
y
=
date.getFullYear();
var
m
=
date.getMonth()
+
1
;
//
1..12
var
d
=
date.getDate();
return
y
+
'
-
'
+
add0(m)
+
'
-
'
+
add0(d);
}
//
"2015-12-31" -> "2016-01-01"
function
getNextDate(dateStr) {
var
dateObj
=
new
Date(dateStr
+
"
00:00:00
"
);
var
nextDayTime
=
dateObj.getTime()
+
24
*
3600
*
1000
;
var
nextDate
=
new
Date(nextDayTime);
return
formatDate(nextDate);
}
assert(getNextDate(
"
2015-12-31
"
)
==
"
2016-01-01
"
);
assert(getNextDate(
"
2015-01-01
"
)
==
"
2015-01-02
"
);
assert(getNextDate(
"
2015-01-31
"
)
==
"
2015-02-01
"
);
//
插入缺省結果。
//
某些天無新注冊,mapreduce就不會生成該條結果,須強制插入。
function
insertDefaultResult(startDateStr) {
var
docs
=
new
Array();
var
endDateStr
=
formatDate(
new
Date());
for
(
var
dateStr
=
startDateStr;
dateStr
<
endDateStr;
dateStr
=
getNextDate(dateStr)) {
docs.push({_id : dateStr, value : {date : dateStr, register :
0
}});
}
//
for
db.retention.result.insert(docs);
}
//
讀取 retention.register 集合,
//
計算每日新注冊量, 記錄于 retention.result.value.register 字段
//
startDate is like: "2015-01-01"
function
calcRegisterCount(startDate) {
var
mapFunction
=
function
() {
var
key
=
this
.date;
var
value
=
{date : key, register :
1
};
emit(key, value);
};
//
mapFunction
var
reduceFunction
=
function
(key, values) {
var
reducedObject
=
{date : key, register :
0
};
values.forEach(
function
(value) {
reducedObject.register
+=
value.register;
}
)
return
reducedObject;
};
//
reduceFunction
var
endDate
=
formatDate(
new
Date());
db.retention.register.mapReduce(mapFunction, reduceFunction,
{
query: {date: {$gte: startDate, $lt: endDate}},
out: {merge:
"
retention.result
"
}
}
);
//
mapReduce()
}
//
function calcRegisterCount()
//
讀取 retention.login 集合,
//
計算留存率,保存于 retention.result 集合。
//
startDate is like: "2015-01-01"
function
calcRetention(startDate) {
var
mapFunction
=
function
() {
var
key
=
this
.register_date;
var
registerDateObj
=
new
Date(
this
.register_date
+
"
00:00:00
"
);
var
loginDateObj
=
new
Date(
this
.date
+
"
00:00:00
"
);
var
days
=
(loginDateObj
-
registerDateObj)
/
(
24
*
3600
*
1000
);
var
value
=
{date : key, register :
0
};
var
field
=
days
+
"
_count
"
;
//
like: 1_count
value[field]
=
1
;
emit(key, value);
};
//
mapFunction
var
reduceFunction
=
function
(key, values) {
var
reducedObject
=
{date : key, register :
0
};
for
(
var
i
=
1
; i
<=
60
; i
++
) {
var
field
=
i
+
"
_count
"
;
reducedObject[field]
=
0
;
}
values.forEach(
function
(value) {
reducedObject.register
+=
value.register;
for
(
var
i
=
1
; i
<=
60
; i
++
) {
var
field
=
i
+
"
_count
"
;
//
like: 1_count
var
count
=
value[field];
if
(
null
!=
count) {
reducedObject[field]
+=
count;
}
//
if
}
//
for
}
//
function
)
//
values.forEach()
return
reducedObject;
};
//
reduceFunction()
var
finalizeFunction
=
function
(key, reducedVal) {
if
(
0
==
reducedVal.register)
return
reducedVal;
for
(
var
i
=
1
; i
<=
60
; i
++
) {
var
field
=
i
+
"
_count
"
;
//
1_count
var
count
=
reducedVal[field];
reducedVal[String(i)]
=
count
*
100
/
reducedVal.register;
}
return
reducedVal;
};
//
finalizeFunction
var
endDate
=
formatDate(
new
Date());
db.retention.login.mapReduce(mapFunction, reduceFunction,
{
query: {date: {$gte: startDate, $lt: endDate}},
out: {reduce:
"
retention.result
"
},
finalize: finalizeFunction,
}
);
//
mapReduce()
}
//
function calcRetention()
參考
-----
用戶留存率_百度百科
http://baike.baidu.com/link?url=28-agScaamT__jLEBdn5VW-a6CHRlf53bDUrVezkeaHd6TMhO0ULm_9JMmcOu541taQjWGe0JypERg2hIwJCAa
游戲玩家的留存率統(tǒng)計實現 - 流子的專欄 - 博客頻道 - CSDN.NET
http://blog.csdn.net/jiangguilong2000/article/details/16119119
在Mongo數據庫里怎么統(tǒng)計留存率呢? - SegmentFault
http://segmentfault.com/q/1010000000652638
posted on 2015-11-10 17:22
金慶
閱讀(639)
評論(0)
編輯
收藏
引用
所屬分類:
2. 網游開發(fā)
只有注冊用戶
登錄
后才能發(fā)表評論。
【推薦】100%開源!大型工業(yè)跨平臺軟件C++源碼提供,建模,組態(tài)!
相關文章:
How are dtLinks created in NavMesh
UE4 Blueprint Multiple Event BeginPlay
第9代游戲主機
Canvas Scaler 的3種模式
幀同步是否允許客戶端指定命令幀號
rpc應答太快造成請求超時
測試 tolua 例子 TestErrorStack
lua變量缺少local造成unity死鎖
C# tolua 之間互傳 byte[]
Unity使用異步grpc
網站導航:
博客園
IT新聞
BlogJava
博問
Chat2DB
管理
Powered by:
C++博客
Copyright © 金慶
久久青青色综合
|
性欧美丰满熟妇XXXX性久久久
|
久久久精品国产sm调教网站
|
人妻无码αv中文字幕久久
|
久久午夜伦鲁片免费无码
|
狠狠色丁香久久综合五月
|
久久有码中文字幕
|
久久久久久国产精品无码超碰
|
国产精品无码久久久久久
|
爱做久久久久久
|
国产精品99久久久精品无码
|
欧美精品一区二区精品久久
|
久久综合偷偷噜噜噜色
|
欧美精品一区二区精品久久
|
2021国产精品午夜久久
|
亚洲午夜精品久久久久久人妖
|
久久久久高潮综合影院
|
日韩美女18网站久久精品
|
性色欲网站人妻丰满中文久久不卡
|
国产精品美女久久久久av爽
|
亚洲国产精品无码久久SM
|
久久久久综合国产欧美一区二区
|
婷婷久久香蕉五月综合加勒比
|
久久精品二区
|
久久亚洲精品成人AV
|
久久久久国产日韩精品网站
|
国产精品久久久久无码av
|
中文字幕乱码人妻无码久久
|
色偷偷88欧美精品久久久
|
国产精品久久波多野结衣
|
亚洲va中文字幕无码久久不卡
|
色婷婷噜噜久久国产精品12p
|
秋霞久久国产精品电影院
|
国产精品九九九久久九九
|
精品久久久久久无码专区不卡
|
青青草原精品99久久精品66
|
国产精品99久久久久久猫咪
|
国产精品久久久久jk制服
|
久久国产色AV免费看
|
av午夜福利一片免费看久久
|
99re这里只有精品热久久
|