一个页面使用两次timepicker.js引起的Maximum call stack size exceeded问题解决办法及bug修复

简介:

能大多数人用过,jquery ui组件中,有个叫datepicker的,用来接收用户输入的日期,截图如下:

      

     这个控件的缺点是不能选择时分秒,如项目中需要的输入时分秒这样的数据,就有点遗憾了。不过幸好有了下面这一款插件的出现,那就是timepicker,专门针对datepicker的这个弱点,添加了选择时分秒的功能,截图如下:

      

      这样一来,就可以输入带有时分秒的日期时间了。

 

      本文正题来了:在同一个页面中使用两次或以上timepicker,会报一个错误:Maximum call stack size exceeded。同时浏览器卡死。查了下这个错误,是内存溢出,也就是说因为不恰当的代码,导致了递归或是死循环,所以浏览器卡死了。

      这可是个致命缺陷啊,比如我一个页面中有两处需要输入日期时间,岂不是只能看着浏览器卡死了?于是本人下决心找到这个问题。为了从原理上了解到底是哪里出现了递归或死循环,我决定把timepicker的源码看一遍。(幸亏源码还不算复杂,代码量不多,而且变量命名规范)~

      没看几行就发现问题了,源码从第30行开始,对datepicker的函数进行了重写,典型代码如下:

 

[javascript]  view plain copy
 
  1. $.datepicker._connectDatepickerOverride = $.datepicker._connectDatepicker;  
  2.         $.datepicker._connectDatepicker = function(target, inst) {  
  3.             $.datepicker._connectDatepickerOverride(target, inst);  

      可以看到,作者先将datepicker的connectDatepicker进行了保存,然后重新定义此函数,在重新定义的第一句,又把刚刚保存好的原函数执行了一遍。

 

      问题就出在这里了,这个$.datepicker._connectDatepicker是个全局变量,第一次重写的时候在里面先执行了一次原函数,也就是说这个插件调用一次之后,$.datepicker._connectDatepicker就不是原来的$.datepicker._connectDatepicker了,已经被重写了。那么在调用第二次的时候,取到的$.datepicker._connectDatepicker已经发生了变化,然后又进行函数重写,又把发生变化的$.datepicker._connectDatepicker执行了一遍,就出问题了。

      明白了原因,解决的办法也就自然有了:如果已经重新定义过$.datepicker._connectDatepicker了,就不要让他再重写一遍了。加上判断语句即可。修改后如下:

 

[javascript]  view plain copy
 
  1. if(!$.datepicker._connectDatepickerOverride){  
  2.         $.datepicker._connectDatepickerOverride = $.datepicker._connectDatepicker;  
  3.         $.datepicker._connectDatepicker = function(target, inst) {  
  4.             $.datepicker._connectDatepickerOverride(target, inst);  

 

      既判断$.datepicker._connectDatepickerOverride是否已经存在。如果没有,才进行重写函数。

     严重提醒一下:在此函数下面还有一系列的重写,都要进行相同的判断!!!

 

      修改后运行一下代码,世界美好了~

-----------------------------------------------------------------------

另外,我在使用timepicker的时候还发现了两点bug,在此也提出来,供大家参考。

第一个:timepicker的弹出层的z-index问题。

jquery ui的datepicker弹出层的z-index是动态计算出来的,因为每次弹出来的值都不一样。而timepicker的值确是定死的,没有跟随datepicker一起计算。这样,当页面中还弹出其他一些层的时候,就会发生这样的情况,如图:

timepicker的层被挡住了,跟datepicker不在同一高度上。(在十分碰巧的时候才能遇到这种情况,很不幸,我就遇到了--!)

解决方法也很简单,在timepicker的resize()函数中,添加关于z-index的代码即可。修改后的代码如下:

 

[javascript]  view plain copy
 
  1. this.tpDiv.css({  
  2.             'height': dpDiv.height(),  
  3.             'top'   : dpDivPos.top,  
  4.             'left'  : dpDivPos.left + dpDiv.outerWidth() + 'px',  
  5.         'z-index': dpDiv.css('z-index')  
  6.         });  

让它的z-index跟datepicker的一致就行。

 

第二个,一个页面中使用多次的情况。

在timepicker的_generateHtml()函数中,有如下一句:

 

[javascript]  view plain copy
 
  1. $('body').append(this.tpDiv);  


也就是把拼接好的html代码放到页面上。作者此处有一点疏漏,在append之前未做任何检测,如果页面中已经有一个timepicker显示了,再append一个,岂不就有两个了。随然之前的那一个是隐藏的,看不到,但是在取值的时候,你的jquery选择器替你选到了两个timepicker,(如果页面中用了三次,那就是三个。。。),

 

截一个用了两次的图:

可以看到时分秒那里重复出现了两次,(应该是12:32:00)

所以此处也应该进行一下判断,页面中是否已经存在timepicker了。修改后的代码如下:

 

[javascript]  view plain copy
 
  1. if($('#'+this._mainDivId).length>0){  
  2.     $('#'+this._mainDivId).remove();  
  3. }  
  4. $('body').append(this.tpDiv);<span style="white-space:pre"> </span>  


这一下,世界就又美好了~

 

--------------------------------------------------

最后补充一点,就是我使用的timepicker版本,是0.2.1,现在最新的好像有0.3了,不过去官网看了一下,好像变动比较大,使用方法不同了,而且秒也不能显示了。官网:http://milesich.com/ 

本文转自吕大豹博客园博客,原文链接:http://www.cnblogs.com/lvdabao/archive/2013/04/22/3035866.html,如需转载请自行联系原作者

相关文章
|
29天前
|
JavaScript 前端开发 程序员
前端原生Js批量修改页面元素属性的2个方法
原生 Js 的 getElementsByClassName 和 querySelectorAll 都能获取批量的页面元素,但是它们之间有些细微的差别,稍不注意,就很容易弄错!
|
4月前
|
开发框架 JavaScript 前端开发
揭秘:如何让你的asp.net页面变身交互魔术师——先施展JavaScript咒语,再引发服务器端魔法!
【8月更文挑战第16天】在ASP.NET开发中,处理客户端与服务器交互时,常需先执行客户端验证再提交数据。传统上使用ASP.NET Button控件直接触发服务器事件,但难以插入客户端逻辑。本文对比此法与改进方案:利用HTML按钮及JavaScript手动控制表单提交。后者通过`onclick`事件调用JavaScript函数`SubmitForm()`来检查输入并决定是否提交,增强了灵活性和用户体验,同时确保了服务器端逻辑的执行。
51 5
|
3月前
|
前端开发 JavaScript API
前端JS读取文件内容并展示到页面上
前端JavaScript使用FileReader API读取文件内容,支持文本类型文件。在文件读取成功后,可以通过onload事件处理函数获取文件内容,然后展示到页面上。
113 2
前端JS读取文件内容并展示到页面上
|
2月前
|
Web App开发 前端开发 JavaScript
JavaScript动态渲染页面爬取——Selenium的使用(一)
JavaScript动态渲染页面爬取——Selenium的使用(一)
63 4
|
2月前
|
Web App开发 数据采集 JavaScript
JavaScript动态渲染页面爬取——Selenium的使用(二)
JavaScript动态渲染页面爬取——Selenium的使用(二)
77 2
|
3月前
|
JavaScript 前端开发
js 回到页面顶部
本文提供了一个JavaScript函数`scrollToTop`,用于平滑滚动页面回到顶部。该函数利用`requestAnimationFrame`和`window.scrollTo`方法逐步减少滚动条距离,直到页面完全回到顶部。
34 1
|
2月前
|
JavaScript 前端开发 API
JavaScript全屏,监听页面是否全屏
JavaScript全屏,监听页面是否全屏
66 0
|
3月前
|
JavaScript 前端开发
js怎么定位不同的页面元素
在JavaScript中,有多种方法定位和选择页面元素。
|
2月前
|
前端开发 JavaScript
JavaScript动态渲染页面爬取——CSS位置偏移反爬案例分析与爬取实战
JavaScript动态渲染页面爬取——CSS位置偏移反爬案例分析与爬取实战
38 0
|
2月前
|
存储 JSON JavaScript
JavaScript动态渲染页面爬取——Pyppeteer爬取实战
JavaScript动态渲染页面爬取——Pyppeteer爬取实战
40 0