Android meta-data

本文深入探讨了Android中meta-data的使用,详细介绍了如何在AndroidManifest.xml中配置和使用meta-data,包括其在不同组件中的作用及如何动态获取。同时,文章提供了代码示例,展示了如何处理不同类型的数据值。

开发板推荐:天空星STM32F407VET6开发板

超高性价比 STM32主控 | 超高主频 | 一板兼容百芯 | 比赛神器 | 沉金彩色丝印

Android 中 meta-data 用在 AndroidManifest.xml 文件中。<meta-data>标签是提供组件额外的数据用的,它本身就是一个键值对(Bundle),可以自定义名称和值(value或resource)。

它可以包含在以下6个组件中:

  • <application>
  • <activity>
  • <activity-alias>
  • <provider>
  • <receiver>
  • <service>

meta-data 主要有以下几个常见用途:

  • 在APP发布的时候,通过修改 meta-data,来标记不同的“发布渠道”,以方便脚本自动化修改、打包、发布。
  • 引入第三方SDK时,配置appId和appKey
  • 作为日志标记,在公共父类中获取该值并作为是否打印日志的标记,用多态化的思想控制日志,免去了冗杂的在代码中进行分别配置。
  • 与 activity-alias 配合使用,meta-data作为标记,来实现类似“拨号-联系人-短信”应用的通过同一个activity打开不同的选项卡。

基础入门 —— 从组件中获取meta-data

使用meta-data的好处是,它只需要在manifest.xml一个文件中配置,就可以在代码中的各个组件里去动态获取,实现相应的需求。集中配置的方式方便了后续的管理与修改,配合上 gradle 的 manifestPlaceholders 还可以做更加集中化的处理。

Manifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.metadatademo"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="23"
        android:targetSdkVersion="25" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="application" >
        <activity
            android:name="MainActivity"
            android:label="activity" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>

            <meta-data
                android:name="meta_data"
                android:value="activity_meta_data_value" />
        </activity>

        <activity-alias
            android:name="MyActivityAlias"
            android:label="activity_alias"
            android:targetActivity=".MainActivity" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.DEFAULT" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>

            <meta-data
                android:name="meta_data"
                android:value="activity_alias_meta_data_value" />
        </activity-alias>

        <meta-data
            android:name="meta_data"
            android:value="application_meta_data_value" />

        <provider
            android:name="MyContentProvider"
            android:authorities="com.chy.authorities"
            android:label="provider" >
            <meta-data
                android:name="meta_data"
                android:value="provider_meta_data_value" />
        </provider>

        <receiver
            android:name="MyBroadcastReceiver"
            android:label="receiver" >
            <intent-filter>
                <action android:name="android.intent.action.PHONE_STATE" />
            </intent-filter>

            <meta-data
                android:name="meta_data"
                android:value="receiver_meta_data_value" />
        </receiver>

        <service
            android:name="MyService"
            android:label="service" >
            <meta-data
                android:name="meta_data"
                android:value="service_meta_data_value" />
        </service>
    </application>
</manifest>

MainActivity.java

public class MainActivity extends Activity {

	private TextView textview;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		textview = (TextView) findViewById(R.id.textview);

		PackageManager pm = getPackageManager();

		//application meta-data
		try {
			ApplicationInfo applicationInfo = pm.getApplicationInfo(getPackageName(), PackageManager.GET_META_DATA);
			addText(applicationInfo.metaData.getString("meta_data"));
		} catch (NameNotFoundException e) {
			e.printStackTrace();
		}

		//"activity"或"activity-alias" meta-data
		try {
			ActivityInfo activityInfo = pm.getActivityInfo(getComponentName(), PackageManager.GET_META_DATA);
			addText(activityInfo.metaData.getString("meta_data"));
		} catch (NameNotFoundException e) {
			e.printStackTrace();
		}

		//"provider" meta-data
		ComponentName providerComponentInfo = new ComponentName(this, MyContentProvider.class);
		try {
			ProviderInfo providerInfo = pm.getProviderInfo(providerComponentInfo, PackageManager.GET_META_DATA);
			addText(providerInfo.metaData.getString("meta_data"));
		} catch (NameNotFoundException e) {
			e.printStackTrace();
		}

		//"receiver" meta-data
		ComponentName receiverComponentInfo = new ComponentName(this, MyBroadcastReceiver.class);
		try {
			ActivityInfo receiverInfo = pm.getReceiverInfo(receiverComponentInfo, PackageManager.GET_META_DATA);
			addText(receiverInfo.metaData.getString("meta_data"));
		} catch (NameNotFoundException e) {
			e.printStackTrace();
		}

		//"service" meta-data
		ComponentName serviceComponentInfo = new ComponentName(this, MyService.class);
		try {
			ServiceInfo serviceInfo = pm.getServiceInfo(serviceComponentInfo, PackageManager.GET_META_DATA);
			addText(serviceInfo.metaData.getString("meta_data"));
		} catch (NameNotFoundException e) {
			e.printStackTrace();
		}
	}

	private void addText(String str) {
		textview.setText(textview.getText().toString() + "\n" + str);
	}
}

MyBroadcastReceiver.java

public class MyBroadcastReceiver extends BroadcastReceiver {

	@Override
	public void onReceive(Context arg0, Intent arg1) {
	}
}

MyContentProvider.java

public class MyContentProvider extends ContentProvider {

	@Override
	public int delete(Uri arg0, String arg1, String[] arg2) {
		return 0;
	}

	@Override
	public String getType(Uri arg0) {
		return null;
	}

	@Override
	public Uri insert(Uri arg0, ContentValues arg1) {
		return null;
	}

	@Override
	public boolean onCreate() {
		return false;
	}

	@Override
	public Cursor query(Uri arg0, String[] arg1, String arg2, String[] arg3, String arg4) {
		return null;
	}

	@Override
	public int update(Uri arg0, ContentValues arg1, String arg2, String[] arg3) {
		return 0;
	}
}

MyService.java

public class MyService extends Service {

	@Override
	public IBinder onBind(Intent arg0) {
		return null;
	}
}

activity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.example.metadatademo.MainActivity" >

    <TextView
        android:id="@+id/textview"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />

</RelativeLayout>

进阶 —— 获取各种类型的值

在 Manifest.xml 配置 meta-data

<application
	android:name=".test.global.MyApplication"
	……>
        <meta-data
            android:name="mipmap"
            android:resource="@mipmap/ic_launcher" />
        <meta-data
            android:name="color"
            android:resource="@color/colorPrimary" />
        <meta-data
            android:name="layout"
            android:resource="@layout/activity_test" />
        <meta-data
            android:name="string"
            android:resource="@string/app_name" />
        <meta-data
            android:name="number"
            android:value="123" />
        <meta-data
            android:name="text"
            android:value="abc" />
            ……
 </application>           

然后在 MyApplication.java 中将它们打印出来

public class MyApplication extends Applicatoin {

	@Override
	public void onCreate() {
		super.onCreate();
		String[] keys = {"mipmap", "color", "layout", "string", "number", "text"};
        for(String key : keys) {
            print(MetaUtil.fromApplication(this, key));
        }
    }

    private void print(Object obj) {
        System.out.println(obj + " : " + obj.getClass());
    }
}

搜索关键词使用 System.out 来进行筛选,可以得出如下结论

  • resource中存储的都是对象的ID,整型
  • value支持 Integer、Float、Boolean、String 这四种常用类型
    在这里插入图片描述

终极 —— 获取字符串

在 meta-data 中,有个很容易让人忽略的小细节。

  • meta-data 的 value 如果是全数字,则获取为数字(较长的数字,为了保持精度,获取时会被自动使用科学计数法表示)
  • 只要全数字的value中出现了一个非数字,则会被当成字符串处理

因此,在引入第三方SDK时,如果有全数字,且超过13位的APPID这种,很容易获取出错(最后得到的是其它数值)

我做了一个工具类,用来解决这种情况(需要被当成字符串的全数字前加一个@作为标记符)

public class MetaUtil {
    /**
     * 由于是自动识别类型,全数字会被识别为Integer或Float,且float和double还会自动被转为科学计数法以保留其最大精度
     */
    private static String toString(Object value) {
        String result = null;
        if (value != null) {
            String valueClassName = value.getClass().getSimpleName();
            switch (valueClassName) {
                case "Long":
                case "Float":
                case "Double":
                    NumberFormat numberFormat = NumberFormat.getInstance();
                    //不使用分组方式显示数据(即科学计数法),但这样可能会导致精度丢失,精度最多13位
                    numberFormat.setGroupingUsed(false);
                    result = numberFormat.format(value);
                    break;
                case "String":
                	//当数字多于13位时(或者保守起见,只要是需要被当成字符串的全数字)都在其首位加上一个@符号
                    result = value.toString();
                    if (result.startsWith("@")) {
                        result = result.substring(1);
                    }
                    break;
                default:
                    result = String.valueOf(value);
                    break;
            }
        }
        return result;
    }

    public static Object fromApplication(Context context, String key) {
        PackageManager packageManager = context.getPackageManager();
        try {
            ApplicationInfo applicationInfo = packageManager.getApplicationInfo(context.getPackageName(), PackageManager.GET_META_DATA);
            return applicationInfo.metaData.get(key);
        } catch (PackageManager.NameNotFoundException e) {
            e.printStackTrace();
        }
        return null;
    }

    public static String fromApplicationToStr(Context context, String key) {
        return toString(fromApplication(context, key));
    }

    public static Object fromActivity(FragmentActivity activity, String key) {
        PackageManager packageManager = activity.getPackageManager();
        try {
            ActivityInfo activityInfo = packageManager.getActivityInfo(activity.getComponentName(), PackageManager.GET_META_DATA);
            return activityInfo.metaData.get(key);
        } catch (PackageManager.NameNotFoundException e) {
            e.printStackTrace();
        }
        return null;
    }

    public static String fromActivityToStr(FragmentActivity activity, String key) {
        return toString(fromActivity(activity, key));
    }

    public static Object fromService(Service service, String key) {
        PackageManager packageManager = service.getPackageManager();
        ComponentName componentName = new ComponentName(service, service.getClass());
        try {
            ServiceInfo serviceInfo = packageManager.getServiceInfo(componentName, PackageManager.GET_META_DATA);
            return serviceInfo.metaData.get(key);
        } catch (PackageManager.NameNotFoundException e) {
            e.printStackTrace();
        }
        return null;
    }

    public static String fromServiceToStr(Service service, String key) {
        return toString(fromService(service, key));
    }

    /**
     * BroadcastReceiver 的 context
     * 静态注册:android.app.ReceiverRestrictedContext
     * 动态注册:在哪个 activity 注册的,就是那个 activity 的 context
     */
    public static Object fromReceiver(Context context, BroadcastReceiver receiver, String key) {
        PackageManager packageManager = context.getPackageManager();
        ComponentName componentName = new ComponentName(context, receiver.getClass());
        try {
            ActivityInfo serviceInfo = packageManager.getReceiverInfo(componentName, PackageManager.GET_META_DATA);
            return serviceInfo.metaData.get(key);
        } catch (PackageManager.NameNotFoundException e) {
            e.printStackTrace();
        }
        return null;
    }

    public static String fromReceiverToStr(Context context, BroadcastReceiver receiver, String key) {
        return toString(fromReceiver(context, receiver, key));
    }
}

开发板推荐:天空星STM32F407VET6开发板

超高性价比 STM32主控 | 超高主频 | 一板兼容百芯 | 比赛神器 | 沉金彩色丝印

框架内部支持中/英文(其他语言只需要在对应的string.xml中取相同的名字即可)内部对话框背景图片、按钮支持自定义了查看版本中的Log只需要过滤AppUpdate开头的Tag重点: 如果没有设置downloadPath则默认为getExternalCacheDir()目录,同时不会申请[存储]权限!目录编译问题效果图功能介绍DownloadManagerUpdateConfiguration使用步骤Demo下载体验版本更新记录结语编译问题因为适配了Android O的通知栏,所以依赖的v7包版本比较高appcompat-v7:26.1.0使用的gradle版本为gradle-4.1-all,所以建议使用Android Studio 3.0及以上的版本打开此项目效果图     功能介绍 支持断点下载 支持后台下载 支持自定义下载过程 支持 设备 >= Android M 动态权限的申请 支持通知栏进度条展示(或者自定义显示进度) 支持Android N 支持Android O 支持中/英文双语 支持自定内置对话框的样式 使用HttpURLConnection下载,未集成其他第三方框架更加详细的文档参阅此处《AppUpdate API文档》DownloadManager:配置文档初始化使用DownloadManager.getInstance(this)属性描述默认值是否必须设置context上下文nulltrueapkUrlapk的下载地址nulltrueapkNameapk下载好的名字nulltruedownloadPathapk下载的位置getExternalCacheDir()falseshowNewerToast是否提示用户 "当前已是最新版本"falsefalsesmallIcon通知栏的图标(资源id)-1trueconfiguration这个库的额外配置nullfalseapkVersionCode更新apk的versionCode (如果设置了那么库中将会进行版本判断下面的属性也就需要设置了)1falseapkVersionName更新apk的versionNamenullfalseapkDescription更新描述nullfalseapkSize新版本的安装包大小(单位M)nullfalseauthorities兼容Android N uri授权应用包名falseUpdateConfiguration:配置文档属性描述默认值notifyId通知栏消息id1011notificationChannel适配Android O的渠道通知详情查阅源码httpManager设置自己的下载过程nullbreakpointDownload是否需要支持断点下载trueenableLog是否需要日志输出trueonDownloadListener下载过程的回调nulljumpInstallPage下载完成是否自动弹出安装页面trueshowNotification是否显示通知栏进度(后台下载提示)trueforcedUpgrade是否强制升级falseonButtonClickListener按钮点击事件回调nulldialogImage对话框背景图片资源(图片规范参考demo)-1dialogButtonColor对话框按钮的颜色-1dialogButtonTextColor对话框按钮的文字颜色-1所有版本:点击查看使用步骤第一步: app/build.gradle进行依赖implementation 'com.azhon:appupdate:1.7.3'第二步:创建DownloadManager,更多用法请查看这里示例代码DownloadManager manager = DownloadManager.getInstance(this); manager.setApkName("appupdate.apk")         .setApkUrl("https://raw.githubusercontent.com/azhon/AppUpdate/master/apk/appupdate.apk")         .setSmallIcon(R.mipmap.ic_launcher)         //可设置,可不设置         .setConfiguration(configuration)         .download();第三步:兼容Android N 及以上版本,在你应用的Manifest.xml添加如下代码<--android:authorities="${applicationId}"  这个值必须与DownloadManager中的authorities一致(不设置则为应用包名)--> <provider     android:name="android.support.v4.content.FileProvider"     android:authorities="${applicationId}"     android:exported="false"     android:grantUriPermissions="true">     <meta-data         android:name="android.support.FILE_PROVIDER_PATHS"         android:resource="@xml/file_paths_public" /> </provider>第四步:资源文件res/xml/file_paths_public.xml内容<?xml version="1.0" encoding="utf-8"?> <paths>     <external-path         name="app_update_external"         path="/" />     <external-cache-path         name="app_update_cache"         path="/" /> </paths>兼容Android O及以上版本,需要设置NotificationChannel(通知渠道);库中已经写好可以前往查阅NotificationUtil.java温馨提示:升级对话框中的内容是可以上下滑动的哦!如果需要实现自己一套下载过程,只需要继承BaseHttpDownloadManager 并使用listener更新进度public class MyDownload extends BaseHttpDownloadManager {}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值