从自定义内容提供程序中访问数据

3
我有两个基于内容提供程序的应用A和B。它们都有自己的内容提供程序,并设置为从A和B读取数据以及反过来。当另一个应用程序在后台时,一切正常。但是如果该应用程序被杀死或不在后台,则找不到另一个内容提供程序。例如,应用程序B想从应用程序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(Query):content://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” 无法说明问题,请发布您的代码和完整的堆栈跟踪信息。 - pskink
请检查清单文件中的<provider>标签,并考虑将内容提供程序代码放入库中。 - Pomagranite
@Pomagranite:我已经放置了<provider>代码。是的,我正在将内容提供程序代码用作库。 - bianca
MyContentProvider的代码是什么样子的? - bryan
@bryan:添加了MyContentProvider代码。值得再次提醒的是,当两个应用程序同时运行(一个在前台,另一个在后台)时,一切都运行良好。 - bianca
显示剩余3条评论
3个回答

0

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

但是,您可以从清单或字符串资源动态读取权限。在Android内容提供程序权限定义是否违反DRY规则?上我的回答中,我概述了我们如何在OpenTasks-Provider中实现此功能。


我正在尝试使用内容提供程序库。我期望调用init()应该初始化使用内容提供程序库创建自己的cp的应用程序。当然它没有起作用。是否有可能将cp包装为库。如何动态初始化它? - bianca
当然可以在库中发布ContentProvider。这也是我们使用TaskProvider的方式(目前我们将其作为jar发布,但我们计划切换到aar)。请查看我们如何“自动初始化”我们的TaskProvider。 (https://github.com/dmfs/opentasks-provider/blob/master/src/org/dmfs/provider/tasks/TaskProvider.java#L159) - Marten

0

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

如果是这样,那么当应用程序尚未启动时,内容提供程序失败的原因可能是这个:在这种情况下,UriMatcher为空,并且query()方法中的switch会回退到default,从而抛出IllegalArgumentException

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


我想动态初始化(添加URIs)UriMatcher。内容提供程序是否可以从清单文件中读取其权限? - bianca
getContext() 可在 onCreate() 中使用(不能在构造函数中)。上下文为您提供了许多信息。在您的情况下,getPackageManager()getPackageName() 等是否提供了您所需的内容? - bwt

0

您正在设置权限为

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

所以你的mAuthority是com.example.provider或者com.example.provider.myprovider

然而,你在清单文件中定义了权限为

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

那是 com.example.appA-provider


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接