百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程文章 > 正文

Android WebView 详解 2 android webview ua

qiyuwang 2024-11-07 13:11 18 浏览 0 评论

[接上文]

// 多数情况下,可通过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) {

}

// <video /> 控件在未播放时,会展示为一张海报图,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) {

}

// 为'<input type="file" />'显示文件选择器,返回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()

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

onPageStarted() -> onPageFinished()

关于 window.location

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

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

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

视口(viewport)

https://developer.android.com/guide/webapps/targeting.html

https://developer.mozilla.org/en-US/docs/Mozilla/Mobile/Viewport_meta_tag

https://developer.mozilla.org/zh-CN/docs/Web/CSS/@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 语法

<meta name="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"]

" />

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

<head>

<title>Example</title>

<meta name="viewport" content="width=device-width, user-scalable=no" />

</head>

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

// 设置初始缩放百分比

// 0表示依赖于setUseWideViewPort和setLoadWithOverviewMode

// 100表示不缩放

web.setInitialScale(0)

管理 Cookies

https://developer.mozilla.org/zh-CN/docs/Web/HTTP/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(url, cookie.split("=")[0] + "=");

}

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<String>> 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<String>> get(URI uri, Map<String, List<String>> 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表示仅清除内存

// 由于内核缓存是全局的因此这个方法不仅仅针对webview而是针对整个应用程序

web.clearCache(true);

预加载(Preload)

一个简单的预加载示例(shouldInterceptRequest)

点击 assets/demo.xml 里的链接”hello”时会加载本地的 assets/hello.html

assets/demo.xml

<html>

<body>

<a href="http://demo.com/assets/hello.html">hello</a>

</body>

</html>

assets/hello.html

<html>

<body>

hello world!

</body>

</html>

重载 shouldInterceptRequest

@Override

public WebResourceResponse shouldInterceptRequest(WebView view, String url) {

return preload("assets/", url);

}

WebResourceResponse preload(String path, String url) {

if (!url.contains(path)) {

return null;

}

String local = url.replaceFirst("^http.*" + path, "");

try {

InputStream is = getApplicationContext().getAssets().open(local);

String ext = MimeTypeMap.getFileExtensionFromUrl(local);

String mimeType = MimeTypeMap.getSingleton().getMimeTypeFromExtension(ext);

return new WebResourceResponse(mimeType, "UTF-8", is);

} catch (Exception e) {

e.printStackTrace();

return null;

}

}

与Javascript交互

启用Javascript

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

settings.setJavaScriptEnabled(true);

注入对象到Javascript

// 注入对象'jsobj',在网页中通过`jsobj.say(...)`调用

web.addJavascriptInterface(new JSObject(), "jsobj")

在API17后支持白名单,只有添加了@JavascriptInterface注解的方法才会注入JS

public class JSObject {

@JavascriptInterface

public void say(String words) {

// todo

}

}

移除已注入Javascript的对象

1

web.removeJavascriptInterface("jsobj")

执行JS表达式

// 弹出提示框

web.loadUrl("javascript:alert('hello')");

// 调用注入的jsobj.say方法

web.loadUrl("javascript:jsobj.say('hello')");

在API19后可异步执行JS表达式,并通过回调返回值

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

vWeb.evaluateJavascript("111+222", new ValueCallback<String>() {

@Override

public void onReceiveValue(String value) {

// value => "333"

}

});

}

地理位置(Geolocation)

https://developer.mozilla.org/zh-CN/docs/Web/API/Geolocation/Using_geolocation

需要以下权限

<uses-permission android:name="android.permission.INTERNET"/>

<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>

<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>

默认可用

settings.setGeolocationEnabled(true);

当H5调用地理位置API时,会先通过WebChromeClient.onGeolocationPermissionsShowPrompt申请授权

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

@Override

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

boolean allow = true; // 是否允许origin使用定位API

boolean retain = false; // 内核是否记住这次制授权

callback.invoke(origin, true, false);

}

// 之前调用 onGeolocationPermissionsShowPrompt() 申请的授权被取消时,隐藏相关的UI。

@Override

public void onGeolocationPermissionsHidePrompt() {

}

注:从API24开始,仅支持安全源(https)的请求,非安全源的请求将自动拒绝且不调用 onGeolocationPermissionsShowPrompt 与 onGeolocationPermissionsHidePrompt

弹框(alert/confirm/prompt/onbeforeunload)

在javascript中使用 alert/confirm/prompt 会弹出对话框,可通过重载 WebChromeClient 的下列方法控制弹框的交互,比如替换系统默认的对话框或屏蔽这些对话框

@Override

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

// 这里处理交互逻辑

// result.cancel(); 表示用户取消了操作(点击了取消按钮)

// result.confirm(); 表示用户确认了操作(点击了确认按钮)

// ...

// 返回true表示自已处理,返回false表示由系统处理

return false;

}

@Override

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

return false;

}

@Override

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

return false;

}

@Override

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

return false;

}

全屏(Fullscreen)

Fullscreen API

https://developer.mozilla.org/zh-CN/docs/DOM/Using_fullscreen_mode

当H5请求全屏时,会回调 WebChromeClient.onShowCustomView 方法

当H5退出全屏时,会回调 WebChromeClient.onHideCustomView 方法

1.manifest

自己处理屏幕尺寸方向的变化(切换屏幕方向时不重建activity)

WebView播放视频需要开启硬件加速

<activity

android:name=".WebViewActivity"

android:configChanges="orientation|screenSize"

android:hardwareAccelerated="true"

android:screenOrientation="portrait" />

2.页面布局

<LinearLayout

android:layout_width="match_parent"

android:layout_height="match_parent"

android:orientation="vertical">

<android.support.v7.widget.Toolbar

android:id="@+id/toolbar"

style="@style/Toolbar.Back"/>

<FrameLayout

android:layout_width="match_parent"

android:layout_height="match_parent">

<WebView

android:id="@+id/web"

android:layout_width="match_parent"

android:layout_height="match_parent"/>

...

</FrameLayout>

</LinearLayout>

3.处理全屏回调

CustomViewCallback mCallback;

View vCustom;

@Override

public void onShowCustomView(View view, CustomViewCallback callback) {

setFullscreen(true);

vCustom = view;

mCallback = callback;

if (vCustom != null) {

ViewGroup parent = (ViewGroup) vWeb.getParent();

parent.addView(vCustom);

}

}

@Override

public void onHideCustomView() {

setFullscreen(false);

if (vCustom != null) {

ViewGroup parent = (ViewGroup) vWeb.getParent();

parent.removeView(vCustom);

vCustom = null;

}

if (mCallback != null) {

mCallback.onCustomViewHidden();

mCallback = null;

}

}

4.设置全屏,切换屏幕方向

void setFullscreen(boolean fullscreen) {

if (fullscreen) {

getWindow().clearFlags(WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN);

getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);

vToolbar.setVisibility(View.GONE);

vWeb.setVisibility(View.GONE);

} else {

getWindow().clearFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN);

getWindow().setFlags(WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN, WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN);

vToolbar.setVisibility(View.VISIBLE);

vWeb.setVisibility(View.VISIBLE);

}

if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_PORTRAIT) {

setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);

} else {

setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);

}

}

内存泄漏

直接 new WebView 并传入 application context 代替在 XML 里面声明以防止 activity 引用被滥用,能解决90+%的 WebView 内存泄漏。

vWeb = new WebView(getContext().getApplicationContext());

container.addView(vWeb);

销毁 WebView

if (vWeb != null) {

vWeb.setWebViewClient(null);

vWeb.setWebChromeClient(null);

vWeb.loadDataWithBaseURL(null, "", "text/html", "utf-8", null);

vWeb.clearHistory();

((ViewGroup) vWeb.getParent()).removeView(vWeb);

vWeb.destroy();

vWeb = null;

}

相关推荐

# 安装打开 ubuntu-22.04.3-LTS 报错 解决方案

#安装打开ubuntu-22.04.3-LTS报错解决方案WslRegisterDistributionfailedwitherror:0x800701bcError:0x80070...

利用阿里云镜像在ubuntu上安装Docker

简介:...

如何将Ubuntu Kylin(优麒麟)19.10系统升级到20.04版本

UbuntuKylin系统使用一段时间后,有新的版本发布,如何将现有的UbuntuKylin系统升级到最新版本?可以通过下面的方法进行升级。1.先查看相关的UbuntuKylin系统版本情况。使...

Ubuntu 16.10内部代号确认为Yakkety Yak

在正式宣布Ubuntu16.04LTS(XenialXerus)的当天,Canonical创始人MarkShuttleworth还非常开心的在个人微博上宣布Ubuntu下个版本16.10的内...

如何在win11的wsl上装ubuntu(怎么在windows上安装ubuntu)

在Windows11的WSL(WindowsSubsystemforLinux)上安装Ubuntu非常简单。以下是详细的步骤:---...

Win11学院:如何在Windows 11上使用WSL安装Ubuntu

IT之家2月18日消息,科技媒体pureinfotech昨日(2月17日)发布博文,介绍了3中简便的方法,让你轻松在Windows11系统中,使用WindowsSubs...

如何查看Linux的IP地址(如何查看Linux的ip地址)

本头条号每天坚持更新原创干货技术文章,欢迎关注本头条号"Linux学习教程",公众号名称“Linux入门学习教程"。...

怎么看电脑系统?(怎么看电脑系统配置)

要查看电脑的操作系统信息,可以按照以下步骤操作,根据不同的操作系统选择对应的方法:一、Windows系统通过系统属性查看右键点击桌面上的“此电脑”(或“我的电脑”)图标,选择“属性”。在打开的...

如何查询 Linux 内核版本?这些命令一定要会!

Linux内核是操作系统的核心,负责管理硬件资源、调度进程、处理系统调用等关键任务。不同的内核版本可能支持不同的硬件特性、提供新的功能,或者修复了已知的安全漏洞。以下是查询内核版本的几个常见场景:...

深度剖析:Linux下查看系统版本与CPU架构

在Linux系统管理、维护以及软件部署的过程中,精准掌握系统版本和CPU架构是极为关键的基础操作。这些信息不仅有助于我们深入了解系统特性、判断软件兼容性,还能为后续的软件安装、性能优化提供重要依据。接...

504 错误代码解析与应对策略(504错误咋解决)

在互联网的使用过程中,用户偶尔会遭遇各种错误提示,其中504错误代码是较为常见的一种。504错误并非意味着网站被屏蔽,它实际上是指服务器在规定时间内未能从上游服务器获取响应,专业术语称为“Ga...

猎聘APP和官网崩了?回应:正对部分职位整改,临时域名可登录

10月12日,有网友反映猎聘网无法打开,猎聘APP无法登录。截至10月14日,仍有网友不断向猎聘官方微博下反映该情况,而猎聘官方微博未发布相关情况说明,只是在微博内对反映该情况的用户进行回复,“抱歉,...

域名解析的原理是什么?域名解析的流程是怎样的?

域名解析是网站正常运行的关键因素,因此网站管理者了解域名解析的原理和流程对于做好域名管理、解决常见解析问题,保障网站的正常运转十分必要。那么域名解析的原理是什么?域名解析的流程是怎样的?接下来,中科三...

Linux无法解析域名的解决办法(linux 不能解析域名)

如果由于误操作,删除了系统原有的dhcp相关设置就无法正常解析域名。  此时,需要手动修改配置文件:  /etc/resolv.conf  将域名解析服务器手动添加到配置文件中  该文件是DNS域名解...

域名劫持是什么?(域名劫持是什么)

域名劫持是互联网攻击的一种方式,通过攻击域名解析服务器(DNS),或伪造域名解析服务器(DNS)的方法,把目标网站域名解析到错误的地址从而实现用户无法访问目标网站的目的。说的直白些,域名劫持,就是把互...

取消回复欢迎 发表评论: