android N 内核,Android Context深入理解

在应用开发过程中我们经常用到Context,有Activity的Context,Application的Context,还有Service的Context,这三个有什么区别?Context在Android中到底是个什么东西?本文阐述Context的定义,Context具体功能实现,以及各种Context的区别和使用方法。最后,本文总结了Android N适配时使用Context的一些注意事项。

定义

Context通常被翻译成上下文,它在Android中代表场景。一个Context就是一个场景,一个场景代表着一组用户与应用交互的过程。比如听音乐、打电话、发短信,这都分别对应着一个场景。

实现

Context是一个抽象类,定义了访问应用环境全局信息的接口;包括访问应用程序资源、打开Activity(startActivity())、启动Service(startService())、发送广播,文件读写等。Activity,Service以及Application均继承自Context,所以我们经常使用的startActivity(),getResource(),getSharedPreference(),getExternalFilesDir(),deleteDatabase()等等方法都来自于Context。UML类图如下所示:

2e8c980c55ff

Context

由上图可知,Activity,Service,Application并没有直接继承Context,而是继承自ContextWrapper。ContextWrapper是Context的包装类,内部包含一个Context的引用,指向Context的具体实现类ContextImpl。ContextWrapper内部的所有方法直接调用ContextImpl对应的方法。

ContextImpl在ActivityThread创建Activity,Service以及Application时被实例化。所以每个应用都有多个ContextImpl实例。ContextImpl内部包含一个指向ActivityThread.PackageInfo的引用mPackageInfo。ContextImpl内部方法功能都是通过调用mPackageInfo对象来实现的。

ActivityThread.PackageInfo是一个重量级类,每一个应用程序仅有一个ActivityThread.PackageInfo实例,所有ContextImpl内部的mPackageInfo都指向ActivityThread.PackageInfo的唯一实例。

使用区别

Application、Activity以及Service均继承自Context,在初始化ContextImpl时使用的参数不一样,所以对应的使用场景也不一样。

Application对应着应用程序全局相关的场景,比如我们可以写一个全局工具类,使用Application获取类加载器,Looper,或者做一些处理权限相关的操作等等(当然Activity与Service也可以)。虽然加载UI相关资源在使用上是合法的,但加载的布局会忽略用户设置订样式并会使用系统默认样式,而且在Android N的分屏模式中会出现不可预知的问题。所以不推荐使用Application的Context加载UI资源。

对话框只能存在于父窗口中,所以只能在Activity中打开,使用Application或者Service的Context会报异常:

android.view.WindowManager$BadTokenException: Unable to add window -- token null is not for an application。

不推荐在Application以及Service中使用startActivity()方法,因为它们会在新的Task中打开Activity。

适配Android N新特性

谷歌在安卓7的预览版中增加了多窗口分屏功能,能够并排的同时使用两个应用。多窗口利用Android的资源系统,基于窗口的大小动态的调整配置。在调整窗口大小时,屏幕大小最小宽度和朝向都会实时更新。所以我们需要使用正确的Context加载资源。对于UI相关资源(布局),最好使用其所在Activity的Context加载。如果使用Application加载布局UI等资源,有可能会出现在多窗口模式下无法正常工作的情况。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Android上,可以使用WebView类的getSettings().getUserAgentString()方法来获取当前内核信息。WebView是一个用于在应用程序中嵌入浏览器功能的类,它可以加载并显示Web页面。 获取当前内核的方法是通过调用WebView的getSettings()方法来获取WebSettings对象,然后再调用getUserAgentString()方法来获取用户代理字符串。用户代理字符串是一个包含了当前设备和浏览器相关信息的字符串,其中也包含了当前WebView使用的内核信息。 代码示例如下: ```java WebView webView = new WebView(context); String userAgent = webView.getSettings().getUserAgentString(); ``` 上述代码中,我们创建了一个WebView对象,并通过调用getSettings()方法获取了一个WebSettings对象。通过WebSettings对象的getUserAgentString()方法,我们最终获取到了当前内核的信息,将其存储在userAgent字符串中。 需要注意的是,此方法只能在WebView已经创建并加载完网页之后才能调用。否则,可能无法获取到正确的内核信息。 当我们获取到userAgent字符串后,可以通过解析字符串来获取当前内核的信息,例如判断是否是Chrome内核、WebView版本等。 总结,要在Android中获取当前内核,可以通过WebView的getSettings().getUserAgentString()方法获取用户代理字符串,并解析该字符串来获取内核信息。 ### 回答2: 在Android中,可以通过以下方式获取当前WebView使用的内核版本。 首先,在项目的build.gradle文件中,添加以下依赖项: implementation 'androidx.webkit:webkit:1.4.0' 然后,在代码中使用以下方法获取内核版本: String webViewVersion = ""; int webViewMajorVersion = 0; if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { WebView webView = new WebView(context); String userAgent = webView.getSettings().getUserAgentString(); Matcher matcher = Pattern.compile("Chrome/([0-9]+)").matcher(userAgent); if (matcher.find()) { webViewVersion = matcher.group(1); } } else { try { Class webViewFactoryClass = Class.forName("android.webkit.WebViewFactory"); Method getFactoryMethod = webViewFactoryClass.getDeclaredMethod("getFactory"); Object webViewFactory = getFactoryMethod.invoke(null); Method getWebViewLibraryMethod = webViewFactory.getClass().getDeclaredMethod("getWebViewLibrary"); String webViewLibrary = (String) getWebViewLibraryMethod.invoke(webViewFactory); webViewMajorVersion = Integer.parseInt(webViewLibrary.split("\\.")[0]); } catch (ClassNotFoundException | NoSuchMethodException | InvocationTargetException | IllegalAccessException e) { e.printStackTrace(); } } 以上代码首先通过WebView的UserAgent获取到Chrome内核的版本号,如果当前Android版本低于Android Lollipop(API 21),则使用反射获取WebView的内核版本。 最后,你可以将获取到的webViewVersion或者webViewMajorVersion用于显示或其他需要的地方。 ### 回答3: 在Android中,可以通过WebView的getSettings()方法获取当前正在使用的Web浏览器内核。 要获取当前内核,首先需要在Android中创建一个WebView对象。然后通过调用getSettings()方法来获取该WebView的设置对象。接下来,可以调用该设置对象的getUserAgentString()方法来获取当前WebView所使用的浏览器内核的信息。getUserAgentString()方法返回一个字符串,该字符串包含了当前浏览器的用户代理信息。用户代理信息通常包括了浏览器的名称、版本号以及内核的信息。 以下是一个简单的代码示例,展示了如何获取当前WebView的内核信息: ```java WebView webView = new WebView(context); WebSettings settings = webView.getSettings(); String userAgent = settings.getUserAgentString(); ``` 需要注意的是,getUserAgentString()方法返回的字符串可能包含了很多其他的信息,而不仅仅是浏览器内核的信息。因此,需要使用适当的方法来解析和提取所需的内核信息。 希望以上解答对您有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值