查看指定日期是本年和本月的第几周

约定:星期一是一周的开始。
其中的计算结果以如下网站为准。
参考:https://wannianli.tianqi.com/...

//给定的日期是第本年中的第几天
    function getDayIndexInYear(t){
        var y, m, d;
        var total = 0;
        var arr = new Array(31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31);
        y = t.getFullYear()
        m= t.getMonth() + 1
        d = t.getDate()
        for (var i = 0; i < m - 1; i++) {
            total = total + arr[i];
        }
        if ((y % 400 == 0 || (y % 4 == 0 && y % 100 != 0)) && m > 2) {
            total = total + d + 1

        } else {
            total = total + d
        }
        return total;
    }
 //给定的日期是今年的第几周
function getWeekIndexInYear(t) {
        var date1 = getDayIndexInYear(t); //给定的日期是本年的第几天
        var _t  = new Date(); //复制对象,以免修改覆盖。
        _t.setYear(t.getFullYear())
        _t.setMonth(t.getMonth())
        _t.setDate(t.getDate())

        _t.setMonth(0)
        _t.setDate(1)

        var d = _t.getDay(); //1. 得到今年的1月1号是星期几。
        var fisrtWeekned = d;
        if(d == 0) {
            fisrtWeekned = 1;//1号就是星期天,第一周的周未是1号
        }
        else{
            fisrtWeekned = 7 - d + 1; //第一周的周未是几号
        }

        if(date1 <= fisrtWeekned){
            return 1
        }
        else {
            return 1 + Math.ceil( (date1 - fisrtWeekned)/7 )
        }
    }
    
    /***
     * 计算今天是本月的第几周
     * @param t
     * @returns {number}
     */
    function getWeekIndexInMonth (t) {
        if(t == undefined ){
            t = new Date();
        }
        else if(t instanceof Date) {
            var _t  = new Date();
            _t.setYear(t.getFullYear())
            _t.setMonth(t.getMonth())
            _t.setDate(t.getDate())

            var date1 = _t.getDate(); //给定的日期是几号

            _t.setDate(1)
            var d = _t.getDay(); //1. 得到当前的1号是星期几。
            var fisrtWeekend = d;
            if(d == 0) {
                fisrtWeekend = 1;
                //1号就是星期天
            }
            else{
                fisrtWeekend = 7 - d + 1; //第一周的周未是几号
            }

            if(date1 <= fisrtWeekend){
                return 1
            }
            else {
                return 1 + Math.ceil( (date1 - fisrtWeekend)/7 )
            }

        }
        else{
            throw 'getFormatDate - error : 你的参数不是日期类型,也不是为空';
        }
    }
    //测试
    console.info(getWeekIndexInYear(new Date('2017-12-30'))
### 使用汇编语言实现闰年判断及相关功能 #### 1. 判断给定年份是否为闰年 为了判断某一年是否为闰年,在汇编语言中可以按照如下逻辑进行处理: - 如果年份能被4整除但不能被100整除,或者是能被400整除,则该年为闰年。 ```assembly section .data ; 定义数据段 year db ? ; 存储输入的年份 section .bss leap_year resb 1 ; 结果缓冲区用于存储是否为闰年的标志位 (0 or 1) section .text global _start _start: mov eax, [year] ; 将年份数字加载到EAX寄存器中 xor edx, edx ; 清零EDX准备做除法运算 div dword 4 ; EAX = 年/4 EDX=余数 cmp edx, 0 ; 检查是否有余数 jne not_divisible_by_4 ; 若不是则跳转至not_divisible_by_4标签处执行后续指令 ; 否则是继续向下执行即满足第一个条件 push eax ; 保护当前商值以便稍后恢复它 mov eax, [year] xor edx, edx div dword 100 ; 测试能否被100整除 pop eax ; 恢复原来的商值 cmp edx, 0 je divisible_by_100 ; 是的话就去测试下一个条件 jmp set_leap_flag_true ; 不是那么就是闰年了可以直接设置标记 divisible_by_100: xor edx, edx div dword 400 ; 继续测试能否被400整除 cmp edx, 0 jne set_leap_flag_false ; 非400倍数则非闰年 set_leap_flag_true: mov byte [leap_year], 1 jmp end_check not_divisible_by_4: mov byte [leap_year], 0 end_check: ``` 这段代码实现了基本的闰年检测算法并将其结果保存到了`leap_year`变量中[^1]。 #### 2. 计算特定日期是一年的第几天 对于计算某个具体日子是一年的第几日,可以通过累加前几个月的日历天数再加上当月已过的天数来得出总。考虑到不同月份有不同的长度以及平年与闰年的差异,这里提供了一个简化版的方法: ```assembly ; 前置工作已完成,包括获取用户输入并将它们分别放入名为day、month year 的内存位置。 ; 此外还假设已经完成了上述关于如何确定一个给定年份是不是闰年的讨论部分。 section .rodata ; 只读数据段定义不可变常量表 days_in_month_non_leap dd 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 days_in_month_leap dd 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 section .bss ; 未初始化的数据段声明可写入空间 total_days resd 1 ; 总计天数暂存区域 ... mov ecx, [month] ; 把要查询的具体月份放到ECX寄存器里面 dec ecx ; 因为我们是从一月份开始累积所以需要减掉一个月 lea esi, [days_in_month_non_leap]; 默认指向非闰年的每月最大天数值列表地址 cmp byte [leap_year], 1 ; 查看之前得到的结果决定使用哪套表格 jz use_leap_table jmp sum_up ; 开始累加前面各个月份的全部天数直到到达目标月份为止 use_leap_table: lea esi, [days_in_month_leap] sum_up: xor ebx, ebx ; EBX用来记录循环次数也就是遍历了多少个元素 clc ; 清进位标志CF以备后面可能要用到 .LOOP_START: lodsd ; AL := *ESI++ 即取出当前位置上的那个四字节整型数送入AL寄存器内同时使指针前进四个单位距离 add [total_days], eax ; 更新总计天数 inc ebx ; 循环计数器自增一次 loopnz .LOOP_START ; 当EBX不等于零时重复执行上面两行命令直至结束整个循环体 add dword [total_days], [day] ; 加上本月实际经过的日子数量从而获得最终答案 ``` 此段程序能够根据传入的参数正确地计算出指定日期对应于当年中的确切序号,并考虑到了是否存在额外的一天(二月末)。这一步骤同样依赖于先前设定好的`leap_year`状态来进行适当调整。 #### 3. 输出每年度每个月末尾那天对应的年度累计天数 最后,为了输出每月底当天在整个自然年内所占据的位置,可以在之前的逻辑基础上进一步扩展,构建一个新的函数或子例程专门负责这项任务。下面给出了一种可行的方式: ```assembly section .rodata months_names db "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec" output_format db "%s ends at day %u of the year.", 0xA, 0xD, 0 section .bss monthly_endings resd 12 ; 数组用于储存各个月份结尾那一天在全年里的编号 ... call check_if_leap_year ; 调用前述方法先确认这一年到底属不属于闰年情况之一 lea edi, [monthly_endings] ; 获取目的数组首地址供之后填充之需 mov ecx, 1 ; 初始化起始索引为1代表从一月份起步 clear_total_days: ; 设置初始累计天数为0 xor eax, eax stosd inc ecx cmp ecx, 13 ; 是否超出范围? jl clear_loop_continue je done_clearing clear_loop_continue: jmp clear_total_days done_clearing: mov ecx, 1 calculate_each_month_ends: pushad mov esi, [days_in_month_non_leap] cmp byte [leap_year], 1 jz switch_to_leap_version jmp start_addition switch_to_leap_version: mov esi, [days_in_month_leap] start_addition: lodsd add [edi-4*ecx+4], eax popad inc ecx cmp ecx, 13 jb calculate_each_month_ends print_results: mov ecx, 1 .LOOP_PRINT: pusha lea eax, [output_format] push eax lea eax, [months_names+(ecx-1)*4] push eax mov eax, [edi-4*(ecx)+4] push eax call printf add esp, 12 popa inc ecx cmp ecx, 13 jb .LOOP_PRINT ret ``` 以上代码片段展示了怎样利用汇编语言创建一个完整的流程图,不仅涵盖了对单个日期属性的分析,还包括了针对整个公历年周期特性的全面解析。通过这种方式,不仅可以有效地解决原始问题提出的各项需求,而且还能确保所有必要的细节都得到了妥善处理。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值