【问题标题】:Accessing data from Custom Content Providers从自定义内容提供者访问数据
【发布时间】:2016-05-14 06:32:02
【问题描述】:

我有两个基于内容提供程序的应用程序 A 和 B。两者都有自己的内容提供程序,并设置为从 A 和 B 读取数据,反之亦然。当另一个应用程序在后台时,一切正常。但是如果应用程序被杀死或不存在于后台,则无法找到另一个内容提供者。例如,App B 想要从 App A 读取数据。当 'A' 在后台运行时,'B' 可以成功地从 'A' 读取数据,但如果 'A' 没有运行,则会出现致命错误(未找到匹配 uri)在后台。

有什么想法吗?

[编辑] 我遇到了与this post 相同的问题。 我在两个应用的清单中都有这个:

   <provider
        android:name="MyContentProvider"
        android:authorities="com.example.${applicationId}-provider"
        android:enabled="true"
        android:exported="true"
        android:grantUriPermissions="true">
    </provider>

这是我得到的错误:

将异常写入包裹 java.lang.IllegalArgumentException:不支持的 URI(查询):内容://com.example.appA-provider/appA 在 com.example.provider.MyContentProvider.query(MyContentProvider.java:142) 在 android.content.ContentProvider.query(ContentProvider.java:1007) 在 android.content.ContentProvider$Transport.query(ContentProvider.java:218) 在 android.content.ContentProviderNative.onTransact(ContentProviderNative.java:112) 在 android.os.Binder.execTransact(Binder.java:461)

注意:只有当另一个应用程序不在后台时才会发生这种情况,否则它会按预期工作(可以正常读取彼此的数据)。

[编辑 2] 这是 MyContentProvider 的代码:

package com.example.provider;

import android.content.ContentProvider;
import android.content.ContentValues;
import android.content.UriMatcher;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteQueryBuilder;
import android.net.Uri;


public class MyContentProvider extends ContentProvider {

    private static DatabaseHelper dbHelper;

    private static final int ALL_ENTRIES = 1;
    private static final int SINGLE_ENTRY = 2;

    private String mAuthority = BuildConfig.APPLICATION_ID;
    private static UriMatcher uriMatcher;

    public Uri CONTENT_URI= null;

    static {
        uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
    }

    public MyContentProvider() {}

    public void init(String packageName, String authority) {
        if (authority == null) {
            setAuthority(packageName, true);
        } else {
            setAuthority(authority, false);
        }

        uriMatcher.addURI(getAuthority(), TABLE_NAME, ALL_ENTRIES);
        uriMatcher.addURI(getAuthority(), TABLE_NAME + "/#", SINGLE_ENTRY);
        CONTENT_URI =
                Uri.parse("content://" + getAuthority() + "/" + TABLE_NAME);
    }

    private void setAuthority(String packageName, boolean isPackageName) {
        if (isPackageName) {
            mAuthority = packageName + ".myprovider";
        } else {
            mAuthority = packageName;
        }
    }

    public String getAuthority() {
        return mAuthority;
    }

    public Uri getContentUri() {
        return CONTENT_URI;
    }

    @Override
    public boolean onCreate() {
        dbHelper = new DatabaseHelper(getContext());
        return false;
    }

    //Return the MIME type corresponding to a content URI
    @Override
    public String getType(Uri uri) {

        if (uri == null) {
            throw new IllegalArgumentException("Content uri is null: " + uri);
        }
        if (uriMatcher == null) {
            throw new IllegalArgumentException("Unsupported Match URI: " + uri);
        }

        switch (uriMatcher.match(uri)) {
            case ALL_ENTRIES:
                return "vnd.android.cursor.dir/vnd." + getAuthority() + "." + TABLE_NAME;
            case SINGLE_ENTRY:
                return "vnd.android.cursor.item/vnd." + getAuthority() + "." + TABLE_NAME;
            default:
                throw new IllegalArgumentException("Unsupported URI: " + uri);
        }
    }

    @Override
    public Uri insert(Uri uri, ContentValues values) {
        Uri _uri = null;
        long id = 0;
        SQLiteDatabase db = dbHelper.getWritableDatabase();
        switch (uriMatcher.match(uri)) {
            case ALL_ENTRIES:
            case SINGLE_ENTRY:
                id = db.insert(TABLE_NAME, null, values);
                getContext().getContentResolver().notifyChange(uri, null);
                _uri = Uri.parse(CONTENT_URI + "/" + id);
                break;
            default:
                throw new IllegalArgumentException("Unsupported URI (insert): " + uri);
        }

        return _uri;
    }

    @Override
    public Cursor query(Uri uri, String[] projection, String selection,
                        String[] selectionArgs, String sortOrder) {

        SQLiteDatabase db = dbHelper.getWritableDatabase();
        SQLiteQueryBuilder queryBuilder = new SQLiteQueryBuilder();

        Cursor cursor = null;
        String id = null;

        switch (uriMatcher.match(uri)) {
            case ALL_ENTRIES:
                queryBuilder.setTables(TABLE_NAME);
                cursor = queryBuilder.query(db, projection, selection,
                        selectionArgs, null, null, sortOrder);
                break;
            case SINGLE_ENTRY:
                queryBuilder.setTables(TABLE_NAME);
                id = uri.getPathSegments().get(1);
                if (id != null && !id.isEmpty()) {
                    queryBuilder.appendWhere(TABLE_NAME + "=" + id);
                }

                cursor = queryBuilder.query(db, projection, selection,
                        selectionArgs, null, null, sortOrder);
                break;
            default:
                throw new IllegalArgumentException("Unsupported URI(Query): " + uri);
        }

        return cursor;
    }
}

【问题讨论】:

  • Match uri not found 没说什么,贴出你的代码和完整的堆栈跟踪
  • 检查清单 标记并考虑将内容提供程序代码放入库中?
  • @Pomagranite:我在上面放了 代码。是的,我使用内容提供程序代码作为库。
  • @pskink:用 代码和堆栈跟踪更新了帖子。它并不多,但这就是我所得到的。我没有看到提供者的代码有任何问题,因为其他应用程序可以成功获取数据。我怀疑我在 配置中遗漏了一些东西。
  • MyContentProvider 的代码是什么样的?

标签: android android-contentprovider


【解决方案1】:

您不能像这样在代码中的其他位置初始化内容提供程序,因为 ContentProvider 可能是您应用的第一个(或唯一一个)被实例化的组件。

但是,您可以从 Manifest 或 String 资源中动态读取权限。在我对Does Android Content Provider authority definition break the DRY rule? 的回答中,我概述了我们如何在 OpenTasks-Provider 中做到这一点。

【讨论】:

  • 我正在尝试使用库作为内容提供者。我期待调用 init () 应该初始化使用内容提供程序库的应用程序来创建它自己的 cp。当然,它不起作用。是否有可能将 cp 包装为库。如何动态初始化?
  • 当然,您可以在库中发布 ContentProvider。这也是我们对 TaskProvider 所做的事情(目前我们将它作为 jar 发送,但我们计划切换到 aar。看看我们如何“自动初始化”我们的 TaskProvider
【解决方案2】:

我没有在内容提供程序中看到对 init() 的调用。它是否仅作为应用程序启动的一部分从其他地方调用?

如果是这样,这可以解释为什么内容提供程序在应用程序尚未启动时失败:在这种情况下,UriMatcher 为空,query() 方法中的switch 回退到default,女巫抛出IllegalArgumentException.

您应该在onCreate() 中调用init() 或在静态初始化程序中完全初始化UriMatcher

【讨论】:

  • 我想动态初始化(添加 Uris)UriMatcher。内容提供者是否有可能从 Manifest 中读取其权限?
  • getContext()onCreate() 中可用(不在构造函数中)。上下文使您可以访问大量信息。在您的情况下,getPackageManager(), getPackageName(), ... 不提供您需要的东西吗?
【解决方案3】:

您将权限设置为

private void setAuthority(String packageName, boolean isPackageName) {
        if (isPackageName) {
            mAuthority = packageName + ".myprovider";
        } else {
            mAuthority = packageName;
        }
    }

所以您的 mAuthority 是 com.example.providercom.example.provider.myprovider

但是,您已将清单中的权限定义为

android:authorities="com.example.${applicationId}-provider"

那是com.example.appA-provider

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-08-07
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多