金慶的專欄
C++博客
::
首頁
::
新隨筆
::
聯(lián)系
::
聚合
::
管理
::
423 隨筆 :: 0 文章 :: 454 評論 :: 0 Trackbacks
公告
我的隨筆
我的評論
我參與的隨筆
留言簿
(12)
給我留言
查看公開留言
查看私人留言
隨筆分類
(502)
1. C/C++(166)
(rss)
2. 網(wǎng)游開發(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)
相冊
公告照片
搜索
積分與排名
積分 - 659027
排名 - 25
最新評論
1.?re: boost::asio::spawn 將一統(tǒng)C++網(wǎng)絡(luò)庫
asio 成為C++首選網(wǎng)絡(luò)庫
--linda
2.?re: log4cxx中文輸出錯誤補丁
評論內(nèi)容較長,點擊標(biāo)題查看
--金慶
3.?re: mingw編譯OrzNet
能發(fā)送一個mingw編譯好的OrzNet庫給我嗎? liuweiqcxy@163.com
謝謝!
--劉威
4.?re: log4cxx中文輸出錯誤補丁
評論內(nèi)容較長,點擊標(biāo)題查看
--bigbad
5.?re: log4cxx中文輸出錯誤補丁
評論內(nèi)容較長,點擊標(biāo)題查看
--bigbad
閱讀排行榜
1.?"multiple definition of" 錯誤(11054)
2.?SVN中邪惡的replace(10972)
3.?VS2005編譯libevent(10429)
4.?混音算法的學(xué)習(xí)與研究(10230)
5.?C調(diào)用lua腳本的效率測試(9019)
評論排行榜
1.?VC6正在被拋棄(35)
2.?VS2005編譯libevent(21)
3.?"multiple definition of" 錯誤(18)
4.?C++引用優(yōu)于指針(17)
5.?ACE與ASIO之間關(guān)于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 %
服務(wù)器記錄新建帳號到 retention.register 集合,
每日記錄帳號登錄到 retention.login 集合,
每日運行統(tǒng)計腳本,統(tǒng)計前一天的留存率。
以下為 mongoDB 留存率相關(guān)的集合,
除了 retention.register 和 retention.login 由服務(wù)器代碼寫入,
其他集合都是由統(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)計每日新增帳號數(shù)。
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
===================
留存率結(jié)果。例如:
{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 導(dǎo)出為 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: 新注冊個數(shù)
1,2,...7,14,30: 第1日,2日,... 7日,14日,30日留存百分率
留存率統(tǒng)計腳本
--------------
linux下用crontab,
windows下用定時任務(wù),
每日凌晨 00:30 運行統(tǒng)計腳本。
允許隔了幾天沒運行,運行時將從上次運行處一直統(tǒng)計到當(dāng)天。
如果是首次運行,則從 retention.register 集合的最早日期開始統(tǒng)計。
一天運行多次也不會影響結(jié)果。
但是不能同時運行多個實例。
需 mongo 客戶端。
可在 mongo 主機(jī)上運行。
mongo my.mongo.host retention.js
生成結(jié)果在 mydb.retention.result 集合中,可用 mongoexport 導(dǎo)出為 csv 文件。
#
!/
bin
/
sh
# retention.sh
# 每日凌晨定時執(zhí)行,統(tǒng)計留存率。
# 需 mongo 客戶端。
# 以下需更改為實際目錄, 將在該目錄下運行。
cd
/
home
/
jinq
/
retention
/
# 以下地址應(yīng)該改為 mongod 服務(wù)器地址。
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 日期,注冊數(shù),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)計開始日期,之前的已經(jīng)統(tǒng)計完成,無需重做。
//
返回字符串,格式:"2015-01-01"
//
獲取 retention.result 的最大 date + 1天, 僅須處理該天及以后的數(shù)據(jù)。
//
如果是初次運行,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
"
);
//
插入缺省結(jié)果。
//
某些天無新注冊,mapreduce就不會生成該條結(jié)果,須強(qiáng)制插入。
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)計實現(xiàn) - 流子的專欄 - 博客頻道 - CSDN.NET
http://blog.csdn.net/jiangguilong2000/article/details/16119119
在Mongo數(shù)據(jù)庫里怎么統(tǒng)計留存率呢? - SegmentFault
http://segmentfault.com/q/1010000000652638
posted on 2015-11-10 17:22
金慶
閱讀(643)
評論(0)
編輯
收藏
引用
所屬分類:
2. 網(wǎng)游開發(fā)
只有注冊用戶
登錄
后才能發(fā)表評論。
【推薦】100%開源!大型工業(yè)跨平臺軟件C++源碼提供,建模,組態(tài)!
相關(guān)文章:
How are dtLinks created in NavMesh
UE4 Blueprint Multiple Event BeginPlay
第9代游戲主機(jī)
Canvas Scaler 的3種模式
幀同步是否允許客戶端指定命令幀號
rpc應(yīng)答太快造成請求超時
測試 tolua 例子 TestErrorStack
lua變量缺少local造成unity死鎖
C# tolua 之間互傳 byte[]
Unity使用異步grpc
網(wǎng)站導(dǎo)航:
博客園
IT新聞
BlogJava
博問
Chat2DB
管理
Powered by:
C++博客
Copyright © 金慶
77777亚洲午夜久久多人
|
国产亚洲精午夜久久久久久
|
久久这里都是精品
|
久久婷婷人人澡人人
|
久久国产成人午夜AV影院
|
久久播电影网
|
伊人 久久 精品
|
亚洲国产精品久久久天堂
|
久久精品国产亚洲AV电影
|
狠狠色婷婷久久一区二区三区
|
99久久99这里只有免费费精品
|
国产成人久久激情91
|
国产A级毛片久久久精品毛片
|
久久嫩草影院免费看夜色
|
狠狠色丁香久久婷婷综合蜜芽五月
|
久久本道综合久久伊人
|
亚洲国产日韩欧美久久
|
无遮挡粉嫩小泬久久久久久久
|
丁香狠狠色婷婷久久综合
|
久久青青国产
|
久久久久AV综合网成人
|
青青青国产精品国产精品久久久久
|
成人国内精品久久久久影院
|
伊人丁香狠狠色综合久久
|
日韩亚洲国产综合久久久
|
久久精品亚洲精品国产色婷
|
亚洲午夜久久影院
|
久久精品无码一区二区WWW
|
日本三级久久网
|
久久久亚洲欧洲日产国码是AV
|
国产精品久久久久aaaa
|
色婷婷噜噜久久国产精品12p
|
精品人妻久久久久久888
|
久久亚洲2019中文字幕
|
精品久久久久久亚洲精品
|
午夜精品久久久久久影视777
|
www.久久热.com
|
久久99国产综合精品免费
|
久久精品视频一
|
久久国产精品国语对白
|
狠狠久久亚洲欧美专区
|