执行AJAX返回HTML片段中的JavaScript脚本

本文探讨了AJAX加载HTML片段时脚本未执行的问题,并提供了两种解决方案:一是利用JavaScript的eval方法执行脚本;二是采用jQuery框架的实现方式,详细解析了jQuery在html(value)方法中处理脚本的流程。
摘要由CSDN通过智能技术生成

在公司框架中如何获取另一个页面的html代码

var url = new URL(entryUrl);
var urlString = url.getURLString();

AjaxUtil.asynchAjaxRequest(urlString, function(result){
    // 异常检测与展示
    if(!AjaxUtil.checkException(result)){
        AjaxUtil.showException(result);
        return;
    }
    // ... 后续业务逻辑
});

下面是转载内容:
如果AJAX加载的数据是一个HTML片段,而且这个HTML片段还包含脚本 <script> 块,那么在你把这数据xmlHttp.responseText用innerHTML方法插入到当前文档一个元素中,你会发现AJAX加载回来的脚本根本没有执行。这是AJAX开发中很常见的问题,如果你不是一直在用JavaScript框架做开发,相信你早就发现这个问题了。本文分析了两个解决办法,其中一个是讲解jQuery框架的实现。
一、 问题描述
下面举个简单的例子,演示问题所在。在下面的例子中,假设变量responseText就是AJAX加载的HTML片段数据,其中包含脚本弹出一条消息,用innerHTML方法插入ID为ajaxData的DIV中,你可能期望看到弹出那个消息框,结果你发现没有,问题就是这样。

<div id="ajaxData"></div>
<script type="text/javascript">var=';
        document.getElementById(ajaxData).innerHTML  responseText;
</script>

二、两种解决办法
1、 利用JavaScript的eval方法执行脚本。
本方法的具体实现思路是把xmlHttp.responseText中的脚本都抽取出来,不管AJAX加载的HTML包含多少个脚本块,我们对找出来的脚本块都调用eval方法执行它即可。下面提供一个封装好的函数:

function executeScript(html)
    {
        var reg = /<script[^>]*>([^\x00]+)$/i;
          //对整段HTML片段按<\/script>拆分
    var htmlBlock = html.split("<\/script>");
              for (var i in htmlBlock) 
                {
                    var blocks;//匹配正则表达式的内容数组,blocks[1]就是真正的一段脚本内容,因为前面reg定义我们用了括号进行了捕获分组
        if (blocks = htmlBlock[i].match(reg)) 
                        {
                            //清除可能存在的注释标记,对于注释结尾-->可以忽略处理,eval一样能正常工作
            var code = blocks[1].replace(/<!--/, '');
                                try 
                                  {
                                      eval(code) //执行脚本
            } 
                                      catch (e) 
                                        {
                                        }
                                    }
                                }
                            }

本方法的使用如下,对HTML用innerHTML方法添加到DOM,紧跟着调用executeScript方法执行脚本块:

document.getElementById("div1").innerHTML = xmlHttp.responseText;
    executeScript(xmlHttp.responseText);

显然这个方法还是存在缺陷的,如果xmlHttp.responseText包含像这样的外部脚本调用:

  <script type="text/javascript" src="/js/common.js"></script>

executeScript方法不能再深入执行这个外部加载的脚本。
2、 学习并使用jQuery框架的实现
jQuery对于AJAX加载HTML,是最终在执行html(value)方法时把整个xmlHttp.responseText数据转换成DOM,然后利用DOM相关操作方法来找出里面的脚本,最后再把这些脚本插入到head中。具体原理也不好说,先举个最简单的例子,然后再分析一下大致思路。先看例子:

$.get('ajax.aspx', function(data)
    {
        $('#div1').html(data);
      });
  

现在假设上面ajax.aspx页面返回的是HTML片段,而且包含一个或多个脚本块,甚至外部脚本引用。div1是AJAX请求发起页的一个DIV标签的ID,整句代码实现的结果是加载ajax.aspx中的HTML填充到一个ID为div1的DIV标签中。
在匿名回调函数中通过typeof(data)可以发现data还是原始的字符串,即等同于xmlHttp.responseText,通过代码执行跟踪发现,对AJAX加载脚本片段的执行处理不在jQuery的AJAX模块代码中,而是在html(value)方法,即把一段包含脚本块的HTML字符串插入DOM时,由它负责抽出脚本进行调用处理。而html(value)方法其实又是调用了append(value)方法……,整个过程大概调用了以下方法,箭头代表调用这些方法的先后顺序:
html -> append -> domManip -> clean -> evalScript -> globalEval
其中clean方法特别关键,这个方法也是jQuery比较重要的方法,其中也涉及修复HTML错误(标签没有结束,表格结构调整等方法)处理脚本。而脚本的抽出也是在这里进行的。看看相关源代码(jQuery1.3.2):

if (fragment) 
    {
        for (var i = 0; ret[i]; i++) 
          {
              if (jQuery.nodeName(ret[i], "script") && (!ret[i].type || ret[i].type.toLowerCase() === "text/javascript")) 
                {
                    scripts.push(ret[i].parentNode ? ret[i].parentNode.removeChild(ret[i]) : ret[i]);
                  }
                  else 
                    {
                        if (ret[i].nodeType === 1) 
                              ret.splice.apply(ret, [i + 1, 0].concat(jQuery.makeArray(ret[i].getElementsByTagName("script"))));
                            fragment.appendChild(ret[i]);
                        }
                    }
                    return scripts;
                  }
               

另外,在evalScript方法中我们还发现如下代码,这里是“同”步加载像这样的外部脚本,解决executeScript方法存在的一个缺陷:

if (elem.src) 
        jQuery.ajax(
        {
            url: elem.src,
            async: false,
              dataType: "script"
            });
     

同时也发现如下代码,这段代码是把xmlHttp.responseText中的脚本删除,因为在这个方法中,jQuery是准备把抽取的脚本放入head区,所以删除可以避免最终的HTML出现重复的脚本块:

   if (elem.parentNode)
    elem.parentNode.removeChild(elem);

最后,在globalEval方法中,发现head.removeChild( script );方法,就是把脚本插入head后马上又移除脚本标签,这也是避免因为重复执行html(value)方法在head区生成重复的脚本块。这个移除是不影响脚本执行的,同是也是不会清除脚本块中的相关变量值。显然,如果你想看看html(data)最终的执行结果,比如抽取后插入到head的脚本块是什么,你可以先临时注释这一行代码。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值