用同样的检索条件从context检索出的对象是一个????所以 在主页的3个brand没法释放,在仅仅处理brand的时候????
和 多个 context无关
我重写了NSManagedObject 子类的 的 dealloc 方法,让它在释放时,输出log,这样就能检测它的释放情况了。
先看一个简单的情况,局部变量里用 insertNewObjectForEntityForName 方法插入一个对象,不保存context。代码如下:
TestFile *file = [NSEntityDescription insertNewObjectForEntityForName:@"TestFile" inManagedObjectContext:[[DBManager sharedManager] managedObjectContext]];
没有log输出,说明没有释放。 之后保存context,释放的log输出了。我又测试了函数完成后2秒后再进行save context,结果log输出就变成了2秒后。
之后我又把这个变量写成全局变量,代码如下:
file = [NSEntityDescription insertNewObjectForEntityForName:@"TestFile" inManagedObjectContext:[[DBManager sharedManager] managedObjectContext]];
这次立即保存context也没有输出log。
从上面这2个测试可以看出,NSManagedObject 也是遵守 oc 的释放规则的:有强引用就不释放。有些特殊的地方是:当建立一个局部变量时,如果这个变量还没被保存,那么context会自动进行强引用,防止数据丢失,这样就出现了 即使是局部变量,对象的生命周期也没有在函数结束后就结束!
测完insert,再测测查询。
先看简单的查询:
HubFile *file = [HubFileDAO retrieveHubFilesByFileID:[NSNumber numberWithInt:]];
函数结束后,立即输出了释放log。换成全局变量后,没有log输出了。
再看update,代码如下:
HubFile *file = [HubFileDAO retrieveHubFilesByFileID:[NSNumber numberWithInt:]]; file.shownName = @"aaaaaa";
函数结束后,没有log输出。就像insert时的情况一样了,context 必须保存后才能释放对象。
由此进一步总结一下,对于一个NSManagedObject对象,如果它有任何新内容没有被context保存,那么对应的context就会对它进行强引用,直到我们对context 进行保存。
再看下面代码
同样的检索条件检索出的对象的地址是一致的!不要把他们当做2个对象!有特殊情况存在,见下面的测试!
再看以下代码:
HubFile *file1 = [HubFileDAO retrieveHubFilesByFileID:[NSNumber numberWithInt:]]; [[DBManager sharedManager].managedObjectContext reset]; NSLog(@"file name is %@",file1.shownName);
输出是 file name is (null)。
由于掉用了reset,NSManagedObject 的数据部分变成了 fault(可以理解为无效的)。虽然这时的file1指针仍然指向NSManagedObject对象,但是对所有data属性都变成了nil。
再看以下代码:
HubFile *file1 = [HubFileDAO retrieveHubFilesByFileID:[NSNumber numberWithInt:]]; [[DBManager sharedManager].managedObjectContext reset]; HubFile *file2 = [HubFileDAO retrieveHubFilesByFileID:[NSNumber numberWithInt:]]; NSLog(@"file1 name is %@",file1.shownName);
NSLog(@"file2 name is %@",file2.shownName);
输出是
file1 name is (null)
file2 name is aaaa
看看NSLog(@"file2 name is %@",file2.shownName);调用后的内存情况:
虽然检索条件相同,但是由于在中间掉用了reset函数,file1和file2指向了不同的地址。可以看出,在context reset后,所有之前查询到的指针都失去了作用,虽然它们的地址不是空,但是无法对任何数据属性进行访问了。再次查询也不会在原来的地址上进行内存分配,而是重新分配了一块内存。简单地说,调用 reset 后 ,就不该使用以前的旧对象引用了!
再看由于使用 relationship 产生的 retain circle
HubBrand *oldBrand = [HubBrandDAO retrieveHubBrandByModelID:@""];
oldBrand.shownName = @"aaaa";
[[DBManager sharedManager] saveContext];
brand对象可以释放。
HubBrand *oldBrand = [HubBrandDAO retrieveHubBrandByModelID:@""];
oldBrand.shownName = @"aaaa";
oldBrand.iconFile.shownName = @"tttttt";
[[DBManager sharedManager] saveContext];
brand 对象无法释放,这里iconFile是一个一对一的relationship,关联一个File对象。
这里有一个解释:http://*.com/questions/1789974/core-data-and-retain-cycles
When you have relationships between managed objects, each object maintains a strong reference to the object or objects to which it is related. In a managed memory environment, this causes retain cycles (see Object Ownership and Disposal) that can prevent deallocation of unwanted objects. To ensure that retain cycles are broken, when you're finished with an object you can use the managed object context method refreshObject:mergeChanges: to turn it into a fault.
根据说明,我们再试试下面代码:
HubBrand *oldBrand = [HubBrandDAO retrieveHubBrandByModelID:@""];
oldBrand.shownName = @"aaaa";
oldBrand.iconFile.shownName = @"tttttt"; //必须在save之后调用,不然无法达到打破retain circle的作用!
//[[DBManager sharedManager].managedObjectContext refreshObject:oldBrand mergeChanges:YES]; [[DBManager sharedManager] saveContext]; [[DBManager sharedManager].managedObjectContext refreshObject:oldBrand mergeChanges:YES];
brand 和 file 对象都被正确释放了!
ios CoreData NSManagedObject 生命周期的更多相关文章
-
(转)iOS应用程序生命周期(前后台切换,应用的各种状态)详解
原文:http://blog.csdn.net/totogo2010/article/details/8048652 iOS应用程序生命周期(前后台切换,应用的各种状态)详解 分类: ...
-
iOS对UIViewController生命周期和属性方法的解析
目录[-] iOS对UIViewController生命周期和属性方法的解析 一.引言 二.UIViewController的生命周期 三.从storyBoard加载UIViewController实 ...
-
转:iOS应用程序生命周期(前后台切换,应用的各种状态)详解
iOS应用程序生命周期(前后台切换,应用的各种状态)详解 分类: iOS开发进阶2012-10-08 15:35 42691人阅读 评论(30) 收藏 举报 iosapplication任务anima ...
-
【iOS开发】iOS对UIViewController生命周期和属性方法的解析
iOS对UIViewController生命周期和属性方法的解析 一.引言 作为MVC设计模式中的C,Controller一直扮演着项目开发中最重要的角色,它是视图和数据的桥梁,通过它的管理,将数据有 ...
-
iOS - ViewController的生命周期
iOS SDK中提供很多原生的ViewController,大大提高了我们的开发效率:那么下面我们就根据开发中我们常用的ViewController谈一谈它的生命周期: (一)按照结构和用法可以对iO ...
-
iOS 应用程序生命周期
开发应用程序都要了解其生命周期. 今天我们接触一下iOS应用程序的生命周期, iOS的入口在main.m文件: int main(int argc, char * argv[]) { @autorel ...
-
iOS中的生命周期
对于一个iOS app来讲,生命周期是一个十分至关重要的东西.对于一个app来讲控制着app的开启.睡眠.关闭等状态:对于一个页面的来讲,控制页面的加载.显示.消失:对于一个View或者一个普通的类来 ...
-
2. iOS程序的生命周期
程序启动-生命周期 来自: QQ: 853740091 1.首先讲解UIApplication对象 (1)UIApplication对象是应用程序的象征,一个UIApplication对象就代表一个 ...
-
iOS控制器的生命周期分析和使用
转自http://blog.csdn.net/qijianli/article/details/7826979 iOS的SDK中提供很多原生ViewController,大大提高了我们的开发效率,下面 ...
随机推荐
-
emacs 新手笔记(三) —— 为 emacs 做一点简单的定制
ilocker:关注 Android 安全(新入行,0基础) QQ: 2597294287 在 emacs 启动时,会加载 ~/.emacs 文件.在该文件中编辑一些 lisp 代码,是一种最为简单的 ...
-
HTML5基础知识(2)--标题标签的使用
1.HTML文档中包含各种级别的标题,各种级别的标题由<h1>到<h6>元素来定义,<h1>至<h6>标题标记中的字母h是英文headline的简称.其 ...
-
CSS媒体查询,CSS根据不同的分辨率显示不同的样式
在写自适应网页的时候,我们需要网页有几种显示方式,我们可以用CSS实现这个功能 使用CSS提供的媒体查询,我们可以根据屏幕分辨率来使用相应的CSS样式 @media screen and (max-w ...
-
OC修饰词 - 内存管理
<招聘一个靠谱的 iOS>—参考答案(上) 说明:面试题来源是微博@我就叫Sunny怎么了的这篇博文:<招聘一个靠谱的 iOS>,其*55题,除第一题为纠错题外,其他54道均 ...
-
JNDI--Java命名和目录接口
JNDI主要用于在容器中配置某些资源,让所有项目可以使用.JNDI可以提供: 1:数据库连接池. 自定义连接池 第三方连接池 Dbcp ...
-
js介绍
---恢复内容开始--- js介绍 最近学习了js,那么我今天给大家介绍下js.希望对初学者有些帮助. js就是常说的JavaScript.JavaScript是目前世界上最流行的编程语言之一.这门语 ...
-
文件I/O实践(2) --文件stat
功能:获取文件元数据 #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> int st ...
-
痞子衡嵌入式:ARM Cortex-M文件那些事(0)- 文件关联
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家讲的是嵌入式开发里的文件关联. 本篇是文件系列第一篇,本系列文章会逐一介绍ARM Cortex-M开发过程中(以IAR集成开发环境为例,其他开发 ...
-
sublime3添加verilog自动补全代码段
前言 sublime默认的verilog自动补全十分垃圾,不过提供了代码段这个功能,你可以自己写个重用率高的代码段减轻工作量.写个模板当tb也很爽啦. 流程 1.打开user文件夹,创建verilog ...
-
html5-超级链接
<!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8&qu ...