Преобразование десятичного времени дня

15

Вступление

Время сбивает с толку. От шестидесяти секунд до минуты, от шестидесяти минут до часа, от двадцати четырех часов до дня (и не говоря уже о том, что надоедливый am / pm!).

В настоящее время нет места для такой глупости, поэтому мы решили принять единственную разумную альтернативу: десятичные дни! То есть каждый день считается целой единицей, а все, что короче, записывается как десятичная дробь этого дня. Так, например: «12:00:00» будет записано как «0.5», а «01:23:45» может быть записано как «0.058159».

Поскольку для привыкания к новой системе потребуется время, вам поручено написать программу, которая может конвертировать между ними в обоих направлениях.

Вызов

Напишите программу на выбранном вами языке, которая, учитывая современное время в формате ISO-8601 «чч: мм: сс», будет возвращать эквивалентную десятичную дробную единицу. Аналогично, с учетом десятичной дроби программа должна возвращать время в современном формате, который был первоначально указан.

Вы можете сделать следующие предположения:

  • Современное время ввода и вывода может варьироваться от «00:00:00» до «24:00:00».
  • Время ввода и вывода десятичного времени может варьироваться от «0» до «1», и должно быть в состоянии принимать / выводить как минимум до 5 знаков после запятой (например, «0.12345»). Больше точности приемлемо
  • Программа должна быть в состоянии знать, какое направление преобразования выполнить на основе входных данных.
  • Вы не можете использовать связанные со временем функции / библиотеки

Победитель будет определен по кратчайшему коду, который соответствует критериям. Они будут выбраны не менее чем за 7 десятичных дней или если / когда было достаточно представлений.

Примеры

Вот (n намеренно) плохо написанный фрагмент кода JavaScript, который будет использоваться в качестве примера:

function decimalDay(hms) {
    var x, h, m, s;
    if (typeof hms === 'string' && hms.indexOf(':') > -1) {
        x = hms.split(':');
        return (x[0] * 3600 + x[1] * 60 + x[2] * 1) / 86400;
    }
    h = Math.floor(hms * 24) % 24;
    m = Math.floor(hms * 1440) % 60;
    s = Math.floor(hms * 86400) % 60;
    return (h > 9 ? '' : '0') + h + ':' + (m > 9 ? '' : '0') + m + ':' + (s > 9 ? '' : '0') + s;
}
decimalDay('02:57:46'); // 0.12344907407407407
decimalDay('23:42:12'); // 0.9876388888888888
decimalDay(0.5); // 12:00:00
decimalDay(0.05816); // 01:23:45
Mwr247
источник
Хм ... 60 почти 64. Интересно, на что было бы похоже время, если бы было 64 секунды в минуту и ​​64 минуты в час (и 16 или 32 часа в день).
1
Должны ли мы справиться с високосными секундами? так 23:59:60 это 1 секунда от конца 86401 второго дня?
Спарр
1
@ Sparr Не нужно беспокоиться о високосных секундах. Это будущее, где мы решили, что глупо считать секунду абсолютной величиной, а также привязывать ее к относительной скорости вращения Земли;)
Mwr247
1
@MichaelT Это был бы мир мечты программистов = P
Mwr247
1
@ Mwr247 да. DNS TTL имеет (имел?) Поле, nгде n2 ^ n секунд. Таким образом, значение «6» имеет TTL около 1 минуты. Значение «12» имело TTL около 1 часа. «15» было около 8 часов и так далее. Это позволило одному байту определить время ожидания и дать вам достаточно контроля для короткого или длительного времени.

Ответы:

6

CJam, 58 56 42 байта

Я уверен, что это слишком долго и можно много играть в гольф. Но здесь для начала:

86400q':/:d_,({60bd\/}{~*i60b{s2Ue[}%':*}?

Попробуйте онлайн здесь

оптимизатор
источник
Хех, у нас есть похожие идеи
aditsu
@aditsu Ой! Не видела твоего до обновления моего, а потом спешила на работу.
Оптимизатор
Вы знаете, что .. не стесняйтесь использовать мой код: 86400q':/:d_,({60bd\/}{~*mo60bAfmd2/':*}?я удаляю свой ответ. Это moтак, что 0.058159 преобразуется в 01:23:45
aditsu
3

Python 2, 159 150 141 + 2 = 143 байта

Простое решение, вероятно, может быть намного короче. Будет работать над этим.

Добавлены два байта для учета входных данных, которые необходимо заключить в "s". Кроме того, Sp3000 указал на проблему с интерпретацией восьмеричных значений eval () и показал способ сократить форматирование, использовать map () и удалить один отпечаток.

n=input();i=float;d=864e2
if':'in n:a,b,c=map(i,n.split(':'));o=a/24+b/1440+c/d
else:n=i(n);o=(':%02d'*3%(n*24,n*1440%60,n*d%60))[1:]
print o

Проверьте это на ideone здесь.

Када
источник
2

Javascript ( ES6 ), 116 110 байтов

f=x=>x[0]?([h,m,s]=x.split(':'),+s+m*60+h*3600)/86400:[24,60,60].map(y=>('0'+~~(x*=y)%60).slice(-2)).join(':')


// for snippet demo:
i=prompt();
i=i==+i?+i:i; // convert decimal string to number type
alert(f(i))

комментарии:

f=x=>
    x[0] ? // if x is a string (has a defined property at '0')
        ([h, m, s] = x.split(':'), // split into hours, minutes, seconds
        +s + m*60 + h*3600) // calculate number of seconds
        / 86400 // divide by seconds in a day
    : // else
        [24, 60, 60]. // array of hours, minutes, seconds
        map(y=> // map each with function
            ('0' + // prepend with string zero
                ~~(x *= y) // multiply x by y and floor it
                % 60 // get remainder
            ).slice(-2) // get last 2 digits
        ).join(':') // join resulting array with colons
nderscore
источник
24:00:00выдает 1но обратное неверно
rink.attendant.6
@ rink.attendant.6 исправлено
nderscore
2

Python 3: 143 байта

i,k,l,m=input(),60,86400,float
if'.'in i:i=m(i)*l;m=(3*':%02d'%(i/k/k,i/k%k,i%k))[1:]
else:a,b,c=map(m,i.split(':'));m=(a*k*k+b*k+c)/l
print(m)

Тот же счетчик байтов, что и в решении Python 2, но, похоже, мы использовали разные подходы к математике.

Serdalis
источник
2

Юлия, 152 143 142 байта

Что ж, я обновил свой подход, чтобы быть менее «юлианским», как говорится, ради игры в гольф. Для лучшего (хотя и менее лаконичного) подхода см. Историю изменений.

x->(t=[3600,60,1];d=86400;typeof(x)<:String?dot(int(split(x,":")),t)/d:(x*=d;o="";for i=t q,x=x÷i,x%i;o*=lpad(int(q),2,0)*":"end;o[1:end-1]))

Это создает безымянную функцию, которая принимает строку или 64-разрядное число с плавающей запятой и возвращает 64-разрядное число или строку с плавающей запятой соответственно. Чтобы назвать его, дайте ему имя, например f=x->....

Ungolfed + объяснение:

function f(x)
    # Construct a vector of the number of seconds in an hour,
    # minute, and second
    t = [3600, 60, 1]

    # Store the number of seconds in 24 hours
    d = 86400

    # Does the type of x inherit from the type String?
    if typeof(x) <: String
        # Compute the total number of observed seconds as the
        # dot product of the time split into a vector with the
        # number of seconds in an hour, minute, and second
        s = dot(int(split(x, ":")), t)

        # Get the proportion of the day by dividing this by
        # the number of seconds in 24 hours
        s / d
    else
        # Convert x to the number of observed seconds
        x *= d

        # Initialize an output string
        o = ""

        # Loop over the number of seconds in each time unit
        for i in t
            # Set q to be the quotient and x to be the remainder
            # from x divided by i
            q, x = divrem(x, i)

            # Append q to o, padded with zeroes as necessary
            o *= lpad(int(q), 2, 0) * ":"
        end

        # o has a trailing :, so return everything up to that
        o[1:end-1]
    end
end

Примеры:

julia> f("23:42:12")
0.9876388888888888

julia> f(0.9876388888888888)
"23:42:12"

julia> f(f("23:42:12"))
"23:42:12"
Алекс А.
источник
2

C, 137 байтов

Программа Full C Принимает ввод на стандартный вывод и выводит на стандартный вывод.

main(c){float a,b;scanf("%f:%f:%d",&a,&b,&c)<3?c=a*86400,printf("%02d:%02d:%02d",c/3600,c/60%60,c%60):printf("%f",a/24+b/1440+c/86400.);}

Развернулся и прокомментировал:

int main() {
    // b is float to save a . on 1440
    float a,b;
    // c is int to implicitly cast floats
    int c;

    // If the input is hh:mm:ss it gets splitted into a, b, c
    // Three arguments are filled, so ret = 3
    // If the input is a float, it gets stored in a
    // scanf stops at the first semicolon and only fills a, so ret = 1
    int ret = scanf("%f:%f:%d", &a, &b, &c);

    if(ret < 3) {
        // Got a float, convert to time
        // c = number of seconds from 00:00:00
        c = a * 86400;
        printf("%02d:%02d:%02d", c/3600, c/60 % 60, c%60);
    }
    else {
        // a = hh, b = mm, c = ss
        // In one day there are:
        // 24 hours
        // 1440 minutes
        // 86400 seconds
        printf("%f", a/24 + b/1440 + c/86400.);
    }
}
Андреа Биондо
источник
Очень четкое использование scanf и% f
некий пользователь
D'о! Я имел в виду "умный".
какой-то пользователь
2

J 85 байт

Результаты:

Т '12: 00: 00 '
0,5

Т 0,5
12 0 0

Т '12: 34: 56 '
0.524259

Т 0,524259
12 34 56

T=:3 :'a=.86400 if.1=#y do.>.(24 60 60#:y*a)else.a%~+/3600 60 1*".y#~#:192 24 3 end.'

Всего 85

Ричард Донован
источник
Добро пожаловать на сайт! Я отредактировал ваш пост, чтобы код отображался в виде кода. Что касается онлайн-ссылки, то лучшая, которую я знаю, это TIO . Я бы дал вам ссылку, но у меня нет опыта работы с J, поэтому я не знаю, как правильно ее вызвать. Кроме того, кажется, что это 91 байт, когда вы включаете первую и последнюю строки. Это верно?
DJMcMayhem
Спасибо за вашу помощь! Программа [a = ... to end.] 77. Заголовок 10. Терминатор 1, так что получается 88. При трех переводах строки получается 91! Я над этим поработаю: о)
Ричард Донован
Теперь до 85-байтового однострочного!
Ричард Донован,
1

Javascript, 194 192 190 188 байт

function(z){if(isNaN(z)){x=z.split(':');return x[0]/24+x[1]/1440+x[2]/86400}h=(z*24)|0;h%=24;m=(z*1440)|0;m%=60;s=(z*86400)|0;s%=60;return""+(h>9?'':0)+h+':'+(m>9?'':0)+m+':'+(s>9?'':0)+s}
SuperJedi224
источник
1

JavaScript ES6, 98 130 байт

s=>s==+s?'246060'.replace(/../g,l=>':'+('0'+~~(s*=+l)%60).slice(-2)).slice(1):s.split`:`.reduce((a,b)=>+b+(+a)*60)*1/864e2;f(0.5);
Downgoat
источник
К сожалению, функции, связанные со временем (такие как «Date» и «toTimeString»), не разрешены в этой задаче. В противном случае, это гораздо более краткий способ сделать это =)
Mwr247
@ Mwr247 о, не видел этого, тогда я это
исправлю
1

C 156 152 байта

Я думал, что это будет легко для C. Но все же получилось довольно большим. :(

n,m=60;d(char*s){strchr(s,58)?printf("%f",(float)(atoi(s)*m*m+atoi(s+3)*m+atoi(s+6))/m/m/24):printf("%02d:%02d:%02d",(n=atof(s)*m*m*24)/m/m,n/m%m,n%m);}

Тестовая программа:

#include <stdio.h>
#include <stdlib.h>

int n,m=60;
d(char*s)
{
    strchr(s,':') ? 
        printf("%f",(float)(atoi(s)*m*m+atoi(s+3)*m+atoi(s+6))/m/m/24):
        printf("%02d:%02d:%02d",(n=atof(s)*m*m*24)/m/m,n/m%m,n%m);
}

int main()
{
    d("01:23:45");
    printf("\n");
    d("02:57:46");
    printf("\n");
    d("23:42:12");
    printf("\n");
    d("12:00:00");
    printf("\n");
    d("0.5");
    printf("\n");
    d("0.05816");
    printf("\n");
    d("0");
    printf("\n");
    d("1");
    printf("\n");
    return 0;
}

Выход:

0.058160
0.123449
0.987639
0.500000
12:00:00
01:23:45
00:00:00
24:00:00
какой-то пользователь
источник
1

PHP, 70 69 байт

<?=strpos($t=$argv[1],58)?strtotime($t)/86400:date("H:i:s",$t*86400);

принимает входные данные из аргумента командной строки, печатает в STDOUT:

Если входные данные содержат двоеточие, преобразуйте в unix time и разделите на (секунды в день),
иначе умножьте числовое значение на (секунды в день) и отформатируйте время unix в hh:mm:ss.

Titus
источник
1

Perl 109 108 101 + 6 ( -plaF:флаг) = 107 байт

$_=$#F?($F[0]*60+$F[1]+$F[2]/60)/1440:sprintf"%02d:%02d:%02d",$h=$_*24,$m=($h-int$h)*60,($m-int$m)*60

С помощью:

perl -plaF: -e '$_=$#F?($F[0]*60+$F[1]+$F[2]/60)/1440:sprintf"%02d:%02d:%02d",$h=$_*24,$m=($h-int$h)*60,($m-int$m)*60' <<< 01:23:45

Попробуйте это на Ideone.

Денис Ибаев
источник
0

Excel, 178 байт

=IF(LEFT(A1,2)="0.",TEXT(FLOOR(A1*24,1),"00")&":"&TEXT(MOD(FLOOR(A1*1440,1),60),"00")&":"&TEXT(MOD(FLOOR(A1*86400,1),60),"00"),((LEFT(A1,2)*60+MID(A1,4,2))*60+RIGHT(A1,2))/86400)
Wernisch
источник