最短的IE判定var ie=!-[1,]分析

最短的IE判定var ie=!-[1,]分析

2011-05-02

以前最短的IE判定借助于IE不支持垂直制表符的特性搞出来的。

1 varie = !+"\v1";

仅仅需要7bytes!参见这篇文章,《32 bytes, ehr ... 9, ehr ... 7!!! to know if your browser is IE》,讲述外国人是如何把IE的判定从32 bytes一步步缩简成7 bytes!的故事 但这纪录今年1月8日被一个俄国人打破了,现在只要6 bytes!它利用了IE与标准浏览器在处理数组的toString方法的差异做成的。对于标准游览器,如果数组里面最后一个字符为逗号,JS引擎会自动剔除它。

1 varie = !-[1,];

这句代码在IE9之前曾被称为世界上最短的IE判定代码。代码虽短但确包含了不少javascript基础知识在里面。在这个例子中代码执行时会先调用数组的toString()方法 ,执行[1,].toString()在IE6,7,8中将会得到"1,"。然后表达式就变为!-"1,"。再尝试把"1,"转换成数值类型得到NaN ,再对NaN取负得到值仍为NaN。最后执行!NaN返回true。下面通过分解这个语句来回顾下代码中所涉及到的javascript知识:

1. 浏览器的数组字面量解析差异

[1,]表示使用javascript的数组字面量定义了一个数组。 在IE6,7,8中数组有两个元素,数组中的值分别为1,undefined。在标准的浏览器中会忽略第一个元素后的undefined,数组只包含一个元素1。

2. 数组的toString()方法

调用数组对象的toString()方法时会对数组中的每个元素调用toString()方法,如果元素的值为NULL或者undefined时会返回空的字符串,然后将得到的每项的值拼成一个使用 逗号","分隔的字符串。

3. 一元减号运算符

使用一元减号运算符时如果运算数是数值类型则直接对运算数取负,否则会先尝试把运算数转换为数值类型,转换过程相当于执行Number函数,然后再对得到的结果取负。

4. 逻辑非运算

执行逻辑非运算时如果操作数为NaN、NULL或undefined 时返回 true。

JavaScript可以这么写:

1 varie = !-[1,];
2 alert(ie);

如果从非IE的角度判定,可以省一个比特,因为我们做兼容时,绝大多数情况都是IE与非IE地开工。 var notIE = -[1,];

1 if(-[1,]){
2 alert("这不是IE浏览器!");
3 }else{
4 alert("这是IE浏览器!");
5 }

通过上面的知识可以得出代码 var ie = !-[1,]; 其实等价于 var ie = !(-Number([1,].toString())); 在IE6\7\8中值为true。

因为IE6/7/8都不会忽略[1,].ToString()这个bug,即得到的是"1,";而-Number([1,].toString())即为-Number("1,")得到的结果是NaN;然后!(-Number([1,].toString()))即为!(NaN)即得到true。一切的前提是IE6/7/8都有[1,].ToString()=>"1,"这个bug,而其它浏览器(应该是大部分吧~~)则是[1,].ToString()=>"1"。

原文:http://www.nowamagic.net/javascript/js_ShortestIeDetermine.php

<!DOCTYPE html> <html lang="zh"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>02</title> <link href="bootstrap.min.css" rel="stylesheet"> <style> .bg { overflow: hidden; background-color: beige; } </style> </head> <body> <section class="bg"> <div class="container col-sm-12"> <div class="col-sm-4 col-sm-offset-4" autocomplete="off"> <fieldset> <legend style="color: #0e0e0e">注 册</legend> <div class="form-group"> <label for="passwordField">Password</label> <input type="password" name="passwordField" id="passwordField" class="form-control" required='required'/> </div> </fieldset> <div> <button type="button" class="btn btn-primary btn-validate">验证</button> <span>   强度等级:</span> <span class="result">未验证</span> </div> </div> </div> <div class="container col-sm-12"> <br/> <div class="col-sm-12 col-sm-offset-4"> <p><strong>你填写的密码的强度等级按如下划分:</strong></p> <dl> <dt>低:</dt> <ol> <li>密码必须大于 8 个字符</li> </ol> <dt>中(在满足低强度要求的前提下,需要满足以下需求):</dt> <dd> <ol> <li>至少需要一个小写字母</li> <li>至少需要一个数字</li> </ol> </dd> <dt>高(在满足中强度要求的前提下,需要满足以下需求):</dt> <dd> <ol> <li>至少需要一个大写字母</li> <li>至少需要一个(除数字和字母外的)特殊字符</li> </ol> </dd> </dl> </div> </div> </section> <script src="jquery.min.js"></script> <script> document.getElementById('passwordField').addEventListener('input', function () { const pwd = this.value; let strengthIndicator = document.querySelector('.result'); // 定义匹配模式 var hasLowercase = /[a-z]/.test(pwd); var hasUppercase = /[A-Z]/.test(pwd); var hasNumber = /\d/.test(pwd); var hasSpecialChar = /[\W_]/.test(pwd); // 计算满足条件的数量 var conditionsMetCount = [hasLowercase, hasUppercase, hasNumber, hasSpecialChar].filter(Boolean).length; // 根据题目给出的标准判断强度级别 if (pwd.length >= 8 && hasLowercase && hasNumber && hasUppercase && hasSpecialChar) { strengthIndicator.textContent = '高强度'; strengthIndicator.style.color = '#27AE60'; // 绿色代表强 } else if (pwd.length >= 8 && (hasLowercase || hasUppercase) && hasNumber ) { strengthIndicator.textContent = '中等强度'; strengthIndicator.style.color = '#F39C12'; // 橙黄色代表一般 } else { strengthIndicator.textContent = '低强度'; strengthIndicator.style.color = '#E74C3C'; // 红色代表弱 } }); </script> </body> </html>
最新发布
03-10
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值