【问题标题】:Can somebody explain how John Resig's pretty.js JavaScript works?有人能解释一下 John Resig 的 pretty.js JavaScript 是如何工作的吗?
【发布时间】:2011-06-28 20:09:35
【问题描述】:

http://ejohn.org/files/pretty.js

// Takes an ISO time and returns a string representing how
// long ago the date represents.
function prettyDate(time){
    var date = new Date((time || "").replace(/-/g,"/").replace(/[TZ]/g," ")),
        diff = (((new Date()).getTime() - date.getTime()) / 1000),
        day_diff = Math.floor(diff / 86400);

    if ( isNaN(day_diff) || day_diff < 0 || day_diff >= 31 )
        return;

    return day_diff == 0 && (
            diff < 60 && "just now" ||
            diff < 120 && "1 minute ago" ||
            diff < 3600 && Math.floor( diff / 60 ) + " minutes ago" ||
            diff < 7200 && "1 hour ago" ||
            diff < 86400 && Math.floor( diff / 3600 ) + " hours ago") ||
        day_diff == 1 && "Yesterday" ||
        day_diff < 7 && day_diff + " days ago" ||
        day_diff < 31 && Math.ceil( day_diff / 7 ) + " weeks ago";
}

// If jQuery is included in the page, adds a jQuery plugin to handle it as well
if ( typeof jQuery != "undefined" )
    jQuery.fn.prettyDate = function(){
        return this.each(function(){
            var date = prettyDate(this.title);
            if ( date )
                jQuery(this).text( date );
        });
    };

prettyDate() 方法究竟是如何返回字符串的?这是你可以在 JavaScript 中做的另一件“奇怪”的事情,还是我只是错过了什么?

编辑:我没有问他如何返回一个值,我问他如何返回一个字符串。

return day_diff == 0 &amp;&amp; (....) 以我用过的任何语言返回布尔值。

【问题讨论】:

  • 呃,使用return语句?在它只是说返回值为零的情况下?
  • "&amp;&amp; returns a boolean in any language I've ever used" - 那你没有使用很多语言。 ;-)
  • 不是所有的东西都用真假值检查“真”和“假”。 JavaScript 使用“真”和“假”值
  • :( C++, C, Java, JavaScript, VB, VB.Net, C#, Python, Perl, Ruby, PHP...我现在觉得很蠢。

标签: javascript


【解决方案1】:

在 JavaScript 中:

  • a || b 等价于a ? a : b
  • a &amp;&amp; b 等价于 a ? b : a
  • 布尔表达式中的任何非空字符串的计算结果为真

有了这些知识,return 语句的逻辑就变得相当简单了。

假设,例如,day_diff = 5

然后一步一步地从上面的语句中获取:

return day_diff == 0 && (
       diff < 60 && "just now" ||
       diff < 120 && "1 minute ago" ||
       diff < 3600 && Math.floor( diff / 60 ) + " minutes ago" ||
       diff < 7200 && "1 hour ago" ||
       diff < 86400 && Math.floor( diff / 3600 ) + " hours ago") ||
   day_diff == 1 && "Yesterday" ||
   day_diff < 7 && day_diff + " days ago" ||
   day_diff < 31 && Math.ceil( day_diff / 7 ) + " weeks ago";

首先,day_diff == 0 将评估为 false 和右侧:

(diff < 60 && "just now" ||
 diff < 120 && "1 minute ago" ||
 diff < 3600 && Math.floor( diff / 60 ) + " minutes ago" ||
 diff < 7200 && "1 hour ago" ||
 diff < 86400 && Math.floor( diff / 3600 ) + " hours ago")

...未评估。双方:

day_diff == 1 && "Yesterday"

...评估为false。接下来是:

day_diff < 7 && day_diff + " days ago"

在此表达式中,day_diff &lt; 7 的计算结果为 true,因此其右侧是一个字符串,将被计算并返回其结果。

进一步阅读:

http://www.ejball.com/EdAtWork/2005/02/19/JavaScriptBooleanOperators.aspx

【讨论】:

  • 我认为他更关心 return 语句中条件内部发生的连接,而不是+ 运算符的含义。
  • 谢谢。那有点下意识了。我正在更新一些我希望更有用的东西。
【解决方案2】:

它在那里说:return ...,然后进入一个基本上是“内联ifs”的长嵌套列表。 ;-)

在 Javascript 中,布尔运算符返回操作数之一的 ,而不仅仅是 truefalse。例如。 0 || 'foo' 返回'foo'。此特性与操作员短路结合使用。 false &amp;&amp; true 不会计算 true 端并立即返回 false,因为整个表达式必须是 false

【讨论】:

  • 但更重要的是,这个发帖人可能只使用过静态类型语言(类型在编译时随处可见)。 JavaScript 是一种动态类型语言,这意味着表达式可以根据运行时条件返回不同的类型。这允许"cat" &amp;&amp; 5 返回"cat" 和“null || 32.1”返回32.1
【解决方案3】:

您是 Java 人吗?因为如果是这样,您可能认为if(x) 需要 x 是一个布尔值,而“x && y”返回一个布尔值。它不在 JavaScript 和许多其他语言(如 Perl)中。在许多弱类型语言中,&& 被称为守卫运算符,||称为默认运算符。他们返回两个参数之一。

【讨论】:

  • 我不知道你为什么称它们为“更进化”,因为它们不是——它们只是不同而已。
  • 对“更进化”的破解感到抱歉,我每天都在使用 Java,但仍然无法相信你不得不说 if (myObject != null) 而不仅仅是 if (myObject)
  • 逻辑操作在 JavaScript、Python、Ruby、Perl 等中的工作方式在 Java 中是行不通的,因为变量类型是不可变的。所以你是对的,它只是不同。
  • @darkporter 编辑了更少自以为是的煽动性评论和更真实的评论。您正在寻找“弱类型”。
【解决方案4】:

return 语句只是一个复杂的 if/else 级联,最终在所有非错误情况下返回一个字符串。

例如

return day_diff == 0 && (
        diff < 60 && "just now" ||
        diff < 120 && "1 minute ago" || [...]

如果 day_diff 为零(意味着日期是今天),则它会检查它是否小于 60。如果该语句为真,那么它将短路评估整个事情的其余部分,并且返回表达式的值,即“刚刚”。如果 diff 不小于 60,它将使子表达式短路,并继续进行diff &lt; 120 检查,依此类推。

字符串在表达式中始终为“真”,并且当这种情况匹配时,它们也成为评估表达式的结果。

这是功能性但相当模糊的代码。不用于教学目的。 :)

【讨论】:

    【解决方案5】:

    是的,这是奇怪的 Javascript 东西。字符串连接的计算结果为 true,prettyDate() 底部的 return 语句在条件句中利用了这一点加上 short-circuiting

    所以基本上,在第一种情况下,如果diff 确实小于 60,diff &lt; 60 &amp;&amp; "just now" 的计算结果为字符串“just now”,因为条件中的所有其他顶级项目都是 OR'd 在一起的,因此 Javascript一旦第一个条件成立,评估者就不会关心它们。下线也是如此。

    【讨论】:

      【解决方案6】:

      该行的最后一条语句的格式为

      return boolExpression && otherBoolExpression
      

      当 javascript 读取此内容时,会发生以下情况:

      1. 如果boolExpression为假,则返回boolExpression
      2. 否则返回otherBoolExpression

      这是 JavaScript 执行短路逻辑的方式。

      由于在这种情况下,otherBoolExpression 使用字符串连接,所以函数返回字符串,只要dayDiff 不为 0。

      【讨论】:

        【解决方案7】:

        基本上你只需要知道这一点

        return day_diff == 0 && "lala" + "lolo"
        

        如果day_diff == 0...由于运算符优先级,将返回lalalolo

        所以这只是一种更短的书写方式

        if (day_diff == 0) {
           if (diff < 60) {
              return "just now"
           } else (...)
        } else {
           if (day_diff == 1) {
             return "..."
           }
        }
        

        【讨论】:

          【解决方案8】:

          它正在玩依赖运算符优先级来处理条件的危险游戏:

          + 胜过 &amp;&amp;,后者胜过 ||

          见:https://developer.mozilla.org/en/JavaScript/Reference/Operators/Operator_Precedence

          我是这样读的:

          (day_diff == 0 && (
                      (diff < 60 && "just now") ||
                      (diff < 120 && "1 minute ago") ||
                      (diff < 3600 && Math.floor( diff / 60 ) + " minutes ago") ||
                      (diff < 7200 && "1 hour ago") ||
                      (diff < 86400 && Math.floor( diff / 3600 ) + " hours ago")
                  )) ||
                  (day_diff == 1 && "Yesterday") ||
                  (day_diff < 7 && day_diff + " days ago") ||
                  (day_diff < 31 && Math.ceil( day_diff / 7 ) + " weeks ago");
          

          【讨论】:

            【解决方案9】:

            有两个变量:diff - 以秒为单位的差异,daydiff - 以天为单位的差异。 如果 daydiff 为零,则返回值基于 diff,否则为天数。

            将“return”读取为一系列决定返回字符串的“if”/“else”子句。

            【讨论】:

              【解决方案10】:

              prettyDate() 函数的返回值取决于后面评估中用到的两个变量day_diffdiff

              return day_diff == 0 && (
                     diff < 60 && "just now" ||
                     diff < 120 && "1 minute ago" ||
                     diff < 3600 && Math.floor( diff / 60 ) + " minutes ago" ||
                     diff < 7200 && "1 hour ago" ||
                     diff < 86400 && Math.floor( diff / 3600 ) + " hours ago") ||
                 day_diff == 1 && "Yesterday" ||
                 day_diff < 7 && day_diff + " days ago" ||
                 day_diff < 31 && Math.ceil( day_diff / 7 ) + " weeks ago";
              

              该评估可以更典型地编写为嵌套 if/else 语句,但如果 time 参数是超过 31 天之前的日期,则不会返回任何字符串。

              if (day_diff == 0) {
                if (diff <60) 
                  return "just now";
                else if (diff < 120)  
                  return "1 minute ago";
                else if (diff < 3600)  
                  return Math.floor( diff / 60 ) + " minutes ago";
                else if (diff < 7200)  
                  return "1 hour ago";
                else if (diff < 86400) 
                  return Math.floor( diff / 3600 ) + " hours ago";
              }
              else if (day_diff == 1)
                  return "Yesterday";
              else if (day_diff < 7)
                  return day_diff + " days ago";
              else if (day_diff < 31)
                return Math.ceil( day_diff / 7 ) + " weeks ago"
              

              (并用模板文字稍微更新,强调返回的字符串 cis)

              if (day_diff == 0) {
                if (diff <60)   return "just now";
                else if (diff < 120)  return "1 minute ago";
                else if (diff < 3600)  return `${Math.floor( diff / 60 )} minutes ago`;
                else if (diff < 7200)  return "1 hour ago";
                else if (diff < 86400)  return `${Math.floor( diff / 3600 )} hours ago`;
              }
              else if (day_diff == 1)  return "Yesterday";
              else if (day_diff < 7)  return `${day_diff} days ago`;
              else if (day_diff < 31) return `${Math.ceil( day_diff / 7 )} weeks ago`;
              

              【讨论】:

                猜你喜欢
                • 1970-01-01
                • 2016-06-13
                • 1970-01-01
                • 1970-01-01
                • 2023-01-18
                • 1970-01-01
                • 2014-10-11
                • 1970-01-01
                • 1970-01-01
                相关资源
                最近更新 更多