在使用coredata 的时候,需要永久性保存数据, 使用 NSManagedObjectContext 的 sava: 方法来实现。
在这个过程中发现各种不能保存,于是乎花了两天的时候来解决它,搜寻各种办法,不过找到的办法无数却始终解决不了我的问题,最后发现了我的问题所在,却让我哭笑不得。
总结在保存的时候遇到问题的解决办法:
1、使用了多个线程操作同一个 NSManagedObjectContext对象
最好是每个线程用一个 NSManagedObjectContext对象
2、NSManagedStoreCoordinator 有缓存数据,需要重新填充
[managedObjectContext setStalenessInterval:0.0]; //强制性从磁盘装载
3、NSManagedObjectContext 合并政策失败的核心数据是无法完成合并
[managedObjectContext setMergePolicy:NSMergeByPropertyObjectTrumpMergePolicy];//设置上下文对象合并
4、数据库的路径问题
检查数据库路径是否正确
5、更新了数据库结构
需要从模拟器或真机上删除项目,并clean 重新编译
6、存储的类型和数据库中定义的类型不匹配问题
最后说说我的问题: 使用coredata 在项目中建立表,它会自己来生成数据库,而在每个表中都有它自己生成的Z_PK,Z_ENT,Z_OPT 这三个字段,
Z_ENT 表在xcdatamodel 中的索引值,如果有6个表那么它的值的区间就是[1,6 ]
Z_OPT 表示的是每条数据被操作的次数,初始化值为1,只要是增删改查都会加1
而我在为了测试我的程序,自己去找到了数据库填写了数据,以至于 z_ent 和 z_opt这两个值未填写,导致了我保存数据时的各种不通过,
而用代码插入的数据 是会自动为这两个字段赋值的,这让我郁闷了很久,这次算是长记性了,在没有弄清这玩意的时候就算是测试数据也最好去用代码生成,不仅检测了代码也不会出现这种错误。。
processPendingChanges改变当前对象的状态图;
processPendingChanges类似于告诉字处理器在文档的编辑后更新它的内部状态,但不保存到磁盘。多用于用户界面上 更新数据;
processPendingChanges常用在单独线程上运行多个上下文;
在MacOS下,它发挥了更大,更普遍的作用;
save保存到磁盘;
调用save将自动调用processPendingChanges;
在我们保存数据到数据库的时候常用 save方法;
save 和 processPendingChanges 都是 NSManagedObjectContext 类的方法;
main.xml如下:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity" > <TextView android:id="@+id/textView" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerHorizontal="true" android:layout_centerVertical="true" android:text="期望有一天不用写代码也可以吃得起馒头" /> </RelativeLayout>
mainActivity如下:
package c.c; import android.app.Activity; import android.os.Bundle; import android.text.Spannable; import android.text.SpannableString; import android.text.SpannableStringBuilder; import android.text.method.LinkMovementMethod; import android.text.style.ClickableSpan; import android.text.style.URLSpan; import android.view.View; import android.widget.TextView; import android.widget.Toast; /** * Demo描述: * 为TextView中的文字设置链接,当点击此链接的时候执行相应的操作 */ public class MainActivity extends Activity { private TextView mTextView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); init(); } private void init() { mTextView=(TextView) findViewById(R.id.textView); String string = mTextView.getText().toString(); addUrlSpan(mTextView, string); mTextView.setFocusable(true); mTextView.setMovementMethod(LinkMovementMethod.getInstance()); catchUrl(/blog_article/mTextView/index.html); } private class TestSpan extends ClickableSpan { private String url; TestSpan(String url) { this.url = url; } @Override public void onClick(View widget) { if (url.equals("showinfo")) { Toast.makeText(MainActivity.this, "点击了链接", Toast.LENGTH_SHORT).show(); System.out.println("执行点击链接后的操作"); } } } private void addUrlSpan(TextView tv, String string) { SpannableString spanString = new SpannableString(string); URLSpan span = new URLSpan("showinfo"); int start = string.length() - 4; int end = string.length() - 1; spanString.setSpan(span, start, end, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE); tv.setText(spanString); } private void catchUrl(/blog_article/TextView textView/index.html) { CharSequence text = textView.getText(); if (text instanceof Spannable) { int end = text.length(); Spannable sp = (Spannable) textView.getText(); URLSpan[] urls = sp.getSpans(0, end, URLSpan.class); SpannableStringBuilder spannableStringBuilder = new SpannableStringBuilder(text); spannableStringBuilder.clearSpans(); for (URLSpan url : urls) { TestSpan testSpan = new TestSpan(url.getURL()); spannableStringBuilder.setSpan(testSpan, sp.getSpanStart(url), sp.getSpanEnd(url), Spannable.SPAN_EXCLUSIVE_INCLUSIVE); } textView.setText(spannableStringBuilder); } } }