0

在我们开始之前,我知道 SMS 内容提供者是无证的。我知道不推荐使用它。

由于无论如何我都在使用它,所以我想要一些帮助来实现我的目标之一。我目前能够毫无问题地添加和删除消息。在我删除了整个线程并尝试恢复它之后,问题就出现了。消息在数据库中,(如果我尝试再次添加它们,我会收到错误消息,

android.database.sqlite.SQLiteConstraintException: PRIMARY KEY must be unique (code 19)

所以我知道消息在那里。)但是如果我打开消息应用程序,它们不会显示。

如果我从线程中删除除一条消息之外的所有消息,则恢复的消息会很好地显示在他们的线程中。关于为什么我无法创建新消息线程的任何想法?

Uri allSmsUri = Uri.parse("content://sms/");

.

getContentResolver().delete(allSmsUri, "_id = ?", new String[] { m.get_id() });

.

        ContentValues values = new ContentValues();
        values.put("person", m.getPerson());
        values.put("_id", m.get_id());
        values.put("thread_id", m.getThread_id());
        values.put("body", m.getBody());
        values.put("address", m.getAddress());
        values.put("read", m.getRead());
        values.put("date", m.getDate());
        values.put("type", m.getType());
        values.put("date_sent", m.getDate_sent());
        values.put("protocol", m.getProtocol());
        values.put("status", m.getStatus());
        values.put("reply_path_present", m.getreply_path_present());
        values.put("subject", m.getSubject());
        values.put("service_center", m.getservice_center());
        values.put("locked", m.getLocked());
        values.put("error_code", m.getError_code());
        values.put("seen", m.getSeen());

getContentResolver().insert(allSmsUri, values);
4

1 回答 1

1

可能是因为当您“删除”一个消息线程时,它在 SMS 内部被标记为已删除,但实际上并未从底层数据库中删除。当您尝试将线程添加回提供程序时,_ID 值与仍在数据库中的行冲突。

一般来说,使用 ContentResolver 方法从捆绑的内容提供者插入或删除数据是一个非常糟糕的想法(“糟糕的糟糕糟糕的糟糕,不好” - Detritus,troll,Men At Arms )。几乎所有这些内容提供商都有一个复杂的内部状态管理系统,如果您尝试“滚动自己的”流程,您一定会搞砸。

相反,您应该坚持使用意图进行插入和删除。好处是:

  • 包含内容提供者的捆绑应用会为您处理状态管理。
  • 尝试做提供者不支持的事情不会犯错误
  • 您获得修改提供程序的临时权限,因此您不必在自己的应用程序中请求权限。用户就是这样。

如果您必须自己动手,请仔细阅读文档,包括 javadoc。如果您没有看到问题的答案,您应该假设有人故意不希望您做某事。你可以自由阅读开源,但你自己。

我是否还必须指出,使用未记录的、不受支持的 API 会保证您的应用程序将来会崩溃?

于 2013-06-13T18:22:34.393 回答