WebVies类介绍,这些年我所经历的所有面试

// 是否支持Javascript,默认值false

settings.setJavaScriptEnabled(true);

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {

// 5.0以上允许加载http和https混合的页面(5.0以下默认允许,5.0+默认禁止)

settings.setMixedContentMode(WebSettings.MIXED_CONTENT_ALWAYS_ALLOW);

}

if (isNetworkConnected(context)) {

// 根据cache-control决定是否从网络上取数据

settings.setCacheMode(WebSettings.LOAD_DEFAULT);

} else {

// 没网,离线加载,优先加载缓存(即使已经过期)

settings.setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK);

}

WebViewClient

// 拦截页面加载,返回true表示宿主app拦截并处理了该url,否则返回false由当前WebView处理

// 此方法在API24被废弃,不处理POST请求

public boolean shouldOverrideUrlLoading(WebView view, String url) {

return false;

}

// 拦截页面加载,返回true表示宿主app拦截并处理了该url,否则返回false由当前WebView处理

// 此方法添加于API24,不处理POST请求,可拦截处理子frame的非http请求

@TargetApi(Build.VERSION_CODES.N)

public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) {

return shouldOverrideUrlLoading(view, request.getUrl().toString());

}

// 此方法废弃于API21,调用于非UI线程

// 拦截资源请求并返回响应数据,返回null时WebView将继续加载资源

// 注意:API21以下的AJAX请求会走onLoadResource,无法通过此方法拦截

public WebResourceResponse shouldInterceptRequest(WebView view, String url) {

return null;

}

// 此方法添加于API21,调用于非UI线程

// 拦截资源请求并返回数据,返回null时WebView将继续加载资源

@TargetApi(Build.VERSION_CODES.LOLLIPOP)

public WebResourceResponse shouldInterceptRequest(WebView view, WebResourceRequest request) {

return shouldInterceptRequest(view, request.getUrl().toString());

}

// 页面(url)开始加载

public void onPageStarted(WebView view, String url, Bitmap favicon) {

}

// 页面(url)完成加载

public void onPageFinished(WebView view, String url) {

}

// 将要加载资源(url)

public void onLoadResource(WebView view, String url) {

}

// 这个回调添加于API23,仅用于主框架的导航

// 通知应用导航到之前页面时,其遗留的WebView内容将不再被绘制。

// 这个回调可以用来决定哪些WebView可见内容能被安全地回收,以确保不显示陈旧的内容

// 它最早被调用,以此保证WebView.onDraw不会绘制任何之前页面的内容,随后绘制背景色或需要加载的新内容。

// 当HTTP响应body已经开始加载并体现在DOM上将在随后的绘制中可见时,这个方法会被调用。

// 这个回调发生在文档加载的早期,因此它的资源(css,和图像)可能不可用。

// 如果需要更细粒度的视图更新,查看 postVisualStateCallback(long, WebView.VisualStateCallback).

// 请注意这上边的所有条件也支持 postVisualStateCallback(long ,WebView.VisualStateCallback)

public void onPageCommitVisible(WebView view, String url) {

}

// 此方法废弃于API23

// 主框架加载资源时出错

public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {

}

// 此方法添加于API23

// 加载资源时出错,通常意味着连接不到服务器

// 由于所有资源加载错误都会调用此方法,所以此方法应尽量逻辑简单

@TargetApi(Build.VERSION_CODES.M)

public void onReceivedError(WebView view, WebResourceRequest request, WebResourceError error) {

if (request.isForMainFrame()) {

onReceivedError(view, error.getErrorCode(), error.getDescription().toString(), request.getUrl().toString());

}

}

// 此方法添加于API23

// 在加载资源(iframe,image,js,css,ajax…)时收到了 HTTP 错误(状态码>=400)

public void onReceivedHttpError(WebView view, WebResourceRequest request, WebResourceResponse errorResponse) {

}

// 是否重新提交表单,默认不重发

public void onFormResubmission(WebView view, Message dontResend, Message resend) {

dontResend.sendToTarget();

}

// 通知应用可以将当前的url存储在数据库中,意味着当前的访问url已经生效并被记录在内核当中。

// 此方法在网页加载过程中只会被调用一次,网页前进后退并不会回调这个函数。

public void doUpdateVisitedHistory(WebView view, String url, boolean isReload) {

}

// 加载资源时发生了一个SSL错误,应用必需响应(继续请求或取消请求)

// 处理决策可能被缓存用于后续的请求,默认行为是取消请求

public void onReceivedSslError(WebView view, SslErrorHandler handler, SslError error) {

handler.cancel();

}

// 此方法添加于API21,在UI线程被调用

// 处理SSL客户端证书请求,必要的话可显示一个UI来提供KEY。

// 有三种响应方式:proceed()/cancel()/ignore(),默认行为是取消请求

// 如果调用proceed()或cancel(),Webview 将在内存中保存响应结果且对相同的"host:port"不会再次调用 onReceivedClientCertRequest

// 多数情况下,可通过KeyChain.choosePrivateKeyAlias启动一个Activity供用户选择合适的私钥

@TargetApi(Build.VERSION_CODES.LOLLIPOP)

public void onReceivedClientCertRequest(WebView view, ClientCertRequest request) {

request.cancel();

}

// 处理HTTP认证请求,默认行为是取消请求

public void onReceivedHttpAuthRequest(WebView view, HttpAuthHandler handler, String host, String realm) {

handler.cancel();

}

// 通知应用有个已授权账号自动登陆了

public void onReceivedLoginRequest(WebView view, String realm, String account, String args) {

}

// 给应用一个机会处理按键事件

// 如果返回true,WebView不处理该事件,否则WebView会一直处理,默认返回false

public boolean shouldOverrideKeyEvent(WebView view, KeyEvent event) {

return false;

}

// 处理未被WebView消费的按键事件

// WebView总是消费按键事件,除非是系统按键或shouldOverrideKeyEvent返回true

// 此方法在按键事件分派时被异步调用

public void onUnhandledKeyEvent(WebView view, KeyEvent event) {

super.onUnhandledKeyEvent(view, event);

}

// 通知应用页面缩放系数变化

public void onScaleChanged(WebView view, float oldScale, float newScale) {

}

WebChromeClient

// 获得所有访问历史项目的列表,用于链接着色。

public void getVisitedHistory(ValueCallback<String[]> callback) {

}

// 控件在未播放时,会展示为一张海报图,HTML中可通过它的’poster’属性来指定。

// 如果未指定’poster’属性,则通过此方法提供一个默认的海报图。

public Bitmap getDefaultVideoPoster() {

return null;

}

// 当全屏的视频正在缓冲时,此方法返回一个占位视图(比如旋转的菊花)。

public View getVideoLoadingProgressView() {

return null;

}

// 接收当前页面的加载进度

public void onProgressChanged(WebView view, int newProgress) {

}

// 接收文档标题

public void onReceivedTitle(WebView view, String title) {

}

// 接收图标(favicon)

public void onReceivedIcon(WebView view, Bitmap icon) {

}

// Android中处理Touch Icon的方案

// http://droidyue.com/blog/2015/01/18/deal-with-touch-icon-in-android/index.html

public void onReceivedTouchIconUrl(WebView view, String url, boolean precomposed) {

}

// 通知应用当前页进入了全屏模式,此时应用必须显示一个包含网页内容的自定义View

public void onShowCustomView(View view, CustomViewCallback callback) {

}

// 通知应用当前页退出了全屏模式,此时应用必须隐藏之前显示的自定义View

public void onHideCustomView() {

}

// 显示一个alert对话框

public boolean onJsAlert(WebView view, String url, String message, JsResult result) {

return false;

}

// 显示一个confirm对话框

public boolean onJsConfirm(WebView view, String url, String message, JsResult result) {

return false;

}

// 显示一个prompt对话框

public boolean onJsPrompt(WebView view, String url, String message, String defaultValue, JsPromptResult result) {

return false;

}

// 显示一个对话框让用户选择是否离开当前页面

public boolean onJsBeforeUnload(WebView view, String url, String message, JsResult result) {

return false;

}

// 指定源的网页内容在没有设置权限状态下尝试使用地理位置API。

// 从API24开始,此方法只为安全的源(https)调用,非安全的源会被自动拒绝

public void onGeolocationPermissionsShowPrompt(String origin, GeolocationPermissions.Callback callback) {

}

// 当前一个调用 onGeolocationPermissionsShowPrompt() 取消时,隐藏相关的UI。

public void onGeolocationPermissionsHidePrompt() {

}

// 通知应用打开新窗口

public boolean onCreateWindow(WebView view, boolean isDialog, boolean isUserGesture, Message resultMsg) {

return false;

}

// 通知应用关闭窗口

public void onCloseWindow(WebView window) {

}

// 请求获取取焦点

public void onRequestFocus(WebView view) {

}

// 通知应用网页内容申请访问指定资源的权限(该权限未被授权或拒绝)

@TargetApi(Build.VERSION_CODES.LOLLIPOP)

public void onPermissionRequest(PermissionRequest request) {

request.deny();

}

// 通知应用权限的申请被取消,隐藏相关的UI。

@TargetApi(Build.VERSION_CODES.LOLLIPOP)

public void onPermissionRequestCanceled(PermissionRequest request) {

}

// 为’'显示文件选择器,返回false使用默认处理

@TargetApi(Build.VERSION_CODES.LOLLIPOP)

public boolean onShowFileChooser(WebView webView, ValueCallback<Uri[]> filePathCallback, FileChooserParams fileChooserParams) {

return false;

}

// 接收JavaScript控制台消息

public boolean onConsoleMessage(ConsoleMessage consoleMessage) {

return false;

}

回调顺序

页面加载回调顺序:

shouldOverrideUrlLoading

onProgressChanged[10]

shouldInterceptRequest

onProgressChanged[…]

onPageStarted

onProgressChanged[…]

onLoadResource

onProgressChanged[…]

onReceivedTitle/onPageCommitVisible

onProgressChanged[100]

onPageFinished

onReceivedIcon

资源加载回调:

shouldInterceptRequest() -> onLoadResource()

发生重定向时回调:

onPageStarted() -> shouldOverrideUrlLoading()

直接loadUrl的回调:

// 无重定向

onPageStarted() -> onPageFinished()

// 有重定向,shouldOverrideUrlLoading 返回 true 时 onPageFinished 仍会执行

onPageStarted() -> redirection -> … -> onPageFinished()

用户点击链接的回调:

// shouldOverrideUrlLoading 返回 true 时不执行onPageStarted/onPageFinished

shouldOverrideUrlLoading() -> …

// 无重定向

shouldOverrideUrlLoading() -> onPageStarted() -> onPageFinished()

// 有重定向

shouldOverrideUrlLoading() -> onPageStarted() -> redirection -> … -> onPageFinished()

// 有重定向(A->B->C)

shouldOverrideUrlLoading(A) -> onPageStarted(A) ->

onPageStarted(B) -> shouldOverrideUrlLoading(B) ->

onPageStarted© -> shouldOverrideUrlLoading© -> onPageFinished©

后退/前进/刷新 时回调:

onPageStarted() -> onPageFinished()

关于 window.location

假设从A页面跳转到B页面

如果页面B中直接输出 window.location=“http://example.com”,那页面B不会被加入回退栈,回退将直接回到A页

如果页面B加载完成后,比如用setTimeout延迟了,那页面B会被加入回退栈,当回退到页面A时会再执行跳转,这会导致回退功能看起来不正常,需要快速回退两次才能回到A页面

视口(viewport)

视口是一个为网页提供绘图区域的矩形。

你可以指定数个视口属性,比如尺寸和初始缩放系数(initial scale)。其中最重要的是视口宽度,它定义了网页水平方向的可用像素总数(可用的CSS像素数)。

多数 Android 上的网页浏览器(包括 Chrome)设置默认视口为一个大尺寸(被称为"wide viewport mode",宽约 980px)。

也有许多浏览器默认会尽可能缩小以显示完整的视口宽度(被称为"overview mode")。

// 是否支持viewport属性,默认值 false

// 页面通过<meta name="viewport" ... />自适应手机屏幕

// 当值为true且viewport标签不存在或未指定宽度时使用 wide viewport mode

settings.setUseWideViewPort(true);

// 是否使用overview mode加载页面,默认值 false

// 当页面宽度大于WebView宽度时,缩小使页面宽度等于WebView宽度

settings.setLoadWithOverviewMode(true);

viewport 语法

content="

height = [pixel_value | “device-height”] ,

width = [pixel_value | “device-width”] ,

initial-scale = float_value ,

minimum-scale = float_value ,

maximum-scale = float_value ,

user-scalable = [“yes” | “no”]

" />

指定视口宽度精确匹配设备屏幕宽度同时禁用了缩放

Example 通过WebView设置初始缩放(initial-scale)

// 设置初始缩放百分比

// 0表示依赖于setUseWideViewPort和setLoadWithOverviewMode

// 100表示不缩放

web.setInitialScale(0)

管理 Cookies

Cookie 是服务器发送到用户浏览器并保存在浏览器上的一块数据,它会在浏览器下一次发起请求时被携带并发送到服务器上。

可通过Cookie保存浏览信息来获得更轻松的在线体验,比如保持登录状态、记住偏好设置,并提供本地的相关内容。

会话Cookie 与 持久Cookie

会话cookie不需要指定Expires和Max-Age,浏览器关闭之后它会被自动删除。

持久cookie指定了Expires或Max-Age,会被存储到磁盘上,不会因浏览器而失效。

第一方Cookie 与 第三方Cookie

每个Cookie都有与之关联的域,与页面域一样的就是第一方Cookie,不一样的就是第三方Cookie。

// 设置接收第三方Cookie

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {

CookieManager.getInstance().setAcceptThirdPartyCookies(vWeb, true);

}

读取/写入/移除 Cookie

// 获取指定url关联的所有Cookie

// 返回值使用"Cookie"请求头格式:“name=value; name2=value2; name3=value3”

CookieManager.getInstance().getCookie(url);

// 为指定的url设置一个Cookie

// 参数value使用"Set-Cookie"响应头格式,参考:https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Headers/Set-Cookie

CookieManager.getInstance().setCookie(url, value);

// 移除指定url下的指定Cookie

CookieManager.getInstance().setCookie(url, cookieName + “=”);

webkit cookie 工具类

public class WebkitCookieUtil {

// 移除指定url关联的所有cookie

public static void remove(String url) {

CookieManager cm = CookieManager.getInstance();

for (String cookie : cm.getCookie(url).split("; ")) {

cm.setCookie[0](url, cookie.split(“=”) + “=”); }

flush();

}

// sessionOnly 为true表示移除所有会话cookie,否则移除所有cookie

public static void remove(boolean sessionOnly) {

CookieManager cm = CookieManager.getInstance();

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {

if (sessionOnly) {

cm.removeSessionCookies(null);

} else {

cm.removeAllCookies(null);

}

} else {

if (sessionOnly) {

cm.removeSessionCookie();

} else {

cm.removeAllCookie();

}

}

flush();

}

// 写入磁盘

public static void flush() {

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {

CookieManager.getInstance().flush();

} else {

CookieSyncManager.getInstance().sync();

}

}

}

同步系统Cookie 与 Webkit Cookie

// 将系统级Cookie(比如new URL(...).openConnection()的Cookie) 同步到 WebView

public class WebkitCookieHandler extends CookieHandler {

private static final String TAG = WebkitCookieHandler.class.getSimpleName();

private CookieManager wcm;

public WebkitCookieHandler() {

this.wcm = CookieManager.getInstance();

}

@Override

public void put(URI uri, Map<String, List> headers) throws IOException {

if ((uri == null) || (headers == null)) {

return;

}

String url = uri.toString();

for (String headerKey : headers.keySet()) {

if ((headerKey == null) || !(headerKey.equalsIgnoreCase(“set-cookie2”) || headerKey.equalsIgnoreCase(“set-cookie”))) {

continue;

}

for (String headerValue : headers.get(headerKey)) {

Log.e(TAG, headerKey + ": " + headerValue);

this.wcm.setCookie(url, headerValue);

}

}

}

@Override

public Map<String, List> get(URI uri, Map<String, List> headers) throws IOException {

if ((uri == null) || (headers == null)) {

throw new IllegalArgumentException(“Argument is null”);

}

String url = uri.toString();

String cookie = this.wcm.getCookie(url);

Log.e(TAG, "cookie: " + cookie);

if (cookie != null) {

return Collections.singletonMap(“Cookie”, Arrays.asList(cookie));

} else {

return Collections.emptyMap();

}

}

}

缓存(Cache)

设置缓存模式

WebSettings.LOAD_DEFAULT 根据cache-control决定是否从网络上取数据

WebSettings.LOAD_CACHE_ELSE_NETWORK 无网,离线加载,优先加载缓存(即使已经过期)

WebSettings.LOAD_NO_CACHE 仅从网络加载

WebSettings.LOAD_CACHE_ONLY 仅从缓存加载

// 网络正常时根据cache-control决定是否从网络上取数据

if (isNetworkConnected(mActivity)) {

settings.setCacheMode(WebSettings.LOAD_DEFAULT);

} else {

settings.setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK);

}

清除缓存

// 传入true表示同时内存与磁盘,false表示仅清除内存

文末

很多人在刚接触这个行业的时候或者是在遇到瓶颈期的时候,总会遇到一些问题,比如学了一段时间感觉没有方向感,不知道该从那里入手去学习,对此我整理了一些资料,需要的可以免费分享给大家

这里笔者分享一份自己收录整理上述技术体系图相关的几十套腾讯、头条、阿里、美团等公司19年的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。如有需要点击这里前往我的GitHub免费获取

【视频教程】

天道酬勤,只要你想,大厂offer并不是遥不可及!希望本篇文章能为你带来帮助,如果有问题,请在评论区留言。

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级安卓工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Android移动开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Android)
img

触这个行业的时候或者是在遇到瓶颈期的时候,总会遇到一些问题,比如学了一段时间感觉没有方向感,不知道该从那里入手去学习,对此我整理了一些资料,需要的可以免费分享给大家

这里笔者分享一份自己收录整理上述技术体系图相关的几十套腾讯、头条、阿里、美团等公司19年的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。如有需要点击这里前往我的GitHub免费获取

[外链图片转存中…(img-Nsm8cJy8-1710985816759)]

[外链图片转存中…(img-6HBDS8yA-1710985816760)]

【视频教程】

[外链图片转存中…(img-RfstF2iK-1710985816760)]

天道酬勤,只要你想,大厂offer并不是遥不可及!希望本篇文章能为你带来帮助,如果有问题,请在评论区留言。

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级安卓工程师,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Android移动开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
[外链图片转存中…(img-kBTE9xaE-1710985816761)]
[外链图片转存中…(img-pCK0KpWM-1710985816762)]
[外链图片转存中…(img-rOUqw3YH-1710985816762)]
[外链图片转存中…(img-L6KvrmLF-1710985816763)]

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注Android)
[外链图片转存中…(img-Dc9e5Q0Q-1710985816763)]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值