- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我已阅读 Transitioning to ARC Release Notes在“摘要”部分。他们告诉:
ARC works by adding code at compile time to ensure that objects live as long as necessary, but no longer. Conceptually, it follows the same memory management conventions as manual reference counting (described in Advanced Memory Management Programming Guide) by adding the appropriate memory management calls for you.
In order for the compiler to generate correct code
我想知道 ARC 更正我们的代码的结果是什么。
我的问题:我们能看到变化吗? (在 alloc 、 retain 、 assign 或 release 的术语中。不是汇编级别!)
原因:因为我认为在没有 ARC 模式的情况下看到老传统开发中的最佳实践代码是件好事。
最佳答案
通过将代码从 ObjC 重写为 ObjC,clang 中的 ARC 无法工作,但会在代码生成期间发出额外的保留/释放 LLVM 位码。这意味着,如果不进入 LLVM IR/汇编级别,就不可能知道编译器如何“修复”它。
If ARC emit the LLVM bitcode as you said. Is it made for the purpose that use less time in compile process? (less complex ObjC code, less header file?)
如果编译器能够减少通过代码的次数,那总是更好。
Can you show me some example or utility that show the code in assembly level?
要获取汇编代码,您可以
直接从编译器生成程序集。在命令行中,添加 -S
调用编译器时标记。结果是 .S
包含汇编代码的文件。在 Xcode 项目中,打开源代码文件,然后转到产品(在菜单栏上)→ 生成输出 → 程序集文件。程序集文件。程序集文件。 p>
生成目标文件,然后反汇编。内置命令 otool -tvV <file>
可以进行反汇编,还有像otx这样的高级工具(免费)或 IDA (免费评估)。
我更喜欢路线 2,因为它产生的垃圾较少,而且可以配置反汇编工具以产生更多有用的信息。不管怎样,无论使用哪种方法,您都需要能够阅读汇编代码。
以这段代码为例:
- (BOOL)application:(UIApplication*)application
didFinishLaunchingWithOptions:(NSDictionary*)launchOptions
{
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
self.window.backgroundColor = [UIColor whiteColor];
[self.window makeKeyAndVisible];
return YES;
}
编译后会生成如下程序集(使用IDA分析):
-[SomeAppDelegate application:didFinishLaunchingWithOptions:]: push {r4-r7,lr} add r7, sp, #0xC str.w r8, [sp,-#0x4]! sub sp, sp, #0x18 movw r1, #(0x343c - 0x2574) ; @selector(alloc) mov r8, r0 movt.w r1, #0 mov r0, (0x3464 - 0x2576) ; _OBJC_CLASS_$_UIWindow add r1, pc add r0, pc ldr r1, [r1] ldr r0, [r0] blx _objc_msgSend mov r1, (0x3440 - 0x258e) ; @selector(mainScreen) mov r6, r0 movw r0, #(0x3468 - 0x2594) ; _OBJC_CLASS_$_UIScreen add r1, pc movt.w r0, #0 add r0, pc ldr r1, [r1] ldr r0, [r0] blx _objc_msgSend mov r7, r7 blx _objc_retainAutoreleasedReturnValue mov r5, r0 cbz r5, L25ba movw r0, #(0x3444 - 0x25b2) ; @selector(bounds) mov r1, r5 movt.w r0, #0 add r0, pc ldr r2, [r0] add r0, sp, #0x8 blx _objc_msgSend_stret b L25c4L25ba: add r0, sp, #0x8 vmov.i32 q8, #0x80 vstmia r0, {d16-d17}L25c4: mov r1, (0x3448 - 0x25d2) ; @selector(initWithFrame:) ldr r0, [sp,#0x10] add r1, pc ldr r2, [sp,#0x8] ldr r3, [sp,#0xc] ldr r4, [sp,#0x14] stmea.w sp, {r0,r4} mov r0, r6 ldr r1, [r1] blx _objc_msgSend mov r4, r0 mov r0, (0x344c - 0x25F2) ; @selector(setWindow:) mov r2, r4 add r0, pc ldr r1, [r0] mov r0, r8 blx _objc_msgSend mov r0, r4 blx _objc_release mov r0, r5 blx _objc_release mov r0, (0x3450 - 0x2610) ; @selector(window) add r0, pc ldr r5, [r0] mov r0, r8 mov r1, r5 blx _objc_msgSend mov r7, r7 blx _objc_retainAutoreleasedReturnValue mov r1, (0x3454 - 0x2630) ; @selector(whiteColor) mov r6, r0 movw r0, #(0x346C - 0x2636) ; _OBJC_CLASS_$_UIColor add r1, pc movt.w r0, #0 add r0, pc ldr r1, [r1] ldr r0, [r0] blx _objc_msgSend mov r7, r7 blx _objc_retainAutoreleasedReturnValue mov r4, r0 mov r0, (0x3458 - 0x2652) ; @selector(setBackgroundColor:) mov r2, r4 add r0, pc ldr r1, [r0] mov r0, r6 blx _objc_msgSend mov r0, r4 blx _objc_release mov r0, r6 blx _objc_release mov r0, r8 mov r1, r5 blx _objc_msgSend mov r7, r7 blx _objc_retainAutoreleasedReturnValue mov r4, r0 mov r0, (0x345C - 0x2680) ; @selector(makeKeyAndVisible) add r0, pc ldr r1, [r0] mov r0, r4 blx _objc_msgSend mov r0, r4 blx _objc_release movs r0, #1 add sp, sp, #0x18 ldr.w r8, [sp], #4 pop {r4-r7,pc}
Without going into detail, you can see there are many _objc_release
and _objc_retainAutoreleasedReturnValue
. These are what ARC inserts during code-gen. Decompiling it by hand, we'll get:
UIScreen* r5 = objc_retainAutoreleasedReturnValue([UIScreen mainScreen]);
CGRect sp8 = r5 != nil ? [r5 bounds] : CGRectZero;
UIWindow* r4 = [[UIWindow alloc] initWithFrame:sp8];
[self setWindow:r4];
objc_release(r4);
objc_release(r5);
UIWindow* r6a = objc_retainAutoreleasedReturnValue([self window])
UIColor* r4a = objc_retainAutoreleasedReturnValue([UIColor whiteColor])
[r6a setBackgroundColor:r4a];
objc_release(r4a);
objc_release(r6a);
UIWindow* r4b = objc_retainAutoreleasedReturnValue([self window])
[r4b makeKeyAndVisible];
objc_release(r4b);
return 1;
这与 @c roald 相同的链接描述。
关于objective-c - 是否可以在编译时看到ARC生成的代码?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10429857/
我遵循了一本名为“Sitepoint Full Stack Javascript with MEAN”的书中的教程,我刚刚完成了第 6 章,应该已经创建了一个带有“数据库”的“服务器”。数据库只不过是
在 Jquery 中,我创建两个数组,一个嵌入另一个数组,就像这样...... arrayOne = [{name:'a',value:1}, {name:'b',value:2}] var arra
这个问题在这里已经有了答案: What is the explanation for these bizarre JavaScript behaviours mentioned in the 'Wa
我被放在别人的代码上,有一个类用作其他组件的基础。当我尝试 ng serve --aot(或 build --prod)时,我得到以下信息。 @Component({ ...,
我正在测试一些代码,并使用数据创建了一个 json 文件。 问题是我在警报中收到“[object Object],[object Object]”。没有数据。 我做错了什么? 这是代码:
我想打印 [object Object],[object Object] 以明智地 "[[{ 'x': '1', 'y': '0' }, { 'x': '2', 'y': '1' }]]"; 在 ja
我有一个功能 View ,我正在尝试以特殊格式的方式输出。但我无法让列表功能正常工作。 我得到的唯一返回是[object Object][object Object] [object Object]
在使用优秀的 Sim.js 和 Three.js 库处理 WebGL 项目时,我偶然发现了下一个问题: 一路走来,它使用了 THREE.Ray 的下一个构造函数: var ray = new THRE
我正在使用 Material UI 进行多重选择。这是我的代码。 {listStates.map(col => (
我的代码使用ajax: $("#keyword").keyup(function() { var keyword = $("#keyword").val(); if (keyword.
我遇到了下一个错误,无法理解如何解决它。 Can't resolve all parameters for AuthenticationService: ([object Object], ?, [o
我正在尝试创建一个显示动态复选框的表单,至少应选中其中一个才能继续。我还需要获取一组选中的复选框。 这是组件的代码: import { Component, OnInit } from '@angul
我正在开发 NodeJs 应用程序,它是博客应用程序。我使用了快速验证器,我尝试在 UI 端使用快速闪存消息将帖子保存在数据库中之前使用闪存消息验证数据,我成功地将数据保存在数据库中,但在提交表单后消
我知道有些人问了同样的问题并得到了解答。我已经查看了所有这些,但仍然无法解决我的问题。我有一个 jquery snipet,它将值发送到处理程序,处理程序处理来自 JS 的值并将数据作为 JSON 数
我继承了一个非常草率的项目,我的任务是解释为什么它不好。我注意到他们在整个代码中都进行了这样的比较 (IQueryable).FirstOrDefault(x => x.Facility == fac
我只是在删除数组中的对象时偶然发现了这一点。 代码如下: friends = []; friends.push( { a: 'Nexus', b: 'Muffi
这两个代码片段有什么区别: object = nil; [object release] 对比 [object release]; object = nil; 哪个是最佳实践? 最佳答案 object
我应该为其他人将从中继承的第一个父对象传递哪个参数,哪个参数更有效 Object.create(Object.prototype) Object.create(Object) Object.creat
我在不同的对象上安排不同的选择器 [self performSelector:@selector(doSmth) withObject:objectA afterDelay:1]; [self per
NSLog(@"%p", &object); 和 NSLog(@"%p", object); 有什么区别? 两者似乎都打印出一个内存地址,但我不确定哪个是对象的实际内存地址。 最佳答案 这就是我喜欢的
我是一名优秀的程序员,十分优秀!