A / B测试平台如何动态取代Objective-C资产?

时间:2014-02-18 10:28:01

标签: ios objective-c ab-testing leanplum

适用于iOS的Leanplum,Apptimize和其他A / B测试平台能够从Web下载资产(nib文件,图像等)并在运行时替换它们。

天真的方法是下载新资产并将其替换为资源包目录,但由于权限,无法将文件写入资源目录。

问题是,这些A / B测试平台在运行时用什么技术来替换资产?

修改

在读取leanplum静态库文件(使用nm)上的符号后,似乎它们是Swizzling可可文件系统API。

例如:(来自nm -m leanplum.a的示例行)

 -[NSBundle(LeanplumExtension) leanplum_URLForResource:withExtension:]

通过使用otool我可以打印实现:

-[NSBundle(LeanplumExtension) leanplum_URLForResource:withExtension:]:
0000000000000069        pushq   %rbp
000000000000006a        movq    %rsp, %rbp
000000000000006d        pushq   %r15
000000000000006f        pushq   %r14
0000000000000071        pushq   %r13
0000000000000073        pushq   %r12
0000000000000075        pushq   %rbx
0000000000000076        subq    $0x18, %rsp
000000000000007a        movq    %rcx, %rbx
000000000000007d        movq    %rdi, 0xffffffffffffffc8(%rbp)
0000000000000081        movq    %rdx, %rdi
0000000000000084        callq   _objc_retain
0000000000000089        movq    %rax, %r14
000000000000008c        movq    %rbx, %rdi
000000000000008f        callq   _objc_retain
0000000000000094        movq    %rax, 0xffffffffffffffd0(%rbp)
0000000000000098        movq    _originalMainBundle(%rip), %rcx
000000000000009f        movq    "+[NSBundle(LeanplumExtension) leanplum_mainBundle]"(%rcx), %rdi
00000000000000a2        movq    0x4487(%rip), %rsi
00000000000000a9        movq    _objc_msgSend(%rip), %r12
00000000000000b0        movq    %r14, %rdx
00000000000000b3        movq    %rax, %rcx
00000000000000b6        callq   *%r12
00000000000000b9        movq    %rax, %rdi
00000000000000bc        callq   _objc_retainAutoreleasedReturnValue
00000000000000c1        movq    %rax, %r13
00000000000000c4        movq    _skippedFiles(%rip), %rax
00000000000000cb        movq    "+[NSBundle(LeanplumExtension) leanplum_mainBundle]"(%rax), %rbx
00000000000000ce        movq    0x4463(%rip), %rsi
00000000000000d5        movq    %r13, %rdi
00000000000000d8        callq   *%r12
00000000000000db        movq    %rax, %rdi
00000000000000de        callq   _objc_retainAutoreleasedReturnValue
00000000000000e3        movq    %rax, %r15
00000000000000e6        movq    0x4453(%rip), %rsi
00000000000000ed        movq    %rbx, %rdi
00000000000000f0        movq    %r15, %rdx
00000000000000f3        callq   *%r12
00000000000000f6        movb    %al, %bl
00000000000000f8        movq    %r15, %rdi
00000000000000fb        callq   _objc_release
0000000000000100        testb   %bl, %bl
0000000000000102        je      0x115
0000000000000104        movq    %r13, %rdi
0000000000000107        callq   _objc_retain
000000000000010c        movq    %rax, %r15
000000000000010f        movq    0xffffffffffffffd0(%rbp), %rbx
0000000000000113        jmp     0x13b
0000000000000115        movq    0x4414(%rip), %rsi
000000000000011c        movq    0xffffffffffffffc8(%rbp), %rdi
0000000000000120        movq    %r14, %rdx
0000000000000123        movq    0xffffffffffffffd0(%rbp), %rbx
0000000000000127        movq    %rbx, %rcx
000000000000012a        callq   *_objc_msgSend(%rip)
0000000000000130        movq    %rax, %rdi
0000000000000133        callq   _objc_retainAutoreleasedReturnValue
0000000000000138        movq    %rax, %r15
000000000000013b        movq    %r13, %rdi
000000000000013e        callq   _objc_release
0000000000000143        movq    %rbx, %rdi
0000000000000146        callq   _objc_release
000000000000014b        movq    %r14, %rdi
000000000000014e        callq   _objc_release
0000000000000153        movq    %r15, %rdi
0000000000000156        addq    $0x18, %rsp
000000000000015a        popq    %rbx
000000000000015b        popq    %r12
000000000000015d        popq    %r13
000000000000015f        popq    %r14
0000000000000161        popq    %r15
0000000000000163        popq    %rbp
0000000000000164        jmpq    _objc_autoreleaseReturnValue
  1. 有人可以验证我的发现吗?
  2. 我想知道他们如何获得这整个API列表?
  3. 如果我用fopen或其他C lib打开图像会怎样?
  4. 如何解读otool的输出?

2 个答案:

答案 0 :(得分:6)

  

问题是,这些A / B测试平台在运行时用什么技术来替换资产?

有根据的猜测是,他们使用method swizzling将标准方法(例如[NSBundle URLForResource:withExtension:])的实现与他们自己的那些方法版本(例如[NSBundle(LeanplumExtension) leanplum_URLForResource:withExtension:])交换。这意味着您的代码使用与您原本相同的方法,但是您会得到不同的行为 - 在这种情况下,返回的URL取决于A / B测试框架决定向用户呈现的资源版本。 / p>

  

我想知道他们如何获得这整个API列表?

仔细工作。用于加载资源的方法数量并不难以管理。

  

如果我用fopen或其他C lib打开图像会怎样?

我希望框架在这种情况下无法交换资源。有很多方法可以加载数据,框架不可能预测所有这些。但是如果你正在使用A / B框架,那么你可能希望将资源替换为适合你的测试,所以尝试打败框架并没有多大意义。

  

如何解读工具的输出?

对于您已展示的案例,请学习阅读汇编语言。 (otool有很多选择,并且它们都不会产生汇编。)

答案 1 :(得分:3)

例如,Leanplum公司为iOS和Android提供了一个可视化界面编辑器:这不需要编码,Leanplum将自动检测元素并允许您更改它们。无需重新提交工程或应用商店。

让我再给你一些关于它的高级见解:

  1. 通过在应用中安装iOS或Android SDK,可以启用名为Visual Editor的功能。在开发模式下并且打开网站仪表板时,SDK会将有关视图层次结构的信息实时发送到您的浏览器。视图层次结构的扫描方式与在常规网站上构建DOM的方式类似。
  2. 您可以在应用中选择任何UI元素,并实时更改其外观。这可以通过识别视图树中的确切元素并将更改发送到SDK来实现。
  3. 这可以通过添加自定义挂钩或称为“混合”的技术来实现。看看这个blog post,它是如何运作的。
  4. 要了解有关Leanplum Visual Interface Editor的更多信息,请查看leanplum.com。他们提供30天的免费试用。

    (免责声明:我是Leanplum的工程师)