Android使用WebView与Native交互的三种方式 ( 附源码 )

news2025/1/7 21:12:16

先附上assets目录中html的源代码文件内容,下面的demo都是使用这几个文件:

javascript.html:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>Carson</title>
    <script>
      function callAndroid(){
        object.hello("hello");
      }
    </script>
</head>
<body>
<button type="button" id="button1" onclick="callAndroid()">点击调用Android代码</button>
</body>
</html>

jsToAndroid.html

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>Carson_Ho</title>
    <script>
      function callAndroid(){
        document.location = "js://webview?name=zxj&age=22";   //document.location 改变uri,使其被shouldOverrideUrlLoading捕获到
      }
      function callAndroid1(){
        document.location = "js://webview1?name=AdvanceDev&age=20"
      }
      function getAndroidReturn(result){
         alert("result is " + result)
         setTimeout(null,3000) //设置延时,这样android在使用loadUrl调用getAndroidReturn的时候,alert才能被捕获(因为界面加载完才可以捕获alert)
      }
      /**
      function showAlert(result){
         alert("result is " + result)
      }
      */
    </script>
</head>
<body>
<button type="button" id="button" onclick="callAndroid()">点击调用Android代码</button>
<button type="button" id="button1" onclick="callAndroid1()">点击调用Android-1代码</button>
</body>
</html>

jsToAndroid2.html

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>Carson_Ho</title>
    <script>
      function callAndroid(){
        var result=prompt("js://webView?name=ZXJ&age=22");
        alert("demo " + result);
      }
    </script>
</head>

<body>
<button type="button" id="button1" onclick="callAndroid()">CallAndroid</button>
</body>
</html>

正文开始

1.JS与Native的交互

一.Android调用JS的方法

目前学习了俩种方法:1. 调用webview的loadUrl 2.调用webview的evaluateJavascript

方法说明: 1. webView.loadUrl(“javascript:callJS()”); 参数是一个字符串,说明调用了javascript中的 callJS方法

  1. webview.evaluateJavascript(arg1,arg2); 需要俩个参数,第一个参数是跟上面的loadUrl一样是个字符串,且格式一致。 第二个参数是一个回调,只有一个方法 onReceiveValue(String value),且该方法的参数value就是所调用的JS方法的返回值 (可以使用lambda)

evaluateJavascript 优势在于异步调用,还可以将执行JS代码的结果带回来。

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>ZhengXJ</title>
    <body>
      <h3>测试--Android调用了JS--界面</h3>
    </body>
    <script>
       function callJS(){
          alert("Android调用了JS的callJS方法");
          return "方法结束"
       }
    </script>
</head>
</html>

java代码:

public class MainActivity extends AppCompatActivity {
    private final String TAG = "MainActivity";
    private WebView webView;
    private Button button;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        webView = findViewById(R.id.webView);
        button = findViewById(R.id.button);

        WebSettings webSettings = webView.getSettings();
        // 设置与JS交互的权限
        webSettings.setJavaScriptEnabled(true);
        webView.loadUrl("file:///android_asset/javascript.html");//必须要的,加载webview页面
        button.setOnClickListener(v -> {
            webView.post(() -> {
                // 调用javascript的callJS()方法
//                webView.loadUrl("javascript:callJS()");
                webView.evaluateJavascript("javascript:callJS()", new ValueCallback<String>() {
                    @Override
                    public void onReceiveValue(String value) {
                        //value是所调用的JS方法的返回值
                        Log.d(TAG, "onReceiveValue: " + value);
                    }
                });
            });
        });


        // 通过设置WebChromeClient对象处理JavaScript的对话框
        // 设置拦截js的Alert()函数
        webView.setWebChromeClient(new WebChromeClient() {
            //拦截JS中的alert()
            @Override
            public boolean onJsAlert(WebView view, String url, String message, JsResult result) {
                AlertDialog.Builder builder = new AlertDialog.Builder(MainActivity.this);
                builder.setTitle("Alert");
                builder.setMessage(message);
                builder.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        result.confirm();  //在onJsAlert返回值会true的时候需要加这个,表示将JS中的Alert也点击确定。否则JS方法不会return
                        //如何onJsAlert的返回值为false则不需要,因为false表示onJsAlert处理完逻辑后,还要交给JS中的Alert去操作,也就是说JS中的Alert也会出现
                    }
                });
                builder.setCancelable(false); //使android中,点击对话框外部或者按下返回键也取消不了对话框。  如果设置为true,则相反
                builder.create().show();

                return false;
            }
        });
    }
}

在这里插入图片描述

点击:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-J6c3rDy8-1686711235234)(C:\Users\SW\AppData\Roaming\Typora\typora-user-images\image-20230530153000363.png)]

二. JS调用Android的代码:

目前学习了三种方法:

  1. 通过webView的addJavascriptInterface(Object object, String name)的方式,利用@JavascriptInterface来映射方法实现调用
  2. 通过webViewCilent的shouldOverrideUrlLoading来拦截Url的方式(需要配合javascript改变document.location的值来获取和拦截Url),解析url然后调用对应的方法
  3. 通过WebChromeClient的onJsAlert()、onJsConfirm()、onJsPrompt()方法回调拦截JS对话框alert()、confirm()、prompt() 消息, 然后解析对应的消息内容来调用对应的方法即可

1. webView的addJavascriptInterface方式

使用步骤:

·创建暴露方法的类,为要暴露的方法加上@JavascriptInterface注解

import android.webkit.JavascriptInterface;

public class AndroidJS extends Object {
    private final String TAG = "AndroidJS";
    @JavascriptInterface
    public void hello(String msg) {
        Log.d(TAG, Thread.currentThread().getId()  + "----" + Thread.currentThread().getName() + "----JS调用了Android的方法: " + msg);
        Log.d(TAG, "当前进程号: " + android.os.Process.myPid());
    }
}

上述log日志 只是用来验证 android通过webview的与JS交互的线程以及进程

结论是:进程号一致,线程不一样,线程是JavaBridge,非主线程main,因此不可以更新UI (多进程开启WebView的方式是解决webview内存泄漏的重要手段之一)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-84YI9O5D-1686711235235)(C:\Users\SW\AppData\Roaming\Typora\typora-user-images\image-20230530155856774.png)]

·调用webview的addJavascriptInterface

        // 通过addJavascriptInterface()将Java对象映射到JS对象
        // 参数1:Javascript对象名  参数2:Java对象名
        // AndroidJS类对象映射到js的object对象
        webView.addJavascriptInterface(new AndroidJS(), "object");

        webView.loadUrl("file:///android_asset/javascript.html"); 

测试用例:

javascript.html:放在了assets目录下

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>Carson</title>
    <script>
      function callAndroid(){
        object.hello("hello");
      }
    </script>
</head>
<body>
<button type="button" id="button1" onclick="callAndroid()">点击调用Android代码</button>
</body>
</html>

完整java代码:

public class MainActivity extends AppCompatActivity {
    private final String TAG = "AndroidJS";
    private WebView webView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Log.d(TAG, "onCreate: " + Thread.currentThread().getId()  + "    " + Thread.currentThread().getName());
        Log.d(TAG,"当前主进程: " + android.os.Process.myPid() + "    ");
        webView = findViewById(R.id.webView);

        WebSettings webSettings = webView.getSettings();
        webSettings.setJavaScriptEnabled(true); // 设置与JS交互的权限

        // 通过addJavascriptInterface()将Java对象映射到JS对象
        // 参数1:Javascript对象名  参数2:Java对象名
        // AndroidJS类对象映射到js的object对象
        webView.addJavascriptInterface(new AndroidJS(), "object");
        webView.loadUrl("file:///android_asset/javascript.html"); //加载html页面

    }

}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-baiM5Yh2-1686711235236)(C:\Users\SW\AppData\Roaming\Typora\typora-user-images\image-20230530161130430.png)]

(按钮是在html页面中的)点击按钮后查看log:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MorZAhJX-1686711235237)(C:\Users\SW\AppData\Roaming\Typora\typora-user-images\image-20230530161235702.png)]

android的hello方法被js调用

2.webViewCilent的shouldOverrideUrlLoading拦截url的方式

使用步骤:

·为WebView添加webViewClient,并重写shouldOverrideUrlLoading方法

 webView.setWebViewClient(new WebViewClient() {
            @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
            @Override
            public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) {
                // 步骤2:根据协议的参数,判断是否是所需的url
                // 一般根据scheme(协议格式)和authority(协议名)判断前两个参数
                // 假定传进来的 url="js://webview?arg1=111&arg2=222"
                Uri uri = Uri.parse(request.getUrl().toString());
                Log.d(TAG, "shouldOverrideUrlLoading: " + uri);
                // 如果authority=预先约定协议里的webview,即代表符合约定的协议
                // 所以拦截url,下面JS开始调用Android需要的方法
                if (uri.getScheme().equals("js")) {
                    if (uri.getAuthority().equals("webview")) {
                        // 步骤3:执行JS所需要的逻辑
                        // 可以在协议上带有参数并传递到Android上
                        HashMap<String, String> params = new HashMap<>();
                        Set<String> collection = uri.getQueryParameterNames();
                        for(String str : collection){
                            params.put(str,uri.getQueryParameter(str));
                        }
                        Log.d(TAG, "shouldOverrideUrlLoading: " + params);
                        String result = "'从Android返回给JS的result'";
                        view.loadUrl("javascript:getAndroidReturn("+ result+")");
//                        view.evaluateJavascript("getAndroidReturn('从Android返回给JS的result')", null);
                        /**
                         * shouldOverrideUrlLoading: {name=zxj, age=22}
                         * ----------------------------------------------
                         * 获取到参数和值之后,就可以灵活调用对应的方法了
                         */
                    } else if (uri.getAuthority().equals("webview1")) {
                        // 可以在协议上带有参数并传递到Android上
                        HashMap<String, String> params = new HashMap<>();
                        Set<String> collection = uri.getQueryParameterNames();
                        for(String str : collection){
                            params.put(str,uri.getQueryParameter(str));
                        }
                        Log.d(TAG, "shouldOverrideUrlLoading: " + params);
                        String result = "'从Android返回给JS的result'";
                        view.loadUrl("javascript:getAndroidReturn("+ result+")"); //不推荐这种去返回方法,而是通过evaluateJavascript 比较好。
                        /**
                         * 原因:如果调用的方法,上述是getAndroidReturn,需要处理内容然后弹出alert等,
                         * 如果执行太快(在webview界面加载完成前执行完调用方法(例如getAndroidReturn)),那么在onJsAlert就拦截不了窗口,就无法知道js做了什么
                         * 而evaluateJavascript需要操作完alert后才能获取到返回值,因此使用它去调用JS的方法传递返回值能拦截到Alert,会更好
                         */
//                        view.evaluateJavascript("getAndroidReturn('从Android返回给JS的result')", null);

                    }

                    return true;//一定要设置为true,表示我已经处理了该事件,否则webview会自行去处理(将跳转到修改后的URL,会导致找不到相关界面报错)
                }

                return super.shouldOverrideUrlLoading(view, request);
            }

        });

·而html的javaScript方法代码中要加上:

document.location = "js://webview?name=zxj&age=22";   
//document.location 改变uri,使其被shouldOverrideUrlLoading捕获到

示例:

jsToAndroid.html代码:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>Carson_Ho</title>
    <script>
      function callAndroid(){
        document.location = "js://webview?name=zxj&age=22";   //document.location 改变uri,使其被shouldOverrideUrlLoading捕获到
      }
      function callAndroid1(){
        document.location = "js://webview1?name=AdvanceDev&age=20"
      }
      function getAndroidReturn(result){
         alert("result is " + result)
         setTimeout(null,3000) //设置延时,这样android在使用loadUrl调用getAndroidReturn的时候,alert才能被捕获(因为界面加载完才可以捕获alert)
      }
      /**
      function showAlert(result){
         alert("result is " + result)
      }
      */
    </script>
</head>
<body>
<button type="button" id="button" onclick="callAndroid()">点击调用Android代码</button>
<button type="button" id="button1" onclick="callAndroid1()">点击调用Android-1代码</button>
</body>
</html>

java调用代码:

public class MainActivity extends AppCompatActivity {
    private final String TAG = "AndroidJS";
    private WebView webView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Log.d(TAG, "onCreate: " + Thread.currentThread().getId()  + "    " + Thread.currentThread().getName());
        Log.d(TAG,"当前主进程: " + android.os.Process.myPid() + "    ");
        webView = findViewById(R.id.webView);

        WebSettings webSettings = webView.getSettings();
        webSettings.setJavaScriptEnabled(true); // 设置与JS交互的权限
        webView.setWebViewClient(new WebViewClient() {
            @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
            @Override
            public boolean shouldOverrideUrlLoading(WebView view, WebResourceRequest request) {
                // 步骤2:根据协议的参数,判断是否是所需的url
                // 一般根据scheme(协议格式)和authority(协议名)判断前两个参数
                // 假定传进来的 url="js://webview?arg1=111&arg2=222"
                Uri uri = Uri.parse(request.getUrl().toString());
                Log.d(TAG, "shouldOverrideUrlLoading: " + uri);
                // 如果authority=预先约定协议里的webview,即代表符合约定的协议
                // 所以拦截url,下面JS开始调用Android需要的方法
                if (uri.getScheme().equals("js")) {
                    if (uri.getAuthority().equals("webview")) {
                        // 步骤3:执行JS所需要的逻辑
                        // 可以在协议上带有参数并传递到Android上
                        HashMap<String, String> params = new HashMap<>();
                        Set<String> collection = uri.getQueryParameterNames();
                        for(String str : collection){
                            params.put(str,uri.getQueryParameter(str));
                        }
                        Log.d(TAG, "shouldOverrideUrlLoading: " + params);
                        String result = "'从Android返回给JS的result为 " + params + "'";
                        view.loadUrl("javascript:getAndroidReturn("+ result+")");
//                        view.evaluateJavascript("getAndroidReturn('从Android返回给JS的result')", null);
                        /**
                         * shouldOverrideUrlLoading: {name=zxj, age=22}
                         * ----------------------------------------------
                         * 获取到参数和值之后,就可以灵活调用对应的方法了
                         */
                    } else if (uri.getAuthority().equals("webview1")) {
                        // 可以在协议上带有参数并传递到Android上
                        HashMap<String, String> params = new HashMap<>();
                        Set<String> collection = uri.getQueryParameterNames();
                        for(String str : collection){
                            params.put(str,uri.getQueryParameter(str));
                        }
                        Log.d(TAG, "shouldOverrideUrlLoading: " + params);
                        String result = "'从Android返回给JS的result为 " + params + "'";
                        view.loadUrl("javascript:getAndroidReturn("+ result+")"); //不推荐这种去返回方法,而是通过evaluateJavascript 比较好。
                        /**
                         * 原因:如果调用的方法,上述是getAndroidReturn,需要处理内容然后弹出alert等,
                         * 如果执行太快(在webview界面加载完成前执行完调用方法(例如getAndroidReturn)),那么在onJsAlert就拦截不了窗口,就无法知道js做了什么
                         * 而evaluateJavascript需要操作完alert后才能获取到返回值,因此使用它去调用JS的方法传递返回值能拦截到Alert,还可以获取调用方法的返回值,会更好
                         */
//                        view.evaluateJavascript("getAndroidReturn('从Android返回给JS的result')", null);

                    }

                    return true;  //一定要设置为true,表示我已经处理了该事件,否则webview会自行去处理(将跳转到修改后的URL,会导致找不到相关界面报错)
                }

                return super.shouldOverrideUrlLoading(view, request);
            }

        });

        //拦截JS的Alert()
        webView.setWebChromeClient(new WebChromeClient(){
            @Override
            public boolean onJsAlert(WebView view, String url, String message, JsResult result) {
                Log.d(TAG, "onJsAlert: " + url);
                AlertDialog.Builder builder = new AlertDialog.Builder(MainActivity.this);
                builder.setTitle("Alert");
                builder.setMessage(message);
                builder.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        result.confirm();  //在onJsAlert返回值会true的时候需要加这个,表示将JS中的Alert也点击确定。否则JS方法不会return
                        //如何onJsAlert的返回值为false则不需要,因为false表示onJsAlert处理完逻辑后,还要交给JS中的Alert去操作,也就是说JS中的Alert也会出现
                    }
                });
                builder.setCancelable(false); //使android中,点击对话框外部或者按下返回键也取消不了对话框。  如果设置为true,则相反
                builder.create().show();

                return true;
            }

        });
        webView.loadUrl("file:///android_asset/jsToAndroid.html"); //此处是利用shouldOverrideUrlLoading 解析URL进行交互

    }

}

运行用例:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kFtVmDMW-1686711235238)(C:\Users\SW\AppData\Roaming\Typora\typora-user-images\image-20230530162612501.png)]

点击第一个按钮:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-pDqvp4as-1686711235239)(C:\Users\SW\AppData\Roaming\Typora\typora-user-images\image-20230530163038135.png)]

看log:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-q3Ju9ccl-1686711235240)(C:\Users\SW\AppData\Roaming\Typora\typora-user-images\image-20230530163105680.png)]

点击第二个按钮:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-w9uikhKE-1686711235241)(C:\Users\SW\AppData\Roaming\Typora\typora-user-images\image-20230530163131111.png)]

看log:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VHJHeQvf-1686711235241)(C:\Users\SW\AppData\Roaming\Typora\typora-user-images\image-20230530163153486.png)]

上述demo模拟JS调用Android的方法,并且将返回值返回给JS,然后JS将返回值以alert的形式展示出来,并被Android端拦截(跟前面的方式一样,用webchromclient的onJsAlert拦截JS中的alert方法)

PS:把返回值返回给JS,只能通过Android调用JS方法,并把返回值以参数的形式传递过去了。跟前面的方式一样,有俩种,loadUrl和evaluateJavaScript的方式。 经过测试,不建议使用loadUrl方式,如果使用loadUrl方式,调用的方法执行太快就结束了(在webview界面加载完成前执行完该调用方法(例如上述的getAndroidReturn)),那么在onJsAlert就拦截不了alert,就无法知道js做了什么。 而evaluateJavascript需要操作完alert后才能获取到返回值,因此使用它去调用JS的方法传递返回值能拦截到Alert,还可以获取调用方法的返回值,会更好

3.通过WebChromeClient的onJsAlert()、onJsConfirm()、onJsPrompt()方法回调拦截JS对话框alert()、confirm()、prompt() 消息, 然后解析对应的消息内容来调用对应的方法即可。

注意:对JS调用Android,在4.2以下存在的漏洞问题资料调研:

在Android 4.2及以下版本中,`addJavascriptInterface`存在一个重要的安全漏洞,即JavaScript可以通过调用Java对象的方法来执行任意代码。这个漏洞使得恶意网页有可能通过`WebView`对象中的Java对象执行任意的本地代码,可能导致用户数据的泄露或设备被控制。

这个漏洞的根本原因是,在Android 4.2及以下的版本中,`addJavascriptInterface`方法默认允许JavaScript访问Java对象的所有公共方法。这意味着,如果你在`WebView`中添加了一个Java对象,恶意网页可以通过调用该对象的任意公共方法来执行恶意代码,而无需用户的授权或知情。
可能导致WebView加载恶意网页的情况,例如:

1. 用户受到社会工程攻击:恶意行为者可能通过欺骗用户点击恶意链接或下载恶意应用程序等方式,让用户访问恶意网页。这可能是通过钓鱼邮件、欺诈性广告、恶意应用程序等方式实施的。
2. 代码注入:如果应用程序的WebView加载的内容依赖于用户提供的数据,例如从服务器或其他第三方来源动态加载内容,那么存在潜在的代码注入风险。恶意用户或攻击者可以在提供的数据中插入恶意代码,以此来加载恶意网页。
3. 未经授权的第三方库:如果应用程序使用了第三方库或服务来加载WebView内容,存在潜在的安全风险。如果这些第三方库存在漏洞或被攻击,可能会导致WebView加载恶意内容。

虽然开发人员可以通过合适的验证和过滤来确保WebView加载的内容是安全的,但不能排除所有的恶意行为。因此,用户也需要保持警惕,避免点击可疑的链接或下载未知来源的应用程序。
在Android 4.2及以上的版本中,`addJavascriptInterface`方法引入了一个新的注解`@JavascriptInterface`,用于标记供JavaScript调用的公共方法。只有被`@JavascriptInterface`注解标记的公共方法才会暴露给JavaScript使用,而未被注解标记的方法将不可见。

这个改变的目的是为了确保开发人员有意识地选择要暴露给JavaScript的方法,并提供了一种明确的方式来定义这些方法。通过使用`@JavascriptInterface`注解,开发人员可以明确指定哪些方法是安全的,以供JavaScript调用,从而减少了潜在的安全风险。

因此,在Android 4.2及以上的版本中,只有通过`@JavascriptInterface`注解标记的方法才能被JavaScript调用,其他未标记的方法将对JavaScript不可见。这种限制大大降低了潜在的安全漏洞,并提高了`addJavascriptInterface`的安全性。

总上所述:以addJavascriptInterface的方式来实现JS调研Android,还是要用4.2以上版本使用最好。记得使用@JavascriptInterface即可

2.使用WebView遇到的问题

·webview报错 net::ERR_UNKNOWN URLSCHEME

原因:

产生原因:webview重定向,其定义没有明确的官方解释,发生的原因是请求的链接(url)在加载完成后发生了变化 (eg.比如你的代码中设置webview加载的是网页A,打开后发现加载的是网页B); 关于 net::ERR_UNKNOWN_URL_SCHEME (如下图所示),因为webview只能识别http和https协议,遇到不属于"http(或https)😕/"开头的自定义协议时就无法识别,便会提示ERR_UNKNOWN_URL_SCHEME这样的错误。

在这里插入图片描述

解决:

重写WebviewClient类中的 shouldOverriderUrlLoading 方法( 选取方法参数为(Webview view , String url )的那种,如下图),该方法可以对webview将要加载的url 进行处理,我们在此处对 会发生重定向的 url 和 不以 “http://”、“https//” 开头的自定义协议 进行拦截处理。该方法的返回值为boolean 类型,表示是否阻止webview继续加载url,默认值为 false。当返回false,表示不进行阻止,webview认为当前的url需要进行处理,会继续加载;返回 true,表示阻止webview继续加载url,等待我们进行处理。

             //加载的url是http/https协议地址
   if (url.startsWith("http://") || url.startsWith("https://")) { 
           view.loadUrl(url);
           return false; //返回false表示此url默认由系统处理,url未加载完成,会继续往下走
   } 

·页面百度点开图片无法加载图片问题:

产生原因不详,但分析可能是因为百度是个搜索引擎,搜到的图片都是来自各大网址中的内容。

解决: webView.getSettings().setDomStorageEnabled(true); 开启WebView的DOM存储功能即可。

注意:还需要 webView.getSettings().setBlockNetworkImage(false); 不阻塞加载网络图片

总结:webview遇到的问题大多都可以通过分析所加载的html页面的特性来配置对应的设置即可。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/644997.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

深入理解多层感知机(MLP):原理与代码解析

文章目录 1. MLP的原理1.1 结构1.2 激活函数1.3 前向传播1.4 反向传播算法 2.MLP分类任务应用3.参考文献&#xff1a; 多层感知机&#xff08;MLP&#xff09;是一种经典的神经网络模型&#xff0c;由多个神经元层组成。它的结构和功能使其成为深度学习中的重要组成部分。MLP在…

【Java算法题】剑指offer_算法之01搜索算法

前言 刷题链接&#xff1a; https://www.nowcoder.com/exam/oj/ta?page2&tpId13&type265 1. 搜索算法 JZ53 数字在升序数组中出现的次数 思路&#xff1a;遍历数组&#xff0c;count记录k值出现次数&#xff0c;返回count public class Solution {public int GetN…

【word wps文字】目录页码中的格式在打印或打印预览时变为和正文页码格式一样,如何调整?

一、问题背景 之前在闲鱼上&#xff0c;有个人找我改word排版&#xff0c;有一个需求就是正文页码两边需要横杠。 但是目录中显示的页码&#xff0c;不需要横杠。 我当时是一个一个在目录中删除横杠的&#xff0c;借助了查找与替换功能。 更改后&#xff0c;目录页码如下所…

Java与SpringBoot对redis的使用方式

目录 1.Java连接redis 1.1 使用Jedis1.2 使用连接池连接redis1.3 java连接redis集群模式 2.SpringBoot整合redis 2.1 StringRedisTemplate2.2 RedisTemplate 1.Java连接redis redis支持哪些语言可以操作 &#xff08;去redis官网查询&#xff09; 1.1 使用Jedis (1)添加jedis…

HTML+CSS实训——Day14——项目其他页面的完善

仓库地址&#xff1a;HTML实训 前言 今天我们继续用老师提供的api&#xff0c;完善一些剩余的功能&#xff0c;因为我的git push好像传乱了&#xff0c;所以仓库大家看看最新的就好&#xff0c;最新的一天一定包括前一天所做的内容。 Collect.htmlcss 收藏界面 <!DOCT…

Dijkstra迪杰斯特拉算法求最短路径(C++实现)

名人说&#xff1a;一花独放不是春&#xff0c;百花齐放花满园。——《增广贤文》 作者&#xff1a;Code_流苏(CSDN)&#xff08;一个喜欢古诗词和编程的Coder&#x1f60a;&#xff09; 目录 〇、Dijkstra迪杰斯特拉算法介绍1、Dijkstra算法是什么&#xff1f;2、Dijkstra算法…

低代码工具:jvs-list(列表引擎)2.1.7功能清单及新增功能介绍

在低代码开发平台中&#xff0c;列表页是一个用于显示数据列表的页面。它通常用于展示数据库中的多条记录&#xff0c;并提供搜索、排序和筛选等功能&#xff0c;以方便用户对数据进行查找和浏览。 jvs-list是jvs快速开发平台的列表页的配置引擎&#xff0c;它和普通的crud 具…

Rocky Linux9.的系统中安装MySQL8 实战

前言 Centos7 已经停止维护&#xff0c;学习其他linux系统势在必行&#xff0c;今天我们要探讨的是&#xff1a; 在Rocky linux9的系统上安装MySQL8 文章目录 前言1. 从Appstream中进行安装1.1 更新系统中的所有软件包1.2 安装MySQL1.3 启动并测试1.4 查看MySQL版本 2. 初始化操…

获取地理位置请求免费天气接口

需求&#xff1a;根据地理位置信息去请求免费的天气接口数据&#xff0c;拿到数据后进行展示&#xff0c;这边我用到了俩个key&#xff0c;一个是高德天气的key和心知天气的key&#xff0c;为什么要这么麻烦呢&#xff0c;是因为之前写过一版不需要获取地理位置&#xff0c;直接…

嵌入式系统开发中的常见挑战和困难

当涉及嵌入式系统开发时&#xff0c;可能会遇到以下一些常见的挑战和困难&#xff1a; 复杂的硬件和软件集成&#xff1a;嵌入式系统通常涉及硬件和软件的紧密集成&#xff0c;需要同时理解和处理硬件和软件层面的问题。这种复杂性可能导致调试和故障排除变得更加困难。 有限…

【哈佛积极心理学笔记】第19讲 让爱情天长地久

第19讲 让爱情天长地久 What makes relationship thrive, some characteristic: work hard the fix mindset: “you are so smart, you are so intelligent” the malleable mindset: “you work so hard” Finding mindset: “finding the right partner” (fix) some thin…

PLC领域从业者的工作待遇现状如何?

目前从事可编程逻辑控制器&#xff08;PLC&#xff09;领域的人员在工作待遇方面相对较好。PLC是工业自动化中广泛使用的控制设备&#xff0c;用于监控和控制各种工业过程和机械设备。以下是关于从事PLC的人员工作待遇的一些常见情况和趋势&#xff1a; 薪资水平&#xff1a;P…

【算法与数据结构】242、LeetCode有效的字母异位词

文章目录 一、题目二、解法三、完整代码 所有的LeetCode题解索引&#xff0c;可以看这篇文章——【算法和数据结构】LeetCode题解。 一、题目 二、解法 思路分析&#xff1a;题目要求判断两个字符串是否为字母异位词。什么是字母异位词呢&#xff1f;顾名思义&#xff0c;就是字…

深入理解相机硬件抽象层

和你一起终身学习&#xff0c;这里是程序员Android 经典好文推荐&#xff0c;通过阅读本文&#xff0c;您将收获以下知识点: 一、概览二、Camera HIDL 接口三 、Camera Provider 主程序四、Camera HAL3 接口 一、概览 始于谷歌的Treble开源项目&#xff0c;基于接口与实现的分离…

【AUTOSAR】UDS协议的代码分析与解读(二)----ECU诊断协议概述

UDSO诊断服务技术规范 1 范围 本规范规定了增强型诊断需求的诊断服务部分的内容&#xff0c;定义了通用电子系统需遵循的UDS通用执行 规则。 本规范适用于集团x事业部所有平台车型&#xff0c;所有电子控 制单元(ECU) 的诊断需求&#xff0c;均需按此规范执行。 本规范定义的…

Python神经网络编程学习笔记

文章目录 神经网络基本原理线性分类器学习率一个线性分类器的局限性逻辑AND、逻辑OR逻辑XOR 神经元sigmoid function的logistic function(逻辑函数) 多层神经元演示只有两层&#xff0c;每层两个神经元的神经网络的工作矩阵大法(点乘)使用矩阵乘法的三层神经网络示例反向传播误…

AJAX概述

1.1什么是AJAX. Ajax即AsynchronousJavascript And XML&#xff1a;异步数据回调。 使用Ajax技术网页应用能够快速地将更新呈现在用户界面上&#xff0c;不需要重载&#xff08;刷新&#xff09;整个页面【只刷新局部】&#xff0c;这使得程序能够更快地回应用户的操作。、 1…

H5U PLC EtherCAT总线伺服回原(6098H=3)

回原方式35请参看下面文章 汇川H5U PLC通过 EtherCAT总线控制伺服回原_RXXW_Dor的博客-CSDN博客大部分运动控制都会对机械回原点进行大篇幅讲解,也可以看出机械回原点的重要性。常规的回原点方式大概有几十种吧,本文会给出常用回原点的注意事项,和编程推荐写法。如果原点回…

二层和三层交换机到底有啥区别?二者如何切换?

概要 计算机网络中的交换机是用于在局域网&#xff08;LAN&#xff09;中转发数据包的重要设备。其中&#xff0c;二层交换机和三层交换机是两种常见的交换机类型。本文将详细介绍二层交换机和三层交换机的特点、工作原理、各自的优缺点以及在思科、华为、瞻博网络三家厂商如何…

互联网架构师联合总结的 Java 面试攻略,GitHub 标星 30K!

2023 年的互联网行业竞争越来越严峻&#xff0c;面试也是越来越难&#xff0c;一直以来我都想整理一套完美的面试宝典&#xff0c;奈何难抽出时间&#xff0c;这套 1000道的 Java 面试手册是行业内各大神联合总结出来的&#xff0c;上传到 Git 上目前 star 数达到了 30K 这套互…