iOS A/B Test介绍和使用

iOS A/B Test介绍和使用

一、什么是A/B Test

现在 App Store 中的应用,就像商场中的商品一样琳琅满目,可以解决用户各个方面的需求。这时,你要想创新,或者做出比竞品更优秀的功能,是越来越不容易。所以,很多公司都必须去做一些实验,看看有哪些功能可以增强自己 App 的竞争力,又有哪些功能可以废弃掉。而进行这样的实验的主要方法,就是 A/B 测试。

A/B 测试,也叫桶测试或分流测试,指的是针对一个变量的两个版本 A 和 B,来测试用户的不同反应,从而判断出哪个版本更有效,类似统计学领域使用的双样本假设测试。简单地说,A/B 测试就是检查 App 的不同用户在使用不同版本的功能时,哪个版本的用户反馈最好。

比如,引导用户加入会员的按钮,要设置为什么颜色更能吸引他们加入,这时候我们就需要进行 A/B 测试。产品接触的多了,我们自然清楚一个按钮的颜色,会影响到用户点击它,并进入会员介绍页面的概率。

二、App 开发中的 A/B Test

从 App 开发层面看,新版本发布频繁,基本上是每月或者每半月会发布一个版本。那么,新版本发布后,我们还需要观察界面调整后情况如何,性能问题修复后线上情况如何,新加功能使用情况如何等。这时,我们就需要进行 A/B 测试来帮助我们分析这些情况,通过度量每个版本的测试数据,来确定下一个版本应该如何迭代。

对于 App 版本迭代的情况简单说就是,新版本总会在旧版本的基础上做修改。这里,我们可以把旧版本理解为 A/B 测试里的 A 版本,把新版本理解为 B 版本。在 A/B 测试中 A 版本和 B 版本会同时存在,B 版本一开始是将小部分用户放到 B 测试桶里,逐步扩大用户范围,通过分析 A 版本和 B 版本的数据,看哪个版本更接近期望的目标,最终确定用哪个版本。总的来说,A/B 测试就是以数据驱动的可回退的灰度方案,客观、安全、风险小,是一种成熟的试错机制。

一个 A/B Test 框架主要包括三部分:

  • 策略服务,为策略制定者提供策略;
  • A/B 测试 SDK,集成在客户端内,用来处理上层业务去走不同的策略;
  • 日志系统,负责反馈策略结果供分析人员分析不同策略执行的结果

其中,策略服务包含了决策流程、策略维度。A/B 测试 SDK 将用户放在不同测试桶里,测试桶可以按照系统信息、地址位置、发布渠道等来划分。日志系统和策略服务,主要是用作服务端处理的,这里我就不再展开了。

下图是 A/B 测试方案的结构图:

A:B 测试方案的结构图.png

三、A/B Test 技术方案

1、整体设计

整体设计.png

2、服务端

服务端1.png

服务端2.png

3、APP端

APP端ABTest处理流程.png

4、大数据端

大数据端需要处理统计的数据有:

  • 符合灰度规则的用户数量
  • 符合灰度规则并启动了APP的用户数量
  • 符合灰度规则并启动了APP且在设置页面点击了【灰度验证】的用户数量
  • 进入测试页面A的用户数
  • 进入测试页面B的用户数

四、A/B Test iOS SDK

这里简单的写了一下逻辑,仅供参考,具体实现可能要复杂一些。开发们根据自己的需求再完善吧。

ZJHABTestManager.h 文件

#import <Foundation/Foundation.h>

/// 业务类型
typedef NS_ENUM(NSInteger, ZJHGrayUserKeyType) {
    ZJHGrayUserKeyTypeOne,   // 业务类型1
    ZJHGrayUserKeyTypeTwo,   // 业务类型2
    ZJHGrayUserKeyTypeThree  // 业务类型3
};

@interface ZJHABTestManager : NSObject

/// 初始化配置
+ (void)configGrayKeys;

/// 取值Bool
+ (BOOL)boolValueWithKeyType:(ZJHGrayUserKeyType)keyType;

/// 取值ABC
+ (NSInteger)abcValueWithKeyType:(ZJHGrayUserKeyType)keyType;

/// 取值Str
+ (NSString *)strValueWithKeyType:(ZJHGrayUserKeyType)keyType;

@end

ZJHABTestManager.m 文件

#import "ZJHABTestManager.h"

@interface ZJHABTestManager ()

@property (nonatomic, strong) NSDictionary *typeKeyDict;
@property (nonatomic, strong) NSDictionary *dataDict;

@end

@implementation ZJHABTestManager

/// 单例
+ (ZJHABTestManager *)shareInstance {
    static ZJHABTestManager *grayUserManager;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        if (!grayUserManager) {
            grayUserManager = [[ZJHABTestManager alloc] init];
        }
    });
    return grayUserManager;
}

/// 初始化配置
+ (void)configGrayKeys {
    ZJHABTestManager *mgr = [ZJHABTestManager shareInstance];
    
    // 配置
    mgr.typeKeyDict = @{ @(ZJHGrayUserKeyTypeOne) : @"ZJHGrayUserKeyTypeOne",
                         @(ZJHGrayUserKeyTypeTwo) : @"ZJHGrayUserKeyTypeTwo",
                         @(ZJHGrayUserKeyTypeThree) : @"ZJHGrayUserKeyTypeThree" };
    
    // 取缓存数据
    mgr.dataDict =
    [[NSUserDefaults standardUserDefaults] objectForKey:@"ZJHABTestDataKey"];
    
    // 请求网络数据
    [self requestUrlDataWithKeyArr:mgr.typeKeyDict.allValues completion:^(NSDictionary *dict) {
        // 数据赋值
        mgr.dataDict = dict;
        // 存数据
        [[NSUserDefaults standardUserDefaults] setObject:dict
                                                  forKey:@"ZJHABTestDataKey"];
    }];
}

/// 请求网络数据
+ (void)requestUrlDataWithKeyArr:(NSArray *)keyArr
                      completion:(void (^)(NSDictionary *dict))completion {
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        if (completion) {
            NSMutableDictionary *mutDict = [NSMutableDictionary dictionary];
            for (NSString *key in keyArr) {
                ;
                mutDict[key] = [NSString stringWithFormat:@"%ld",random() % 3];
            }
            completion(mutDict);
        }
    });
}

/// 取值Bool
+ (BOOL)boolValueWithKeyType:(ZJHGrayUserKeyType)keyType {
    NSString *str = [self strValueWithKeyType:keyType];
    return [str boolValue];
}

/// 取值ABC
+ (NSInteger)abcValueWithKeyType:(ZJHGrayUserKeyType)keyType {
    NSString *str = [self strValueWithKeyType:keyType];
    return [str integerValue];
}

/// 取值Str
+ (NSString *)strValueWithKeyType:(ZJHGrayUserKeyType)keyType {
    ZJHABTestManager *mgr = [ZJHABTestManager shareInstance];
    NSString *typeStr = mgr.typeKeyDict[@(keyType)];
    return mgr.dataDict[typeStr];
}

@end



参考链接
什么是 A/B 测试?:https://www.zhihu.com/question/20045543
A/B 测试:验证决策效果的利器:https://time.geekbang.org/column/article/93097
iOS A/B Test 方案探索:https://www.jianshu.com/p/ba7ba95a524e

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,053评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,527评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,779评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,685评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,699评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,609评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,989评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,654评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,890评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,634评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,716评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,394评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,976评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,950评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,191评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,849评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,458评论 2 342

推荐阅读更多精彩内容

  • 引子 公元2016年末,2017年初,某做旅行产品的互联网公司内,产品经理疯狂的提 A/BTest 需求,以至于该...
    李剑飞的简书阅读 8,625评论 7 51
  • 用到的组件 1、通过CocoaPods安装 2、第三方类库安装 3、第三方服务 友盟社会化分享组件 友盟用户反馈 ...
    SunnyLeong阅读 14,598评论 1 180
  • 黑色的海岛上悬着一轮又大又圆的明月,毫不嫌弃地把温柔的月色照在这寸草不生的小岛上。一个少年白衣白发,悠闲自如地倚坐...
    小水Vivian阅读 3,093评论 1 5
  • 渐变的面目拼图要我怎么拼? 我是疲乏了还是投降了? 不是不允许自己坠落, 我没有滴水不进的保护膜。 就是害怕变得面...
    闷热当乘凉阅读 4,233评论 0 13
  • 感觉自己有点神经衰弱,总是觉得手机响了;屋外有人走过;每次妈妈不声不响的进房间突然跟我说话,我都会被吓得半死!一整...
    章鱼的拥抱阅读 2,168评论 4 5