iOS之UIWebView無法獲取web標題

来源:http://www.cnblogs.com/fishbay/archive/2017/07/21/7217490.html
-Advertisement-
Play Games

最近遇到了一個問題,就是在`UIWebView`的代理方法里,執行`document.title`的`js`代碼無法獲取網頁標題 ...


最近遇到了一個問題,就是在UIWebView的代理方法里,執行document.titlejs代碼無法獲取網頁標題,代碼如下:

- (void)webViewDidFinishLoad:(UIWebView *)webView
{
    // 取載入html文件的標題名
    NSString *title = [webView stringByEvaluatingJavaScriptFromString:@"document.title"];
}

出現這個問題,我首先確定是不是代碼的問題,經過分析,發現代碼沒有改動,但這次卻無法獲取網頁標題,甚是奇怪。經過查找分析,問題是在這一版中,前端人員把網頁的標題設置放在了非同步操作里,導致UIWebView在載入網頁完成後,在代理方法webViewDidFinishLoad:里無法立即獲取標題,因為獲取標題的方法是非同步的,而網頁載入完就會調用該代理方法,那時候的網頁title還沒有值,所以獲取不到title的值。

下麵是網頁非同步獲取title的代碼,使用了jQueryAjax技術來非同步獲取title

    $.ajax({
        url: remoteur+'/api/innerMessageApi/noticeMessage.htm?callBackFunc=xx',
        type: 'get',
        dataType: 'jsonp',
        jsonpCallback:"xx",
        data: {msgId: msgId},

        success: function(res){
            console.log(res);
            if ( res.successFlag == 'Y' ){
                content = res.content;
                title   = res.title;
            }
        },
        complete:function(res){
            document.title = title;
            $('body').append(content);
        }
    })

出現這個問題的時候,恰是項目要上線的那天晚上,在集體加班時,遇到這個問題,感覺那一夜,被深深傷害。。。

好了,閑話不多說,下麵來介紹解決辦法:

方法一

如果只是考慮iOS方面的解決方法,可能就是延時獲取title,具體就是在webViewDidFinishLoad:里通過延時來執行document.title來獲取標題,雖然可以解決,但是有風險,因為網頁獲取標題是非同步的,而非同步的時間就不確定,所以延時的時間也不確定,雖然可以加大延時的時間,但是不是完美的解決方法

   dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)),        dispatch_get_main_queue(), ^{
       self.title = [webView stringByEvaluatingJavaScriptFromString:@"document.title"];
   });

方法二

如果考慮網頁端,可以把網頁中獲取title的非同步操作改成同步操作,根據上述的js代碼,可以添加一個同步的欄位async: false,修改後的網頁代碼如下:

    $.ajax({
        url: remoteur+'/api/innerMessageApi/noticeMessage.htm?callBackFunc=xx',
        type: 'get',
        dataType: 'jsonp',
        jsonpCallback:"xx",
        data: {msgId: msgId},
        // 設置同步操作
        async: false,

        success: function(res){
            // 同步設置標題
             document.title = res.title;
        complete:function(res){
            ...
        }
    })

雖然這樣可以解決該問題,但是依然不是很好的解決辦法,比如網頁在載入時,用同步的方式獲取網頁標題,假如同步操作被阻塞,那麼網頁載入就被阻塞,進而導致網頁無法展示,所以依然不是最優的解決方法。

方法三

如果把網頁端和iOS端結合起來,可以在網頁端非同步獲取標題,在獲取到標題後通過js調用原生的方法來設置標題,這樣既可以不因同步獲取標題而阻塞網頁載入過程,也不會因延時獲取標題而造成延時時間無法確定的問題,所以該方法可以完美解決這個問題

js端代碼:

    $.ajax({
        url: remoteur+'/api/innerMessageApi/noticeMessage.htm?callBackFunc=xx',
        type: 'get',
        dataType: 'jsonp',
        jsonpCallback:"xx",
        data: {msgId: msgId},
        // 設置同步操作
        async: false,

        success: function(res){
            // 同步設置標題
            document.title = res.title;
            // js調用原生方法來設置標題
            setWebViewTitle(title); 
        complete:function(res){
            ...
        }
    })

iOS端代碼:

    context[@"setWebViewTitle"] = ^(){
       NSArray *args = [JSContext currentArguments];
       if (args.count == 1) {
            // 設置標題,只需要傳遞一個參數
            self.title = [args firstObject];
        }
    };

其實,js調用原生的方式很多,這裡只是一種比較簡單的方式,具體用哪種方式都是可以的,如果對這一塊的知識不甚瞭解,可以參考其它資料。


我找了網上許多的資料,發現動態修改網頁標題的大多都是同步操作,比較少介紹用非同步操作動態修改標題的,所以我把動態設置標題的方法總結如上,希望給有需要的朋友一點幫助。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • css命名規範 頁面結構命名 導航命名 功能命名 CSS樣式命名 CSS樣式表命名 ...
  • JS中的常用類型轉換(一般用強制轉換):1.強制轉為整數:parseInt;寫法:x = parseInt(x); 2.強制轉換位小為:parseFloat;寫法:x = parseFloat(x); 3.檢測類型:x = parseInt(x); alert(typeof(true)); JS中常 ...
  • HTML5 SQL本地資料庫簡單示例 ...
  • 關鍵詞:屬性、標題、段落、、文本格式化、樣式、鏈接、表格、列表、塊、佈局、表單、框架、背景、實體 ...
  • 新特性概覽 參考文章:http://www.cnblogs.com/Wayou/p/es6_new_features.html 這位前輩寫的很好,建議深入學習 ———————————————————————————————————————————— let命令 let命令用來聲明變數,用法類似於va ...
  • KVO 就是key value observing (鍵值監聽 /觀察者模式)/是一種回調機制 觀察者模式 : 一個目標對象管理所有依賴於他的觀察者對象 /併在它自身的狀態改變時主動通知觀察者對象 /這個制動通知通常是通過調用各觀察者對象所提供的介面方法來實現的 /觀察者模式比較完美的將目標對象和觀 ...
  • 介紹: ARC是自iOS 5之後增加的新特性,完全消除了手動管理記憶體的煩瑣,編譯器會自動在適當的地方插入適當的retain、release、autorelease語句。你不再需要擔心記憶體管理,因為編譯器為你處理了一切 註意:ARC 是編譯器特性,而不是 ios 運行時特性(除了weak指針系統),它 ...
  • 將數據存儲到文件中並讀取數據 1、新建FilePersistenceTest項目,並修改activity_main.xml中的代碼,如下:(只加入了EditText,用於輸入文本內容,不管輸入什麼按下back鍵就丟失,我們要做的是數據被回收之前,將它存儲在文件中) 2、修改MainActivity中 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...