热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

PHP面试任务:一周计算

不久前,有人给了我一个面试的任务。我是写一个函数,推导出的标准一周工作七天的任何日历(甚至一个虚构的),提供我知道闰年发生频率,如果有的话,一年有多少个月,每个月有多少天。

不久前,有人给了我一个面试的任务。我是写一个函数,推导出的标准一周工作七天的任何日历(甚至一个虚构的),提供我知道闰年发生频率,如果有的话,一年有多少个月,每个月有多少天。

这是一个相当常见的介绍性job-interview任务,在本文中我将解决和解释其背后的数学。我没有电影那么随意扔简化和修正我——我相信我是一个不必要的复杂的。本文将呈现接近问题的两个方面——一个可以让你精神这么做动态(给你的朋友留下深刻印象,我猜?),和一个电脑更友好(更少的代码行,更多)。

日历的定义我得到了如下:

  • 每年有13个月
  • 甚至每一个月21天,每个奇怪的月有22
  • 13月缺少一天每个闰年
  • 闰年是任何被5整除
  • 每周7天:星期天,星期一,星期二,星期三,星期四,星期五,星期六

任务如下:

考虑到1900年的第一天是星期一,编写一个函数,将打印给定日期的星期。的例子中,输入{:17日月:11日:2013 }输出“星期六”。

在本文的其余部分,我将使用以下日期格式:dd.mm。yyyy,因为它是什么是有意义的.

准备

在开始任何聪明的风险之前,有一个适当的环境设置是很重要的,以避免浪费时间在那些可能是提前准备。我总是建议你进入编码面试任务跃跃欲试的开发环境,可以测试您的代码在片刻的注意。

创建一个新文件夹包含两个子文件夹: classes, public。是的,这是一个一次性的任务,它可以解决一个简单的程序功能,但我喜欢彻底。你就会明白为什么。

在 classes子文件夹,创建一个空的PHP类 CalendarCalc.php。在 public子文件夹,创建一个文件 index.php用下面的内容:


如果你可以在浏览器中打开这个显示“Hello”,您已经准备好开始。

CalendarCalc初始化

使事情更容易验证和想象,我创建了一个演示方法从1.1.1900打印出整个日历。22.13.2013。这将使我们能够轻松地检查我们的计算功能。不过,首先初始化类一样:

iNumDays = count($this->aDays);
        $this->iStartDayIndex = array_search('Monday', $this->aDays);
        $this->aInput = array('d' => $day, 'm' => $month, 'y' => $year);
    }

    public function demo() {
    }
}

让我们解释受保护的属性。

$aDays是一个数组。定义它确保每个星期有数字索引分配——至关重要的在确定星期后的一天。我们缓存它的长度 $iNumDays财产。这让我们扩展数组在以后的日子里,如果我们选择——另一个任务可能会问同样的计算,但提到本周可能已经或多或少地超过7天。

$iStartDayIndex是周一的索引(在本例中),因为一开始天周一(1.1.1900)被定义为在任务描述。当我们开始一天的索引,我们可以用它与计算抵消得到真正的星期。你会明白我的意思。

$aInput是一个数组来保存输入值。当我们实例化CalendarCalc,我们通过的日期值我们想知道本周的日子。这个属性存储这些值,可供我们每想出calc方法,从而确保我们不需要它们,或者更糟的是,重复它们在另一个函数调用。的逻辑 $aInput, $iStartDayIndex和 $iNumDays是在 __construct方法。

其他属性都是不言而喻的。

现在,填充 demo()方法用下面的内容:

public function demo() {

        $demoYear = $this->startYear;
        $totalDays = 0;

        while ($demoYear <2014) {

            echo "

$demoYear

"; $demoMOnth= 1; while ($demoMonth <14) { echo ""; echo ""; $dayCount = ($demoMonth % 2 == 1) ? 22 : 21; $dayCount = ($demoMOnth== 13 && $demoYear % 5 == 0) ? 21 : $dayCount; $demoDay = 1; echo ""; while ($demoDay <= $dayCount) { $index = ++$totalDays % 7; if ($demoDay == 1) { for ($i = 0; $i <$index-1; $i++) { echo ""; } if ($index == 0 || $index == 7) { $i = 6; while ($i--) { echo ""; } } } echo ""; if ($index == 0) { echo ""; } $demoDay++; } echo ""; $demoMonth++; } echo "
Month $demoMonth
MondayTuesdayWednesdayThursdayFridaySaturdaySunday
$demoDay

"; $demoYear++; } }

别烦试图理解这种方法——它是完全不重要。这只是帮助我们验证工作,并根据第二个解决方案实际上是部分我们将在本文中呈现。

改变的内容索引。php文件:

demo();

…并在浏览器中打开它。您应该看到一个日历输出与一下图:

我们现在有一种检查结果17.11.2013真理(注意日期。确实是周六)。

精神的方式

心理的方法计算其实很简单。首先,我们需要的数量闰年日期之间的基地,和给定日期。1900是被5整除,本身就是一个闰年。跳跃的数量因此年之间的差异日期输入日期和基地,除以5,四舍五入(只有充分运行年统计,自然),1900年加一。创建一个新的方法 CalendarCalc被称为 calcFuture并给它这个内容:

$iLeaps = floor(($this->aInput['y'] - $this->startYear) / $this->leapInterval + 1);

我们甚至还被告知,每个月有21天,和每一个奇怪的月有22:

1 = > 22
2 = > 21
3 = > 22
4 = > 21
5 = > 22
6 = > 21
7 = > 22
8 = > 21
9 = > 22
10 = > 21
11 = > 22
12 = > 21
13 = > 22(或21日在闰年)

总天数在一年,因此,280年,或者279年在闰年。如果我们的模7 280%,0,因为280年是被7整除。在闰年,模是6。

这意味着每年的日历在同一天开始,除了闰年,当它开始那天,在前一年的第一天。因此,如果1.1.1900。周一:

  • 1.1.1901。是星期一
  • 1.1.1902。是星期天
  • 1.1.1903。是星期天
  • 1.1.1904。是星期天
  • 1.1.1905。是星期六
  • 1.1.1906。是星期六
  • 等…

根据这一点,我们可以计算的数量一天行动,直到我们的输入。看到我们知道我们有23跳跃,直到输入日期(2013),我们搬回一天23倍。23%的模7是2,这意味着我们总算圆满3次,然后两天(这是抵消)——1.1.2013。是星期六。检查演示日历,看看自己。

让我们先来了解代码。“飞跃”线以上,后添加以下:

$iOffsetFromCurrent = $iLeaps % $this->iNumDays;

        $iNewIndex = $this->iStartDayIndex - $iOffsetFromCurrent;

        if ($iNewIndex <0) {
            $iFirstDayInputYearIndex = $this->iStartDayIndex + $this->iNumDays - $iOffsetFromCurrent;
        } else {
            $iFirstDayInputYearIndex = $iNewIndex;
        }

首先,我们计算偏移量。然后,我们计算日子的新索引数组,它取决于是否新指数是积极的。这给了我们一周的日子我们输入年开始。

我们也知道,每个月X与下个月21天使当日月开始X,因为21% 7 = 0。然而奇怪的几个月里,开始提前一天(22% 7 = 1)。因此,如果1月从星期六开始,2月从周日开始,周日3月,4月,星期一,等等。我们得出这样的结论:每一个奇怪的月,通过了今年年初以来直到我们输入日期1月一天拥有先进的指数。我们在11月,所以奇怪有5个月。新抵消+ 5或在我们的案例中,2013年11月,周四开始。让我们把它变成代码立即在前面行。

$iOddMOnthsPassed= floor($this->aInput['m'] / 2);

$iFirstDayInputMOnthIndex= ($iFirstDayInputYearIndex + $iOddMonthsPassed) % $this->iNumDays;

现在剩下的就是看看远离本月初我们输入日期的一天。

$iTargetIndex = ($iFirstDayInputMonthIndex + $this->aInput['d']-1) % $this->iNumDays;

return $this->aDays[$iTargetIndex];

我们添加减一天(因为天还没有通过!),模7日的天数。我们得到的数量是我们的目标指数,可靠地给我们星期六。

从现在,整个 calcFuture的方法 CalendarCalc是这样的:

/**
     * A more "mental" way of calculating the day of the week
     * @return mixed
     */
    public function calcFuture() {
        $iLeaps = floor(($this->aInput['y'] - $this->startYear) / $this->leapInterval + 1);
        $iOffsetFromCurrent = $iLeaps % $this->iNumDays;

        $iNewIndex = $this->iStartDayIndex - $iOffsetFromCurrent;

        if ($iNewIndex <0) {
            $iFirstDayInputYearIndex = $this->iStartDayIndex + $this->iNumDays - $iOffsetFromCurrent;
        } else {
            $iFirstDayInputYearIndex = $iNewIndex;
        }

        $iOddMOnthsPassed= floor($this->aInput['m'] / 2);

        $iFirstDayInputMOnthIndex= ($iFirstDayInputYearIndex + $iOddMonthsPassed) % $this->iNumDays;

        $iTargetIndex = ($iFirstDayInputMonthIndex + $this->aInput['d']-1) % $this->iNumDays;

        return $this->aDays[$iTargetIndex];
    }

machine-friendly方式

也许更简单的方法是计算的天数,已基本日期、模,7和得到抵消。没有很多人可以计算数字的大小,不过,这就是为什么它更machine-friendly。

再一次,我们需要跳跃:

public function calcFuture2() {
    $iTotalDays = 0;

    $iLeaps = floor(($this->aInput['y'] - $this->startYear) /    $this->leapInterval + 1);
}

然后,考虑到年。经过数年的280次,减去的跳跃数占了天,加一,因为今年仍在进行中。

$iTotalDays = (280 * ($this->aInput['y'] - $this->startYear)) - $iLeaps + 1;

然后,我们添加在总结所有的运行。

$iTotalDays += floor($this->aInput['m'] / 2) * 21 + floor($this->aInput['m'] / 2) * 22;

最后,我们添加输入日期的日子,再减去一天因为当前尚未通过:

$iTotalDays += $this->aInput['d'] - 1;
        return $this->aDays[$iTotalDays % $this->iNumDays];

死很简单,不是吗?

结论

看到一个生活例子计算,请检查在这里。你可以浏览目录包含在该URL查看文件,或者您可以下载完整的源代码,演示网站,以及最后的 CalendarCalc类,从GitHub。回购/演示稍微比本文中提供的代码——一些html5boilerplate用于更有组织性和启用ajax请求来检查您输入的日期作为他们,所以你不需要重新加载屏幕和再生日历每次检查日期。

如果你有改进的替代方案或建议,请在下面的评论中让他们-就像我说的我没有数学奇才,欢迎有机会学习更多的知识。例如,一个人应该考虑角情况下,边缘日期,或日期在过去需要更多的修改原来的算法。我将这些留给你。随时提交拉请求,你会得到一个喊出这篇文章!

希望你喜欢这和学习新东西!祝你好运在你的采访!


推荐阅读
author-avatar
懒得张开眼睛看你
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有