适用于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
答案 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将自动检测元素并允许您更改它们。无需重新提交工程或应用商店。
让我再给你一些关于它的高级见解:
要了解有关Leanplum Visual Interface Editor的更多信息,请查看leanplum.com。他们提供30天的免费试用。
(免责声明:我是Leanplum的工程师)