17

一个 android 库项目包含一些提供者,它们的权限在合约类中定义如下:

public static final String CONTENT_AUTHORITY = "my.com.library.providers.tester";
private static final Uri BASE_CONTENT_URI = Uri.parse("content://" + CONTENT_AUTHORITY);

现在有很多应用程序项目使用这个库项目。我目前遇到的问题是,对于每个应用程序项目,我需要在每个应用程序的库项目中都有一个单独的分支,只是为了拥有唯一的内容权限。这会产生一些版本管理问题(例如将功能/错误修复从一个分支传播到每个其他分支等)。相反,我想将定义内容权限的责任委托给应用程序项目。有没有办法做到这一点?

4

2 回答 2

23

我知道这是一个老话题,但今天遇到了这个问题,我们已经开发了很长一段时间,所以还没有准备好检查我们的内容提供者合同中的所有静态并更改它们,也因为我们的内容提供者和数据库是由生成的Eclipse的Mechanoid 插件(是的,我也是作者!:))

我想出的解决方案是向我们生成的合约添加一个静态初始化程序,该合约使用反射来查找一个类,并在其存在时使用静态 CONTENT_AUTHORITY 字段,如果不回退到默认值:

public class QuxContract  {
    public static final String CONTENT_AUTHORITY = initAuthority();

    private static String initAuthority() {
        String authority = "com.example.app.data.qux";

        try {

            ClassLoader loader = QuxContract.class.getClassLoader();

            Class<?> clz = loader.loadClass("com.example.app.data.QuxContentProviderAuthority");
            Field declaredField = clz.getDeclaredField("CONTENT_AUTHORITY");

            authority = declaredField.get(null).toString();
        } catch (ClassNotFoundException e) {} 
        catch (NoSuchFieldException e) {} 
        catch (IllegalArgumentException e) {
        } catch (IllegalAccessException e) {
        }

        return authority;
    }

    private static final Uri BASE_CONTENT_URI = Uri.parse("content://" + CONTENT_AUTHORITY);
// ...

现在在链接到库项目的每个项目中都可以提供自己的权限:

package com.example.app.data;

public class QuxContentProviderAuthority {
    public static final String CONTENT_AUTHORITY = "com.example.app.data.baz";
}

另外,不要忘记更改清单中的权限

于 2013-01-29T20:56:33.513 回答
22

应用程序是唯一绝对需要了解权限的应用程序,因为它是<provider>在清单中使用android:authorities属性声明的应用程序。

因此,原则上,它应该“正常工作”,只要您从提供程序中删除所有特定于权限的逻辑,例如:

  • 那些静态数据成员(现在移动到托管应用程序)
  • UriMatcher(自己滚动一些不检查权威,但专注于其余部分的东西Uri

如果出于某种原因,您绝对确定您的提供者需要知道其权限,那么应用程序必须在提供者用于实际工作之前将其提供给提供者。可能的方法包括:

  • 由于 aContentProvider是一个自然单例,将它分配给一个静态数据成员,然后通过自定义Application类的自定义方法向它提供权限字符串(因为首先初始化提供程序,所以这应该有效)

  • 如果您只支持 API 级别 11+,请Application使用自定义类call()ContentResolverContentProvider

  • 假设唯一真正的调用(例如 to query(), insert())是有效的,并且只是根据你第一次Uri看到的内容来惰性初始化你的权限

于 2012-05-28T22:39:24.713 回答