web-dev-qa-db-ja.com

月の最初と最後の日

月と年の2つの数値が与えられた場合、その月の最初と最後の日をどのように計算できますか?私の目標は、次の3行を出力することです。

  1. 月/年(テキスト形式の月ですが、それは簡単です)

  2. 月の各日:現在の曜日の名前:金。 &土。 & 太陽。 [...]

  3. 月内の日番号:1&2&3 [...]&28&..?

GNU dateまたはBSD date(OS X))を使用した解決策を探しています。

6
alecail

しばらく前に私は同様の問題を抱えていました。私の解決策があります:

 $ ./get_dates.sh 2012 07
The first day is 01.2012.07, Sunday
The last day is 31.2012.07, Tuesday
 $ cal
 July 2012
Su Mo Tu We Th Fr Sa
 1  2  3  4  5  6  7
 8  9 10 11 12 13 14
15 16 17 18 19 20 21
22 23 24 25 26 27 28
29 30 31

スクリプト自体:

#!/bin/bash
# last day for month
lastday()  {
#                ja   fe   ma   ap   ma   jn   jl   ag   se   oc   no   de
  mlength=('xx' '31' '28' '31' '30' '31' '30' '31' '31' '30' '31' '30' '31')

  year=$1
  month=$2

  if [ $month -ne 2 ] ; then
    echo ${mlength[$month]}
    return 0
  fi

  leap=0
  ((!(year%100))) && { ((!(year%400))) && leap=1 ; } || { ((!(year%4))) && leap=1 ; }

  feblength=28
  ((leap)) && feblength=29
  echo $feblength
}

# date to Julian date
date2jd() {

  year=$1
  month=$2
  day=$3
  lday=$(lastday $year $month) || exit $?

  if ((day<1 || day> lday)) ; then
    echo day out of range
    exit 1
  fi

  echo $(( jd = day - 32075
                + 1461 * (year + 4800 - (14 - month)/12)/4
                        + 367 * (month - 2 + (14 - month)/12*12)/12
                    - 3 * ((year + 4900 - (14 - month)/12)/100)/4
                - 2400001 ))
}

jd2dow()
{
  days=('Sunday' 'Monday' 'Tuesday' 'Wednesday' 'Thursday' 'Friday' 'Saturday')

  jd=$1
  if ((jd<1 || jd>782028)) ; then
    echo julian day out of range
    return 1
  fi

  ((dow=(jd+3)%7))

  echo ${days[dow]}
}


echo -n "The first day is 01.$1.$2, "
jd2dow $(date2jd $1 $2 01)
echo -n "The last day is $(lastday $1 $2).$1.$2, "
jd2dow $(date2jd $1 $2 $(lastday $1 $2))

必要なマシンにGNU dateがなかったので、日付で解決しませんでした。もっと美しい解決策があるかもしれません。

3
rush
# Last month:
l_first_date=$(date -d "`date +%Y%m01` -1 month" +%Y-%m-%d)
l_last_date=$(date -d "`date +%Y%m01` -1 day" +%Y-%m-%d)

# This month:
t_first_date=$(date -d "`date +%Y%m01`" +%Y-%m-%d)
t_last_date=$(date -d "`date +%Y%m01` +1 month -1 day" +%Y-%m-%d)

# Next month:
n_first_date=$(date -d "`date +%Y%m01` +1 month" +%Y-%m-%d)
n_last_date=$(date -d "`date +%Y%m01` +2  month -1 day" +%Y-%m-%d)

# Print everything
echo "Last month: $l_first_date to $l_last_date"
echo "This month: $t_first_date to $t_last_date"
echo "Next month: $n_first_date to $n_last_date"
8
apseyy

私は正直になります。あなたが質問をしている方法から、私はあなたがいくつかの宿題を割り当てられたという感覚を得ます、それで私は読者のための練習として答えからいくつかのステップを残します:

dateのマニュアルページをよく見るとよいでしょう。特に-dフラグを使用すると、特定の日を調べることができます。 Y年のM月の最初の日は "M/01/Y"

last月の日を取得するには、指定された月の数に1を加え、日付の1日を差し引くことが最善の策です。

ヒント:dateは、実際にはいくつかの広範な演算を受け入れることができます。たとえば、date -d "01/07/2012 + 1 month - 1 day"と言うと、正しい答えが得られます。

2)および3)で必要な出力を表示する方法は、dateマンページの「フォーマット」セクションを調べることで確認できます。

ヒント:%a%dをご覧ください

7
Shadur
**Previous Month Start and End date**

month_year=$(date +'%m %Y' | awk '!--$1{$1=12;$2--}1')
m=${month_year% *}
y=${month_year##* }
d=$(cal $m $y | paste -s - | awk '{print $NF}')
first_date=$(printf '01-%02s-%s' $m $y)
last_date=$(printf '%s-%02s-%s' $d $m $y)
echo $first_date $last_date

**Currunt Month Start and End date**

month_year=$(date +'%m %Y' | awk '!$1{$1=12;$2--}1')
m=${month_year% *}
y=${month_year##* }
d=$(cal $m $y | paste -s - | awk '{print $NF}')
first_date=$(printf '01-%02s-%s' $m $y)
last_date=$(printf '%s-%02s-%s' $d $m $y)
echo $first_date $last_date

**Next Month Start and End date**

month_year=$(date +'%m %Y' | awk '!++$1{$1=12;$2--}1')
m=${month_year% *}
y=${month_year##* }
d=$(cal $m $y | paste -s - | awk '{print $NF}')
first_date=$(printf '01-%02s-%s' $m $y)
last_date=$(printf '%s-%02s-%s' $d $m $y)
echo $first_date $last_date
3
date -d "20121101 + 1 month - 1 day" +%Y%m%d
2
omri