Webview详解(下)
第三阶段:性能优化
加载速度优化
缓存策略
缓存策略可以显著减少网络请求,提升页面加载速度。常用的缓存策略包括 HTTP 缓存和本地资源预加载。
1. HTTP 缓存
HTTP 缓存利用 HTTP 协议中的缓存机制(如 Cache-Control
、ETag
等)来缓存网络资源。
-
Android 实现:
WebSettings webSettings = webView.getSettings(); webSettings.setCacheMode(WebSettings.LOAD_DEFAULT); // 使用默认缓存策略
-
iOS 实现:
let configuration = WKWebViewConfiguration() configuration.websiteDataStore = WKWebsiteDataStore.default() let webView = WKWebView(frame: .zero, configuration: configuration)
2. 本地资源预加载
将静态资源(如 HTML、CSS、JS、图片)打包到应用中,直接从本地加载。
-
Android 实现:
webView.loadUrl("file:///android_asset/index.html"); // 从 assets 加载本地资源
-
iOS 实现:
if let filePath = Bundle.main.path(forResource: "index", ofType: "html") {let url = URL(fileURLWithPath: filePath)webView.loadFileURL(url, allowingReadAccessTo: url) }
懒加载与非关键资源延迟加载
懒加载和延迟加载可以减少页面初始加载时间,提升用户体验。
1. 图片懒加载
图片懒加载是指当图片进入可视区域时才加载。
- HTML 实现:
<img data-src="image.jpg" class="lazyload" /> <script>document.addEventListener("DOMContentLoaded", function() {var lazyloadImages = document.querySelectorAll(".lazyload");var lazyloadObserver = new IntersectionObserver(function(entries) {entries.forEach(function(entry) {if (entry.isIntersecting) {var img = entry.target;img.src = img.dataset.src;lazyloadObserver.unobserve(img);}});});lazyloadImages.forEach(function(img) {lazyloadObserver.observe(img);});}); </script>
2. 非关键资源延迟加载
将非关键资源(如广告、统计脚本)延迟加载,优先加载核心内容。
- HTML 实现:
<script>window.addEventListener("load", function() {setTimeout(function() {var script = document.createElement("script");script.src = "non-critical.js";document.body.appendChild(script);}, 3000); // 延迟 3 秒加载}); </script>
WebView 预初始化与复用
WebView 的初始化和销毁开销较大,通过预初始化和复用可以提升性能。
1. WebView 预初始化
在应用启动时预先初始化 WebView,避免页面加载时的延迟。
-
Android 实现:
public class WebViewManager {private static WebView webView;public static void init(Context context) {if (webView == null) {webView = new WebView(context);webView.getSettings().setJavaScriptEnabled(true);}}public static WebView getWebView() {return webView;} }// 在 Application 中预初始化 public class MyApplication extends Application {@Overridepublic void onCreate() {super.onCreate();WebViewManager.init(this);} }
-
iOS 实现:
class WebViewManager {static let shared = WebViewManager()private var webView: WKWebView?private init() {webView = WKWebView(frame: .zero)}func getWebView() -> WKWebView? {return webView} }// 在 AppDelegate 中预初始化 func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {_ = WebViewManager.shared.getWebView()return true }
2. WebView 复用
通过池化机制复用 WebView,减少频繁创建和销毁的开销。
-
Android 实现:
public class WebViewPool {private static final int POOL_SIZE = 3;private static List<WebView> webViewPool = new ArrayList<>();public static void init(Context context) {for (int i = 0; i < POOL_SIZE; i++) {WebView webView = new WebView(context);webView.getSettings().setJavaScriptEnabled(true);webViewPool.add(webView);}}public static WebView getWebView() {if (webViewPool.isEmpty()) {return new WebView(context);}return webViewPool.remove(0);}public static void releaseWebView(WebView webView) {if (webViewPool.size() < POOL_SIZE) {webViewPool.add(webView);} else {webView.destroy();}} }
-
iOS 实现:
class WebViewPool {static let shared = WebViewPool()private let poolSize = 3private var webViewPool: [WKWebView] = []private init() {for _ in 0..<poolSize {let webView = WKWebView(frame: .zero)webViewPool.append(webView)}}func getWebView() -> WKWebView {if webViewPool.isEmpty {return WKWebView(frame: .zero)}return webViewPool.removeFirst()}func releaseWebView(_ webView: WKWebView) {if webViewPool.count < poolSize {webViewPool.append(webView)}} }
渲染性能提升
硬件加速原理与限制
WebView 硬件加速的原理
-
GPU 渲染的优势:
- GPU 专为图形处理设计,能够高效处理复杂的图形操作(如 CSS 动画、Canvas 绘制、视频播放等)。
- 通过硬件加速,WebView 可以将网页内容的渲染任务交给 GPU,减少 CPU 的负担,从而提升渲染性能。
-
分层渲染:
- WebView 将网页内容划分为多个图层(Layer),每个图层独立渲染。
- GPU 负责将这些图层合成最终的显示画面。
-
硬件加速的流程:
- 内容绘制:WebView 将网页内容(如 HTML、CSS、JavaScript)解析为图形指令。
- 图层生成:将图形指令转换为 GPU 可处理的图层。
- GPU 渲染:GPU 对图层进行渲染,并将结果输出到屏幕。
WebView 硬件加速的实现
在 Android 中,WebView 的硬件加速默认是开启的,但可以通过以下方式手动控制:
1. 全局启用硬件加速
在 AndroidManifest.xml
中为应用启用硬件加速:
<applicationandroid:hardwareAccelerated="true"... >
</application>
2. 为特定 Activity 启用硬件加速
在 AndroidManifest.xml
中为特定 Activity 启用硬件加速:
<activityandroid:name=".WebViewActivity"android:hardwareAccelerated="true"... >
</activity>
3. 在代码中动态控制硬件加速
在代码中动态启用或禁用硬件加速:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {webView.setLayerType(View.LAYER_TYPE_HARDWARE, null); // 启用硬件加速// webView.setLayerType(View.LAYER_TYPE_SOFTWARE, null); // 禁用硬件加速
}
WebView 硬件加速的限制
尽管硬件加速可以显著提升性能,但在某些场景下可能会带来问题或限制。
1. 兼容性问题
- 低端设备:部分低端设备的 GPU 性能较差,启用硬件加速可能导致卡顿或崩溃。
- Android 版本:硬件加速在 Android 3.0(API 11)及以上版本才完全支持。
2. 渲染异常
- 图层合成问题:硬件加速可能导致某些复杂的 CSS 或 JavaScript 动画渲染异常。
- 文字模糊:在某些设备上,硬件加速可能导致文字渲染模糊。
3. 内存占用
- 图层缓存:硬件加速会为每个图层分配显存,可能导致内存占用增加。
- 内存泄漏:如果 WebView 未正确释放,硬件加速可能导致内存泄漏。
4. 性能瓶颈
- 过度绘制:如果网页内容过于复杂,硬件加速可能导致 GPU 负载过高,反而降低性能。
- 图层数量限制:GPU 对图层数量有限制,过多的图层可能导致渲染失败。
硬件加速的优化建议
-
按需启用硬件加速:
- 在需要高性能渲染的场景(如动画、视频播放)启用硬件加速,其他场景可以禁用。
-
减少图层数量:
- 优化网页内容,减少不必要的图层划分,降低 GPU 负载。
-
测试与兼容性:
- 在不同设备和 Android 版本上测试硬件加速的效果,确保兼容性。
-
监控性能:
- 使用工具(如 Android Profiler)监控 GPU 和内存使用情况,及时发现性能瓶颈。
-
动态切换渲染模式:
- 根据设备性能和场景需求,动态切换硬件加速和软件渲染模式。
示例代码:动态控制硬件加速
public class WebViewActivity extends AppCompatActivity {private WebView webView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_webview);webView = findViewById(R.id.webView);webView.getSettings().setJavaScriptEnabled(true);// 根据设备性能动态启用硬件加速if (isHighPerformanceDevice()) {webView.setLayerType(View.LAYER_TYPE_HARDWARE, null); // 启用硬件加速} else {webView.setLayerType(View.LAYER_TYPE_SOFTWARE, null); // 禁用硬件加速}webView.loadUrl("https://www.example.com");}private boolean isHighPerformanceDevice() {// 根据设备信息判断是否为高性能设备return Runtime.getRuntime().availableProcessors() > 4; // 例如,CPU 核心数大于 4}@Overrideprotected void onDestroy() {super.onDestroy();if (webView != null) {webView.destroy();webView = null;}}
}
总结
硬件加速 | 优点 | 限制 |
---|---|---|
提升性能 | 加速图形渲染,减少 CPU 负担 | 在低端设备上可能导致卡顿或崩溃 |
优化动画效果 | 支持复杂的 CSS 和 JavaScript 动画 | 可能导致渲染异常(如文字模糊、图层合成问题) |
减少 CPU 负载 | 将图形渲染任务交给 GPU | 增加内存占用,可能导致内存泄漏 |
兼容性问题 | 在 Android 3.0 及以上版本支持 | 在低版本 Android 上不可用 |
复杂CSS/JS动画优化技巧
1. 使用硬件加速
- 启用GPU加速:通过CSS属性
transform
和opacity
触发硬件加速,这些属性通常由GPU处理,性能更高。.element {transform: translateZ(0); /* 触发GPU加速 */ }
- 避免使用
top
,left
,margin
等属性:这些属性会导致重排(reflow),性能较差,优先使用transform
。
2. 优化CSS动画
- 使用
will-change
:告知浏览器元素即将发生变化,提前优化渲染。.element {will-change: transform, opacity; }
- 减少复杂选择器:避免使用嵌套过深或复杂的选择器,减少样式计算时间。
- 避免频繁重绘:减少
box-shadow
,border-radius
,gradient
等属性的使用,这些属性会导致重绘(repaint)。
3. 优化JavaScript动画
- 使用
requestAnimationFrame
:代替setTimeout
或setInterval
,确保动画与浏览器的刷新率同步。function animate() {// 动画逻辑requestAnimationFrame(animate); } requestAnimationFrame(animate);
- 减少DOM操作:频繁的DOM操作会导致重排和重绘,尽量缓存DOM元素或使用虚拟DOM技术。
- 使用 Web Workers:将复杂的计算任务移到Web Workers中,避免阻塞主线程。
4. 减少动画复杂度
- 简化动画效果:避免过多元素同时动画,或减少动画的持续时间。
- 合并动画:将多个动画合并为一个,减少渲染次数。
- 使用精灵图(Sprite Sheet):将多帧动画合并为一张图,通过
background-position
实现动画,减少HTTP请求和资源加载。
5. 优化资源加载
- 压缩CSS/JS文件:减少文件大小,加快加载速度。
- 延迟加载非关键资源:确保动画所需资源优先加载,非关键资源延迟加载。
- 使用CDN:加速静态资源的加载速度。
6. 降低分辨率或帧率
- 降低分辨率:对于高分辨率屏幕,适当降低动画元素的分辨率。
- 降低帧率:如果设备性能不足,可以适当降低动画帧率(如30fps)。
7. 测试与调试
- 使用开发者工具:通过Chrome DevTools等工具分析性能瓶颈,检查重排、重绘和帧率。
- 模拟低性能设备:在开发者工具中模拟低端设备,测试动画的流畅性。
8. 使用WebAssembly或Canvas
- WebAssembly:对于极其复杂的动画,可以考虑使用WebAssembly提高性能。
- Canvas:对于大量元素的动画,使用Canvas渲染比DOM操作更高效。
9. WebView特定优化
- 启用硬件加速:确保WebView的硬件加速已启用。
webView.setLayerType(View.LAYER_TYPE_HARDWARE, null);
- 优化WebView设置:禁用不必要的功能(如JavaScript接口)以提升性能。
- 使用WebView的缓存机制:减少重复加载资源的开销。
第四阶段:安全防护
WebView安全实践
常见漏洞
-
XSS(跨站脚本攻击)
- 原因:WebView加载的内容中包含了恶意脚本,攻击者可以通过注入脚本窃取用户数据或执行恶意操作。
- 防御:
- 对用户输入进行严格的过滤和转义。
- 使用
setDefaultTextEncodingName("UTF-8")
防止字符编码漏洞。 - 避免直接加载不可信的HTML内容。
-
任意代码执行
- 原因:WebView允许执行JavaScript代码,攻击者可能通过恶意代码获取设备权限或执行危险操作。
- 防御:
- 禁用JavaScript(除非必要):
webView.getSettings().setJavaScriptEnabled(false);
- 避免使用
addJavascriptInterface
,如果必须使用,确保对接口方法进行严格校验。
- 禁用JavaScript(除非必要):
-
协议漏洞
- 原因:WebView允许加载本地文件(
file://
协议)或其他自定义协议,可能导致敏感文件泄露或恶意代码执行。 - 防御:
- 禁用
file://
协议:webView.getSettings().setAllowFileAccess(false); webView.getSettings().setAllowFileAccessFromFileURLs(false); webView.getSettings().setAllowUniversalAccessFromFileURLs(false);
- 限制加载的协议范围,仅允许
https://
和http://
。
- 禁用
- 原因:WebView允许加载本地文件(
安全配置
-
禁用File协议
- 防止WebView加载本地文件,避免敏感数据泄露。
webView.getSettings().setAllowFileAccess(false);
- 防止WebView加载本地文件,避免敏感数据泄露。
-
限制JavaScript权限
- 禁用不必要的JavaScript功能,减少攻击面。
webView.getSettings().setJavaScriptEnabled(false);
- 禁用不必要的JavaScript功能,减少攻击面。
-
关闭其他危险配置
- 禁用自动加载图片、插件等:
webView.getSettings().setLoadsImagesAutomatically(false); webView.getSettings().setPluginState(WebSettings.PluginState.OFF);
- 禁用自动加载图片、插件等:
-
设置严格的内容安全策略(CSP)
- 通过HTTP头部或Meta标签限制资源加载:
<meta http-equiv="Content-Security-Policy" content="default-src 'self'; script-src 'self'">
- 通过HTTP头部或Meta标签限制资源加载:
HTTPS证书校验与中间人攻击防御
-
启用HTTPS
- 确保WebView仅加载HTTPS内容:
webView.getSettings().setMixedContentMode(WebSettings.MIXED_CONTENT_NEVER_ALLOW);
- 确保WebView仅加载HTTPS内容:
-
自定义证书校验
- 实现
WebViewClient
的onReceivedSslError
方法,严格校验证书:webView.setWebViewClient(new WebViewClient() {@Overridepublic void onReceivedSslError(WebView view, SslErrorHandler handler, SslError error) {// 严格处理SSL错误,避免忽略证书问题handler.cancel(); // 拒绝加载不安全的页面} });
- 实现
-
防止中间人攻击
- 证书锁定(Certificate Pinning):在客户端固定服务器的公钥或证书,防止中间人伪造证书。
- 使用OkHttp等库实现证书锁定。
- 示例:
String hostname = "example.com"; CertificatePinner certificatePinner = new CertificatePinner.Builder().add(hostname, "sha256/AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA=").build(); OkHttpClient client = new OkHttpClient.Builder().certificatePinner(certificatePinner).build();
- 证书锁定(Certificate Pinning):在客户端固定服务器的公钥或证书,防止中间人伪造证书。
-
启用HSTS(HTTP Strict Transport Security)
- 强制使用HTTPS,防止降级攻击。
- 服务器配置HSTS头部:
Strict-Transport-Security: max-age=31536000; includeSubDomains; preload
其他安全建议
-
定期更新WebView
- 使用最新版本的WebView,修复已知漏洞。
-
限制调试功能
- 发布版本中禁用WebView调试:
WebView.setWebContentsDebuggingEnabled(false);
- 发布版本中禁用WebView调试:
-
输入验证与输出编码
- 对所有用户输入进行验证,并对输出内容进行编码,防止XSS。
-
日志与监控
- 记录WebView的异常行为,及时发现潜在攻击。
内容安全策略
1. 内容安全策略(Content Security Policy, CSP)
CSP通过限制资源加载的来源,防止XSS、数据泄露和跨域攻击。
配置示例
Content-Security-Policy: default-src 'self'; script-src 'self' https://trusted.cdn.com; style-src 'self'; img-src 'self' data:; font-src 'self'; connect-src 'self'; frame-ancestors 'none';
常用指令
default-src 'self'
:默认仅允许加载同源资源。script-src 'self' https://trusted.cdn.com
:仅允许加载同源和指定CDN的脚本。style-src 'self'
:仅允许加载同源的样式表。img-src 'self' data:
:仅允许加载同源和Base64编码的图片。font-src 'self'
:仅允许加载同源的字体。connect-src 'self'
:仅允许同源的AJAX请求。frame-ancestors 'none'
:禁止页面被嵌入到iframe中,防止点击劫持。
Meta标签配置
如果无法通过HTTP头配置CSP,可以在HTML中使用Meta标签:
<meta http-equiv="Content-Security-Policy" content="default-src 'self'; script-src 'self'; style-src 'self';">
2. 防止点击劫持(X-Frame-Options)
通过 X-Frame-Options
头,防止页面被嵌入到iframe中,避免点击劫持攻击。
配置示例
X-Frame-Options: DENY
可选值
DENY
:禁止页面被嵌入到任何iframe中。SAMEORIGIN
:仅允许同源页面嵌入。ALLOW-FROM https://example.com
:仅允许指定域名嵌入。
3. 防止MIME类型嗅探(X-Content-Type-Options)
通过 X-Content-Type-Options
头,禁止浏览器对响应内容进行MIME类型嗅探,防止恶意文件执行。
配置示例
X-Content-Type-Options: nosniff
4. 防止跨域资源共享滥用(Cross-Origin Resource Sharing, CORS)
通过 Access-Control-Allow-Origin
头,限制跨域请求的来源,防止数据泄露。
配置示例
Access-Control-Allow-Origin: https://trusted-site.com
严格配置
- 仅允许特定域名访问:
Access-Control-Allow-Origin: https://trusted-site.com
- 禁止所有跨域请求:
Access-Control-Allow-Origin: null
5. 防止跨站脚本攻击(X-XSS-Protection)
通过 X-XSS-Protection
头,启用浏览器的XSS过滤器,防止反射型XSS攻击。
配置示例
X-XSS-Protection: 1; mode=block
可选值
1
:启用XSS过滤器。1; mode=block
:启用XSS过滤器,并阻止页面加载。0
:禁用XSS过滤器。
6. 防止数据泄露(Referrer-Policy)
通过 Referrer-Policy
头,控制Referrer信息的发送,防止敏感数据泄露。
配置示例
Referrer-Policy: no-referrer
常用值
no-referrer
:不发送Referrer信息。same-origin
:仅在同源请求中发送Referrer信息。strict-origin
:仅在HTTPS->HTTPS请求中发送Referrer信息。strict-origin-when-cross-origin
:跨域请求时仅发送源信息。
7. 防止缓存敏感数据(Cache-Control)
通过 Cache-Control
头,控制浏览器缓存行为,防止敏感数据被缓存。
配置示例
Cache-Control: no-store, no-cache, must-revalidate, max-age=0
常用指令
no-store
:禁止缓存任何内容。no-cache
:缓存内容但每次请求时验证。must-revalidate
:缓存内容但过期后必须重新验证。max-age=0
:缓存内容立即过期。
8. 防止跨域脚本攻击(Cross-Origin-Opener-Policy, COOP)
通过 Cross-Origin-Opener-Policy
头,防止跨域窗口访问,避免跨域脚本攻击。
配置示例
Cross-Origin-Opener-Policy: same-origin
可选值
same-origin
:仅允许同源窗口访问。same-origin-allow-popups
:允许同源窗口和弹出窗口访问。
9. 防止跨域嵌入攻击(Cross-Origin-Embedder-Policy, COEP)
通过 Cross-Origin-Embedder-Policy
头,限制跨域资源的嵌入,防止跨域攻击。
配置示例
Cross-Origin-Embedder-Policy: require-corp
可选值
require-corp
:仅允许加载同源或明确允许的跨域资源。
10. 防止信息泄露(Strict-Transport-Security, HSTS)
通过 Strict-Transport-Security
头,强制使用HTTPS,防止降级攻击和信息泄露。
配置示例
Strict-Transport-Security: max-age=31536000; includeSubDomains; preload
常用指令
max-age=31536000
:HSTS有效期为1年。includeSubDomains
:HSTS应用于所有子域名。preload
:将域名加入HSTS预加载列表。
第五阶段:实战与场景化
典型场景实践
1. H5与原生登录状态同步
场景需求
H5页面需要与原生应用共享登录状态,避免用户重复登录。
实现方案
-
通过JSBridge通信
- 原生应用将登录状态(如Token)通过JSBridge传递给H5。
- H5通过JSBridge获取登录状态并存储在LocalStorage或Cookie中。
- 示例:
// H5获取登录状态 window.JSBridge.getLoginStatus(function(status) {if (status.isLoggedIn) {localStorage.setItem('token', status.token);} });
-
通过URL参数传递
- 原生应用在加载H5页面时,将Token作为URL参数传递给H5。
- H5解析URL参数并存储Token。
- 示例:
// H5解析URL参数 const urlParams = new URLSearchParams(window.location.search); const token = urlParams.get('token'); if (token) {localStorage.setItem('token', token); }
-
通过Cookie共享
- 原生应用和H5页面使用相同的域名,通过Cookie共享登录状态。
- 原生应用将Token写入Cookie,H5页面读取Cookie。
-
通过WebView的JavaScript接口
- 原生应用通过
addJavascriptInterface
暴露登录状态接口。 - H5通过JavaScript调用接口获取登录状态。
- 示例:
// 原生代码 webView.addJavascriptInterface(new WebAppInterface(context), "Android");
// H5代码 const token = Android.getToken(); localStorage.setItem('token', token);
- 原生应用通过
2. 内嵌地图/支付/视频播放的混合方案
内嵌地图
-
使用高德/百度/Google地图的H5 API
- 在H5页面中直接使用地图的JavaScript API。
- 示例:
<script src="https://webapi.amap.com/maps?v=2.0&key=your_api_key"></script> <div id="map" style="width: 100%; height: 400px;"></div> <script>var map = new AMap.Map('map', {center: [116.397428, 39.90923],zoom: 13}); </script>
-
原生地图与H5交互
- 通过JSBridge调用原生地图组件,传递位置信息。
内嵌支付
-
调用原生支付SDK
- 通过JSBridge调用原生支付接口,如支付宝、微信支付。
- 示例:
window.JSBridge.startPayment({orderId: '123456',amount: '100.00',callback: function(result) {if (result.success) {alert('支付成功!');}} });
-
使用H5支付页面
- 在H5页面中直接调用支付网关,跳转到支付页面。
内嵌视频播放
-
使用HTML5视频标签
- 在H5页面中使用
<video>
标签播放视频。<video controls><source src="https://example.com/video.mp4" type="video/mp4"> </video>
- 在H5页面中使用
-
调用原生视频播放器
- 通过JSBridge调用原生视频播放器,提升性能和体验。
- 示例:
window.JSBridge.playVideo('https://example.com/video.mp4');
3. 渐进式Web应用(PWA)集成
PWA的核心特性
- Service Worker:实现离线缓存和资源预加载。
- Web App Manifest:定义应用的元数据,支持添加到主屏幕。
- Push Notifications:支持推送通知。
集成步骤
-
注册Service Worker
- 在H5页面中注册Service Worker,实现离线缓存。
if ('serviceWorker' in navigator) {navigator.serviceWorker.register('/service-worker.js').then(function(registration) {console.log('Service Worker 注册成功');}); }
- 在H5页面中注册Service Worker,实现离线缓存。
-
配置Web App Manifest
- 创建
manifest.json
文件,定义应用的名称、图标、主题色等。{"name": "My PWA","short_name": "PWA","start_url": "/index.html","display": "standalone","icons": [{"src": "/icon-192x192.png","sizes": "192x192","type": "image/png"}] }
- 在HTML中引入Manifest文件:
<link rel="manifest" href="/manifest.json">
- 创建
-
实现推送通知
- 使用Push API和Notification API实现推送通知。
Notification.requestPermission().then(function(permission) {if (permission === 'granted') {new Notification('Hello, World!');} });
- 使用Push API和Notification API实现推送通知。
-
与原生应用集成
- 在原生应用中加载PWA的入口页面,并通过JSBridge实现原生功能扩展。
调试与维护
1. Chrome DevTools远程调试WebView
Chrome DevTools 提供了一个强大的远程调试功能,可以用来调试移动设备上的WebView。以下是具体步骤:
-
启用WebView调试:
- 在Android应用代码中启用WebView调试功能:
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {WebView.setWebContentsDebuggingEnabled(true); }
- 在Android应用代码中启用WebView调试功能:
-
连接设备:
- 使用USB线将Android设备连接到开发电脑。
- 在设备的开发者选项中启用USB调试。
-
打开Chrome DevTools:
- 在Chrome浏览器中,打开
chrome://inspect
页面。 - 确保设备已经连接,并且WebView正在运行。
- 在
chrome://inspect
页面中,你会看到设备上正在运行的WebView实例,点击“inspect”打开DevTools进行调试。
- 在Chrome浏览器中,打开
-
调试:
- 使用DevTools进行各种调试操作,如查看DOM、调试JavaScript、监控网络请求等。
2. 性能监控与内存泄漏检测工具
性能监控和内存泄漏检测是确保应用稳定性和流畅性的重要手段。以下是一些常用的工具和方法:
-
Chrome DevTools Performance Panel:
- 使用Chrome DevTools的Performance面板记录和分析WebView的性能,包括CPU使用率、内存占用、帧率等。
- 可以捕获一段时间内的性能数据,并生成详细的报告,帮助分析性能瓶颈。
-
Memory Panel:
- 使用Memory面板进行内存分析,可以拍摄堆快照,查看内存分配情况,检测内存泄漏。
- 可以通过比较不同时间点的堆快照,找出内存泄漏的对象。
-
Lighthouse:
- Lighthouse是Chrome DevTools内置的性能分析工具,可以生成性能、可访问性、最佳实践等方面的报告,帮助优化WebView性能。
-
Android Profiler:
- 对于原生应用中的WebView,可以使用Android Studio的Profiler工具进行性能监控和内存分析。
- Profiler可以实时监控CPU、内存、网络等资源的使用情况。
-
LeakCanary:
- LeakCanary是一个专门用于检测Android应用中内存泄漏的工具。
- 它可以自动检测内存泄漏,并提供详细的泄漏路径信息。
3. 线上异常监控(JS错误捕获)
线上异常监控是确保应用稳定性的重要环节,特别是在生产环境中。以下是一些常用的方法和工具:
-
window.onerror:
- 使用
window.onerror
全局错误处理函数捕获JavaScript运行时错误:window.onerror = function(message, source, lineno, colno, error) {console.error('Error:', message, 'at', source, 'line', lineno);// 可以将错误信息发送到服务器 };
- 使用
-
try-catch:
- 在关键代码块中使用
try-catch
捕获异常:try {// 可能出错的代码 } catch (error) {console.error('Caught error:', error);// 处理错误或上报 }
- 在关键代码块中使用
-
第三方监控工具:
- Sentry:一个开源的错误监控工具,支持JavaScript、Android、iOS等多种平台,可以捕获和上报异常。
- Bugsnag:另一个流行的错误监控工具,提供详细的错误报告和分析功能。
- TrackJS:专门用于JavaScript错误监控的工具,提供实时错误跟踪和分析。
-
自定义错误上报:
- 可以将捕获的错误信息通过Ajax请求发送到服务器,存储并进行分析:
function reportError(error) {var url = '/logError';var data = {message: error.message,stack: error.stack,timestamp: new Date().toISOString()};fetch(url, {method: 'POST',headers: {'Content-Type': 'application/json'},body: JSON.stringify(data)}); }
- 可以将捕获的错误信息通过Ajax请求发送到服务器,存储并进行分析:
相关文章:
Webview详解(下)
第三阶段:性能优化 加载速度优化 缓存策略 缓存策略可以显著减少网络请求,提升页面加载速度。常用的缓存策略包括 HTTP 缓存和本地资源预加载。 1. HTTP 缓存 HTTP 缓存利用 HTTP 协议中的缓存机制(如 Cache-Control、ETag 等࿰…...
scss基础用法
SCSS(Sassy CSS)是Sass的增强版本,作为CSS的预处理器,它提供了多种功能来提高代码的可维护性和效率。以下是SCSS的基础用法: 变量(Variables) 用于存储常用的值,如颜色、字体大小等。…...
知能行每日综测
题目1 自己的做法 答案 题目2 自己的 答案 题目3 注意:这道做错了,你们可以看看我哪里错了 题目4 我的 答案 题目5 没思路,不会做 已更改 题目6 答案 第七题 我的 不会 现在补综测最后一个...
c++ vs和g++下的string结构
话不多说进入正题.注:下述结构是在32位平台下进行验证,32位平台下指针占4个字节. vs下string的结构 string总共占28个字节,内部结构稍微复杂一点,先是有一个联合体,联合体用来定义 string中字符串的存储空间:(联合体的…...
海量数据处理
1.海量数据处理问题 给两个文件,分别有100亿个query,只有1G内存,如何找到两个文件交集? 解决方案一: 可以先用布隆过滤器,一个文件的query放进布隆过滤器,另一个文件依次查找,在的…...
洛谷题单1-P5706 【深基2.例8】再分肥宅水-python-流程图重构
题目描述 现在有 t t t 毫升肥宅快乐水,要均分给 n n n 名同学。每名同学需要 2 2 2 个杯子。现在想知道每名同学可以获得多少毫升饮料(严格精确到小数点后 3 3 3 位),以及一共需要多少个杯子。 输入格式 输入一个实数 t …...
【HarmonyOS 5】初学者如何高效的学习鸿蒙?
【HarmonyOS 5】初学者如何高效的学习鸿蒙? 一、前言 在全球科技格局风云变幻的当下,谷歌安卓系统的管控逐步收紧,加之国际形势愈发复杂,打造中国人自主的操作系统,已成为时代发展的必然要求,这不仅是突破…...
Java NIO之FileChannel 详解
关键点说明 文件打开选项: StandardOpenOption.CREATE - 文件不存在时创建 StandardOpenOption.READ/WRITE - 读写权限 StandardOpenOption.APPEND - 追加模式 StandardOpenOption.TRUNCATE_EXISTING - 清空已存在文件 缓冲区操作: ByteBuffer.wrap…...
数据可视化(matplotlib)-------图表样式美化
目录 一、图表样式概述 (一)、默认图表样式 (二)、图表样式修改 1、局部修改 2、全局修改 二、使用颜色 (一)、使用基础颜色 1、单词缩写或单词表示的颜色 2、十六进制/HTML模式表示的颜色 3、RGB…...
Go 语言中,关于客户端初始化的最佳实践
在 Go 语言中,关于客户端初始化的最佳实践确实需要注意以下几点: 全局单例模式是推荐做法,尤其对于需要保持长连接或需要复用资源的客户端(如数据库、Redis、HTTP 客户端等)并发安全是必须保证的,需要确保…...
MyBatis的第一天笔记
1. MyBatis 概述 1.1 什么是框架 框架是对通用代码的封装,提前写好了一堆接口和类,可以直接引入使用框架一般以jar包形式存在Java常用框架:SSM三大框架(Spring SpringMVC MyBatis)、SpringBoot、SpringCloud等 1.…...
区块链赋能,为木材货场 “智” 造未来
区块链赋能,为木材货场 “智” 造未来 在当今数字化浪潮席卷的时代,软件开发公司不断探索创新,为各行业带来高效、智能的解决方案。今天,让我们聚焦于一家软件开发公司的杰出成果 —— 区块链木材货场服务平台,深入了…...
IvorySQL:兼容Oracle数据库的开源PostgreSQL
今天给大家介绍一款基于 PostgreSQL 开发、兼容 Oracle 数据库的国产开源关系型数据库管理系统:IvorySQL。 IvorySQL 由商瀚高软件提供支持,主要的功能特性包括: 完全兼容 PostgreSQL:IvorySQL 基于 PostgreSQL 内核开发…...
Python 序列构成的数组(切片)
切片 在 Python 里,像列表(list)、元组(tuple)和字符串(str)这类 序列类型都支持切片操作,但是实际上切片操作比人们所想象的要强大 很多。 这一节主要讨论的是这些高级切片形式的…...
Pre-flash和Main flash
在相机拍照过程中,Pre-flash(预闪光) 和 Main flash(主闪光) 是常见的两种闪光灯使用模式,通常用于提高低光环境下的拍摄质量,尤其在自动曝光(AE)和自动对焦(…...
【区块链安全 | 第十篇】智能合约概述
部分内容与前文互补。 文章目录 一个简单的智能合约子货币(Subcurrency)示例区块链基础交易区块预编译合约 一个简单的智能合约 我们从一个基础示例开始,该示例用于设置变量的值,并允许其他合约访问它。 // SPDX-License-Identi…...
判断质数及其优化方法
判断质数(素数)及其优化方法 质数是指 大于1的自然数,且 只有1和它本身两个正约数。以下是几种判断方法及其优化策略。 目录 基础方法(试除法)优化1:仅检查到√n优化2:跳过偶数优化3ÿ…...
【源码阅读/Vue Flask前后端】简历数据查询功能
目录 一、Flask后端部分modelServiceroute 二、Vue前端部分index.js main.vue功能界面templatescriptstyle 一般就是三个层面,model层面用来建立数据库的字段,service用来对model进行操作,写一些数据库操作的代码,route就是具体的…...
R语言对偏态换数据进行转换(对数、平方根、立方根)
我们进行研究的时候经常会遇见偏态数据,数据转换是统计分析和数据预处理中的一项基本技术。使用 R 时,了解如何正确转换数据有助于满足统计假设、标准化分布并提高分析的准确性。在 R 中实现和可视化最常见的数据转换:对数、平方根和立方根转…...
链表(C++)
这是本人第二次学习链表,第一次学习链表是在大一上的C语言课上,首次接触,感到有些难;第二次是在大一下学习数据结构时(就是这次),使用C再次理解链表。同时,这也是开启数据结构学习写…...
算法-前缀和与差分
一、前缀和(Prefix Sum) 1. 核心思想 前缀和是一种预处理数组的方法,通过预先计算并存储数组的前缀和,使得后续的区间和查询可以在**O(1)**时间内完成。 2. 定义 给定数组 nums,前缀和数组 prefixSum 的每个元素 p…...
网关接口超时?用Java实现接口快速返回,后台继续执行的方法
网关接口超时?用Java实现接口快速返回,后台继续执行的方法 在开发过程中,我们经常会遇到网关接口由于超时限制而导致请求失败的情况。然而,有些接口本身就需要较长时间来执行任务,这时我们不能简单地增加超时时间&…...
HTTP---基础知识
天天开心!!! 文章目录 一、HTTP基本概念1. 什么是HTTP,又有什么用?2. 一次HTTP请求的过程3.HTTP的协议头4.POST和GET的区别5. HTTP状态码6.HTTP的优缺点 二、HTTP的版本演进1.各个版本的应用场景2、注意要点 三、HTTP与…...
python基础学习三(元组及字符串的使用)
文章目录 元组什么是元组元组的创建方式为什么要将元组设计成不可变序列元组的遍历集合集合的相关操作集合操作集合的数学操作集合生成式列表,字典,元组,集合总结 字符串字符串的驻留机制判断字符串的操作方法字符串的比较操作字符串的切片操…...
c#winform,倒鸭子字幕效果,typemonkey字幕效果,抖音瀑布流字幕效果
不废话 直接上效果图 C# winform 开发抖音的瀑布流字幕。 也是typemonkey插件字幕效果 或者咱再网上常说的倒鸭子字幕效果 主要功能 1,软件可以自定义添加字幕内容 2,软件可以添加字幕显示的时间区间 3,可以自定义字幕颜色,可以随…...
1、C51单片机(STC8G2K64S4)串口实验
一、串口1接线图 1、下面是单片机外接电路图,P30,P31分别用于RXD和TXD功能引脚 2、我们来查看单片机手册 串口1需要设置的寄存器 串口1的功能脚配置选择位,看电路图选择的是P3.0,P3.1。 3、串口1:SCON控制寄存器 设置为0x50:0101 0000。&a…...
ue材质学习感想总结笔记
2025 - 3 - 27 1.1 加法 对TexCoord上的每一个像素加上一个值,如果加上0.1,0.1, 那么左上角原来0,0的位置变成了0.1,0.1 右上角就变成了1.1,1.1,那么原来0,0的位置就去到了左上角左上边,所以图像往左上偏移。 总而言…...
MFC TRACE 宏的使用说明
书籍:《Visual C 2017从入门到精通》的2.7 字符串 环境:visual studio 2022 内容:几个字符串类型->(将单字节char*转换为宽字节wchar_t *)(将宽字节wchar_t* 转换为单字节char *) 问题&am…...
latex笔记
1、基本结构 \documentclass[a4paper, 12pt]{article} %文档类型 \begin{document}\title{My First Document}\author{My Name}\date{\today}\maketitleA sentence of text. \end{document}2、带有章、节、小节的结构 \documentclass[a4paper, 12pt]{article}\begin{document…...
Unity编辑器功能及拓展(3) —[Attribute]特性
在 Unity 中,[Attribute]格式的特性是用于扩展编辑器功能、控制序列化行为和调整 Inspector 显示,进行编辑器拓展的核心工具。 一.基础编辑器拓展 1.基础序列化控制 1.[SerializeField] 强制显示私有变量到Inspector 2.[HideInInspector] 隐藏该字段在Inspect…...
Rust基础语法
以下是 Rust 语言基础语法的核心要点,结合与 JavaScript 的对比,帮助前端开发者快速掌握核心概念: 一、变量与常量 1. 变量声明 Rust:变量默认不可变,需用 mut 显式声明可变性。let x 5; // 不可变变量 le…...
<tauri><rust><GUI>基于rust和tauri,实现一个大寰电爪PGHL(串口设备)定制化控制程序
前言 本文是基于rust和tauri,由于tauri是前、后端结合的GUI框架,既可以直接生成包含前端代码的文件,也可以在已有的前端项目上集成tauri框架,将前端页面化为桌面GUI。 环境配置 系统:windows 10平台:visual studio code语言:rust、javascript库:tauri2.0概述 本文是…...
Sentinel 相关知识点
Sentinel 实现原理? Sentinel 是面向分布式服务架构的流量控制组件,主要以流量为切入点,从限流、流量整形、熔断降级、系统负载保护等多个维度来帮助开发者保障微服务的稳定性。以下是 Sentinel 的实现原理: 核心概念 资源&…...
DFS飞机降落
问题描述 NN 架飞机准备降落到某个只有一条跑道的机场。其中第 ii 架飞机在 TiTi 时刻到达机场上空,到达时它的剩余油料还可以继续盘旋 DiDi 个单位时间,即它最早可以于 TiTi 时刻开始降落,最晚可以于 TiDiTiDi 时刻开始降落。降落…...
SpringCould微服务架构之Docker(5)
Docker的基本操作: 镜像相关命令: 1.镜像名称一般分两部分组成:[repository]:[tag]。 2. 在没有指定tag时,默认是latest,代表着最新版本的镜像。 镜像命令的案例: 镜像操作常用的命令: dock…...
音乐webpack(通杀webpack-1)
本文章中所有内容仅供学习交流使用,不用于其他任何目的,不提供完整代码,抓包内容、敏感网址、数据接口等均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关! 本文章未经许可…...
数据结构与算法——顺序表之手撕OJ题
文章目录 一、前言二、拿捏OJ题2.1移除元素2.2删除有序数组中的重复项2.3合并两个有序数组 三、总结 一、前言 Do you study today?up在上一次已经讲解完毕了有关顺序表的所有知识,不知道大家是否已经沉淀完毕了呢?有一句老话说得好啊——光看不练假把…...
减少采样空间方法 变成后验概率
又 因为后验概率很难计算 --所以通过引入变分分布来近似 后验概率分布 同时 引入 kl散度来度量 近似的效果好不好 什么是kl散度 kl散度带变分: 第一个问题 :积分变期望 问题二:贝叶斯公式 第三个问题:为啥可以独立出来 因为相比…...
如何使用K8S快速部署测试环境
目录 一、Windows 系统使用 Rancher Desktop 二、Linux系统 集群使用 Ansible 一键部署 三、Linux系统使用 kubeadm 快速搭建单节点集群 四、Kubernetes (K8S) 快速部署测试环境 4.1 准备 K8S 集群 4.2部署测试应用 4.3访问测试服务 4.4持久化存储(可选&…...
GAMES101-现代计算机图形学入门(Animation/simulation)
目录 一些科普Keyframe AnimatorPhysical Simulation质点弹簧系统 Mass Spring Rope粒子系统运动学 Forward Kinematics逆运动学Inverse KinematicsRiggingMotion Capture 第二次课 cont.Single Particle Simulation流体模拟 Fluid Simulation GitHub主页:https://g…...
2两数相加解题记录
哎呀,以为这道题也不用写题解的……结果还是有坑没跳出来。 最开始想法先计算总和再求出链表 func addTwoNumbers(l1 *ListNode, l2 *ListNode) *ListNode {// 先算出这个值。测试用例会int类型溢出total : 0wei : 1for l1!nil && l2!nil {total (l1.Vall…...
uniapp 获取dom信息(封装获取元素信息工具函数)
在uniapp开发中,需要获取到dom的信息,需要用到uniapp的指定方式 uni.createSelectorQuery(),但是每次需要用到的时候都需要很长一段的繁琐代码,本篇文章将呈现获取dom信息方法封装,话不多说,上菜࿱…...
Mybatis_Plus中常用的IService方法
查询 方法名 查询记录总数 /*** 查询总记录数** see Wrappers#emptyWrapper()*/default long count() {return count(Wrappers.emptyWrapper());} 方法实现 Testpublic void testGetCount(){long count userService.count();System.out.println("总记录数:&…...
【华为OD技术面试真题 - 技术面】- Java面试题(16)
华为OD面试真题精选 专栏:华为OD面试真题精选 目录: 2024华为OD面试手撕代码真题目录以及八股文真题目录 线程创建的方式 1. 通过继承Thread类 创建一个自定义线程类,继承Java中的Thread类,并重写run()方法。然后通过调用start()方法来启动线程。 示例代码: // 继承Th…...
React(六)React过渡动画-CSS编写方式
React过渡动画 react-transition-group介绍 在开发中,我们想要给一个组件的显示和消失添加某种过渡动画,提高用户体验→可通过react-transition-group实现。React曾为开发者提供过动画插件 react-addons-css-transition-group,后由社区维护…...
计算机视觉初步(环境搭建)
1.anaconda 建议安装在D盘,官网正常安装即可,一般可以安装windows版本 安装成功后,可以在电脑应用里找到: 2.创建虚拟环境 打开anaconda prompt, 可以用conda env list 查看现有的环境,一般打开默认bas…...
JAVA反序列化深入学习(九):CommonsCollections7与CC链总结
CC7 依旧是寻找 LazyMap 的触发点 CC6使用了 HashSet而CC6使用了 Hashtable JAVA环境 java version "1.8.0_74" Java(TM) SE Runtime Environment (build 1.8.0_74-b02) Java HotSpot(TM) 64-Bit Server VM (build 25.74-b02, mixed mode) 依赖版本 Apache Commons …...
软考《信息系统运行管理员》- 6.1 信息系统安全概述
信息系统安全的概念 信息系统安全是指保障计算机及其相关设备、设施(含网络)的安全,运行环境的安全, 信息的安全,实现信息系统的正常运行。 信息系统安全包括实体安全、运行安全、信息安全和 人员安全等几个部分。 影响信息系统安全的因素…...
MDK中结构体的对齐、位域、配合联合体等用法说明
测试环境:STM32H7R3MDK 5.39AC5 注:PC、PowerPC等环境不适用本文。 一.字节对齐 一般采用自然对齐(默认方式),提高数据存取速度。 采用1字节对齐,变量在内存中无空隙,紧密存储,节省存…...
C++实现布隆过滤器
1.布隆过滤器概念 位图虽然能高效且节省存储数据,但是类型必须是整型,不能存储其它类型。布隆过滤器是由布隆提出的一中紧凑型,比较巧妙的概率型数据结构,特点是高效的插入和查询,可以得知什么是一定不存在或者可能存…...