【问题标题】:Syncing android contacts with server将 android 联系人与服务器同步
【发布时间】:2017-01-15 20:21:06
【问题描述】:

我正在创建一个 android 应用程序,用于跟踪保存在设备上的所有联系人(出现在默认 android 联系人应用程序中的联系人)并将它们保存在服务器上的 MySQL 表中。

我设法使用ContectResolver 读取了所有联系人数据:

    //to read only android address book
    String where = ContactsContract.Contacts.IN_VISIBLE_GROUP + " = '1'";

    String[] projection = new String[] { ContactsContract.Contacts._ID, ContactsContract.Contacts.LOOKUP_KEY, ContactsContract.Contacts.HAS_PHONE_NUMBER};

    ContentResolver cr = context.getContentResolver();
    Cursor cursor = cr.query(ContactsContract.Contacts.CONTENT_URI, projection, where, null, null);

然后,我使用ContactsContract.Contacts._ID 查询其他联系人数据:

// Perform a query to retrieve the contact's name parts
        String[] nameProjection = new String[] {
                ContactsContract.CommonDataKinds.StructuredName.GIVEN_NAME,
                ContactsContract.CommonDataKinds.StructuredName.MIDDLE_NAME,
                ContactsContract.CommonDataKinds.StructuredName.FAMILY_NAME,
                ContactsContract.CommonDataKinds.StructuredName.DISPLAY_NAME
        };

        Cursor nameCursor = cr.query(
                ContactsContract.Data.CONTENT_URI,
                nameProjection,
                ContactsContract.Data.MIMETYPE + " = '" +
                        ContactsContract.CommonDataKinds.StructuredName.CONTENT_ITEM_TYPE + "' AND " +
                        ContactsContract.CommonDataKinds.StructuredName.CONTACT_ID
                        + " = ?", new String[] { id }, null);

        // Retrieve the name parts
        String firstName = "", middleName = "", lastName = "", displayName = "";
        if(nameCursor.moveToNext()) {
            firstName = nameCursor.getString(nameCursor.getColumnIndex(
                    ContactsContract.CommonDataKinds.StructuredName.GIVEN_NAME));
            middleName = nameCursor.getString(nameCursor.getColumnIndex(
                    ContactsContract.CommonDataKinds.StructuredName.MIDDLE_NAME));
            lastName = nameCursor.getString(nameCursor.getColumnIndex(
                    ContactsContract.CommonDataKinds.StructuredName.FAMILY_NAME));
            displayName = nameCursor.getString(nameCursor.getColumnIndex(
                    ContactsContract.CommonDataKinds.StructuredName.DISPLAY_NAME));
        }

然后我将所有这些数据(包括联系人的电子邮件地址和电话号码)发送到服务器。

我希望能够像这样定期备份设备的联系人,但能够通过将旧数据与新数据进行比较来检测联系人数据的变化。但要做到这一点,我需要在 android 设备中的联系人和我保存联系人数据的数据库之间建立某种链接。

经过一番搜索,我找到了 3 个选项:

ContactsContract.Contacts._ID
ContactsContract.Contacts.LOOKUP_KEY
ContactsContract.RawContacts._ID

但据我发现,所有这些都不可靠,并且在某些情况下可能会发生变化 - 使我的数据库与设备联系人列表之间的链接无效。

我考虑过使用ContentObserver 检测联系人变化的选项。但是,我希望能够检测到联系人的变化,即使我的应用程序已被卸载,然后一些联系人已更改,然后我的应用程序已重新安装。

我是否可以使用每个联系人的可靠识别密钥来了解某个联系人何时被更改或删除?

编辑:

我现在发现这个变量存在:ContactsContract.ContactsColumns.CONTACT_LAST_UPDATED_TIMESTAMP

问题是它只在 API 级别 18 中引入。我正在研究最小 API 15。有什么东西可以代替缺少的 API 级别吗?

【问题讨论】:

    标签: android synchronization android-contacts


    【解决方案1】:

    恐怕无法知道特定联系人是否发生了变化。因此,您可以在整个地址簿 URI 上注册观察者。这样,您就可以在通讯录中的任何内容发生变化时进行收听。

    现在有了观察者,您可以扫描所有联系人并将数据“同步”到您的服务器。

    在您的特定情况下,一对 ContactsContract.Contacts._ID 和 ContactsContract.Contacts.LOOKUP_KEY 应该足够可靠。但要知道是否有任何变化,您需要:

    • 将所有联系人发送到服务器并在服务器上执行同步逻辑(ID 和查找键应存储在 MySQL 上)
    • 在您的设备(sqllite db、realm、任何其他存储)上保留联系人的本地副本,以避免每次同步时都将未更改的联系人发送到服务器

    【讨论】:

    • 感谢您的回复,我编辑了我的问题 - 了解 ContactsContract.ContactsColumns.CONTACT_LAST_UPDATED_TIMESTAMP。如果我按照您描述的方式实现它,您将如何检查服务器端的 ID 和 LOOKUP KEY?我的意思是,如果只有一个匹配就足够了,还是有可能新联系人将获得与过去的联系人相同的 ID 或 LOOKUP KEY?
    • 如果联系人的ID + LOOKUP KEY发生变化,则不太可能与其他联系人相同。因此,如果您修改联系人,它的 LOOKUP KEY 可能会被更改,所以基本上您正在创建新联系人并且必须删除旧联系人
    猜你喜欢
    • 1970-01-01
    • 2018-11-01
    • 1970-01-01
    • 1970-01-01
    • 2016-12-19
    • 2016-09-17
    • 1970-01-01
    • 1970-01-01
    • 2011-10-09
    相关资源
    最近更新 更多