Tantangan jam pasir


43

Jam pasir

Jam pasir ini memiliki 60 "pasir waktu", dan sepenuhnya mengisi setiap ruang. Jam memiliki lebar 19 karakter dan tinggi 13 karakter. Setiap ruang memiliki 5 baris pasir dan ada baris di tengah yang dapat menampung 1 pasir. Baris atas dapat menampung 17 pasir, 15 berikutnya dan seterusnya (lihat di bawah). Pasir jatuh ke ruang bawah dengan kecepatan satu pasir per detik.

       START                     3 SECONDS LATER               38 SECONDS LATER

███████████████████            ███████████████████            ███████████████████
█.................█   17       █..............   █   14       █                 █    0
██...............██   15       ██...............██   15       ██               ██    0
███.............███   13       ███.............███   13       ███.......      ███    7  
█████.........█████   09       █████.........█████   09       █████.........█████   09 
███████.....███████   05       ███████.....███████   05       ███████.....███████   05 
█████████.█████████   01       █████████.█████████   01       █████████.█████████   01 
███████     ███████            ███████  .  ███████            ███████  .  ███████  
█████         █████            █████    .    █████            █████    .    █████   
███             ███            ███      .      ███            ███....         ███
██               ██            ██               ██            ██...............██  
█                 █            █                 █            █.................█  
███████████████████            ███████████████████            ███████████████████   

Tantangan

Perlihatkan jam pasir (tidak perlu angka atau judul) setelah periode waktu tertentu (0 ≤ t ≤ 60).

Aturan Pengkodean

  1. Jam pasir harus terlihat persis seperti yang ditunjukkan di sini. Anda dapat mengganti karakter dan / atau .karakter dengan apa pun yang Anda suka agar sesuai dengan bahasa Anda (Unicode, masalah kompatibilitas ASCII).
  2. Input harus berupa angka seperti 45 atau 7. Tampilkan jam setelah sekian banyak detik ini.
  3. Output dapat ditampilkan atau disimpan ke file. Tidak diperlukan teks atau label tambahan seperti yang ditunjukkan di atas - hanya jam pasir yang kita butuhkan.
  4. Jika pengguna memasukkan t> 60, Anda tidak perlu menangani kesalahan.

Poin

  1. Kode terpendek menang.

Saya tidak tahu apakah itu hanya saya, tetapi jam pasir terlihat sangat berbeda di editor daripada di pos itu sendiri. Itu mungkin karena karakter █ tidak ditampilkan dengan lebar yang sesuai.
Dennis

1
Apakah sebutir pasir menyentuh lapisan saat ini, lalu sejajarkan ke kiri satu detik kemudian?
xnor

2
Apa yang harus menjadi output untuk t == 59?
edc65

3
Selamat telah membuat daftar Pertanyaan Jaringan Panas Stack Exchange!
Alex A.

1
Ini sebenarnya bukan jawaban, karena tidak sesuai dengan spesifikasi, tetapi lihat di sini untuk pengiriman di Desmos. (Tekan tombol putar dan atur kecepatan ke 0,15x untuk ~ 1 unit / detik)
Conor O'Brien

Jawaban:


27

JavaScript ( ES6 ), 203 208 233 270 256 karakter

Edit Direvisi menggunakan perulangan alih-alih urutan panggilan.

Edit baris atas dan bawah yang ditambahkan yang hilang.

Suatu fungsi mengembalikan output. Jalankan cuplikan di Firefox untuk menguji.

f=w=>[h='█'[R='repeat'](19),...[17,15,13,9,5,1,5,9,13,15,17].map((d,i)=>(t=i>5?-v:v,v-=i<5?d:1-d,e=' '[R](d/2),b='█'[R](10-d/2),b+('.'[R](t<d&&d-t)+e+' .'[i>4&w>i-6&t>=d|0]+e).slice(0,d)+b),v=w),h].join`
`

// Less golfed
F=
w=>[h='█'.repeat(19),
    ... [17, 15, 13, 9, 5, 1, 5, 9, 13, 15, 17].map(
     (d,i) => (
       t = i>5 ? -v : v,
       v -= i<5 ? d : 1-d, 
       e = ' '.repeat(d / 2),
       b = '█'.repeat(10 - d / 2),
       b + 
        ('.'.repeat(t < d && d - t) 
         + e + ' .'[i > 4 & w > i-6 & t >= d | 0]
         + e).slice(0,d)
       + b
     ), v = w
    ),
    h].join('\n')

// TEST            

O.innerHTML=f(+I.value)

function tick(d) {
  var i=+I.value+d
  I.value=i
  O.innerHTML=f(i)
}

var int=0;

function auto()
{
  function go()
  {
    var t = I.value;
    O.innerHTML=f(++t)
    if (t>70)t=0;
    I.value = t;
  }
  if (A.checked && !int)
  { 
    int = setInterval(go, 200);
  }
  else if (!A.checked && int)
  {
    clearInterval(int);
    int = 0;
  }
}
input[type=text] { width: 3em }
<button onclick='tick(-1)'>-</button>
<input type=text id=I value=0 onchange='tick(0)' >
<button onclick='tick(1)'>+</button>
<input type=checkbox id=A onclick='auto()'>Fly time
<pre id=O><pre>


5
Untuk keingintahuan saya, mengapa hal ini dibatalkan dua kali? Saya tidak dapat menemukan masalah dengan itu.
manatwork

Jawaban ini luar biasa ! Buka di Firefox, teman!
Cristian Lupascu

1
Jawaban yang bagus, tetapi bagian atas dan dasar jam pasir sepertinya tidak ada.
Renae Lider

@Manatwork tidak yakin (tentu saja) tetapi memang ada cacat memang
edc65

Tolong buat setInterval (), javascript akan bagus untuk itu:setTimeout( tick, 1000);
Martijn

18

Python 2, 200

t=input()+1
s=' '*t+'.'*60+' '*70
n=0
d=sum((1<t<56,2<t<48,3<t<36,4<t<22))
for c in'ctrplhdhlprtc':i=ord(c)-99;print[s[n+i:n:-1],[s[180-n-i+d:][:i],'.'][5+d*3>i>0]][n>59].center(i).center(19,'#');n+=i

xnor telah membuat versi 197 byte dalam obrolan .

Saya akan memposting penjelasan, tapi saya lupa bagaimana cara kerjanya ...

Juga, inilah versi animasi dengan kutukan:

jam pasir

from curses import*
w=initscr()

for t in range(1,61):
    s=' '*t+'.'*60+' '*70
    n=0
    d=sum((1<t<56,2<t<48,3<t<36,4<t<22))
    for i in 0,17,15,13,9,5,1,5,9,13,15,17,0:w.addstr([s[n+i:n:-1],[s[180-n-i+d:][:i],'.'][5+d*3>i>0]][n>59].center(i).center(19,'#')+"\n");n+=i
    w.refresh()
    w.clear()
    napms(999)

endwin()

1
kudos, Anda membuat saya menemukan bahwa string memiliki metode center ()!
pelaku diet

13

Python 2.7, 362 356 347

e,c,x,a,b,n=' .#ab\n';R,r,s,l,T,m=range,str.replace,'',19,[1,2,3,5,7,9],-1
for t in[T,T[:m][::m]]:s+=''.join([n+x*y+c*(l-y*2)+x*y for y in t]);c=b
s=list(s)
for i in R(130,220,20):s[i]=a
for _ in R(input()):s[s.index('.')]=e;i=s.index(a)if a in s else 219-s[::m].index(b);s[i]='.'
for l in(x*l+r(r(''.join(s),a,e),b,e)+n+x*l).split(n):print l[::m]

jam pasir

Output pada 38 detik:

###################
#                 #
##               ##
###.......      ###
#####.........#####
#######.....#######
#########.#########
#######  .  #######
#####    .    #####
###...   .      ###
##...............##
#.................#
###################

2
Apa jenis kode Python itu? Saya akan senang belajar kode seperti mereka. Harap membimbing ke arah yang benar.
Rajat Saxena

1
Saya tidak yakin apakah ini jawaban yang valid. Jika Anda melihat contoh yang dibuatnya selama 38 detik, Anda dapat melihat bahwa titik terendah yang masih jatuh tidak berada di garis di mana titik-titiknya sejajar. (di baris ke-4 dari bawah, di tengah)
RononDex

3
@RononDex: Seperti yang dijelaskan dalam komentar pertanyaan, urutan pengisian tidak begitu ketat - implementasi saya menempelkan butiran pasir ke kiri dengan cara yang sedikit berbeda dari pertanyaan, tetapi masih berlaku
diet

3

C 544

Inilah yang saya miliki sejauh ini untuk solusi C.

#include <stdio.h>
int main(int z,char **a){int r,i,y=i=0,v,d,t,m,s=atoi(a[1]),n[13]={0,43,28,15,6,1,0,1,5,13,25,39,0};char H[13][20];while(y<13){int x,b=x=i=0;v=y-6;t=3+abs(v);m=2*abs(v);d=t<m?t:m;d=9-d;if(d==0)d=10;while (b<d){H[y][b]='#';H[y][18-b]='#';b++;}while(x<19-2*b){if(x<=s-n[y])H[y][x+b]=v>0?' ':'.';else H[y][x+b]=v>0?'.':' ';x++;}if(s>58)r=0;else if(s==58)r=1;else if(s==57)r=2;else if(s==56)r=3;else if(s>38)r=4;else if(s>24)r=3;else if(s>12)r=2;else if(s>4)r=1;while(i<r){H[7+i][9]='.';i++;}H[y][19]='\n';y++;}fputs(H,stdout);}

Dikompilasi dengan perintah berikut:

gcc -w -o hourglass hourglass.c  // I realize I should have cast H as a char *, but since it works this way, I just decided to suppress the error from the compiler instead to save space.

Diakui, bahasa ini memiliki banyak sekali- termasuk pernyataan adalah sedikit cacat keluar dari blok, tapi saya benar-benar hanya mencari alasan untuk berlatih menggunakan C.

Saya harap Anda menyukai solusi saya, dan beri tahu saya jika Anda melihat cara untuk meningkatkan.


2
Ada banyak cara untuk meningkatkan, membuatnya lebih buruk C tetapi jauh lebih pendek.
edc65

1
Petunjuk: foralih-alih where, ?:alih-alih if else, variabel int global tidak perlu int, includetidak diperlukan. (Saya suka matematika Anda)
edc65

1
317: r,i,y,x,b,d,n[]={0,43,28,15,6,1,0,1,5,13,25,39,0};char H[14][20];main(s,a)char**a;{for(s=atoi(a[1]);y<13;){b=x=i=0;d=abs(y-6);d+=3<d?3:d;d=9-d?9-d:10;r=s>55?59-s:s>38?4:s>24?3:s>12?2:s>4?1:r;for(;b<19;)H[y][b++]=35;for(;x<19-2*d;x++)H[y][x+d]=(x>s-n[y]?" .":". ")[y>6];for(;i<r;)H[7+i++][9]=46;H[y++][b]=10;}puts(H);}(waspadalah, karakter aneh yang tak terlihat ditambahkan ke komentar panjang seperti ini)
edc65

3

Matlab, 252 byte

Idenya adalah membangun sebuah matriks yang terlihat seperti ini:

0   0   0   0   0   0   0   0   0   0   0   0   0   0   0   0   0   0   0
0  17  16  15  14  13  12  11  10   9   8   7   6   5   4   3   2   1   0
0   0  32  31  30  29  28  27  26  25  24  23  22  21  20  19  18   0   0
0   0   0  45  44  43  42  41  40  39  38  37  36  35  34  33   0   0   0
0   0   0   0   0  54  53  52  51  50  49  48  47  46   0   0   0   0   0
0   0   0   0   0   0   0  59  58  57  56  55   0   0   0   0   0   0   0
0   0   0   0   0   0   0   0   0  60   0   0   0   0   0   0   0   0   0
0   0   0   0   0   0   0 116 117  61 118 119   0   0   0   0   0   0   0
0   0   0   0   0 108 109 110 111  62 112 113 114 115   0   0   0   0   0
0   0   0  96  97  98  99 100 101  63 102 103 104 105 106 107   0   0   0
0   0  82  83  84  85  86  87  88  64  89  90  91  92  93  94  95   0   0
0  66  67  68  69  70  71  72  73  65  74  75  76  77  78  79  80  81   0
0   0   0   0   0   0   0   0   0   0   0   0   0   0   0   0   0   0   0

Dari sana mudah untuk mengisi entri dengan string yang bergantung n(mengisi semua entri yang lebih besar dari ndan lebih kecil dari n+60dengan titik-titik)

function c=f(n);
b=zeros(13,19);
z=[0,17,32,45,54,59];
y=-2:3;
for k=2:6;
    d=k+sum(k>4:5);
    b(k,d:20-d)=z(k):-1:z(k-1)+1;
    b(14-k,d:19-d)=68+(z(k-1):z(k)-2)-k;
end;
b(8:12,11:19)=b(8:12,10:18);
b(7:12,10)=60:65;c=[ones(13,19)*32,''];
c(~b)='¶';c(n<b)=46;c(b>n+60)=32

Karena n=38kita mendapatkan hasil ini:

¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶
¶                 ¶
¶¶               ¶¶
¶¶¶.......      ¶¶¶
¶¶¶¶¶.........¶¶¶¶¶
¶¶¶¶¶¶¶.....¶¶¶¶¶¶¶
¶¶¶¶¶¶¶¶¶.¶¶¶¶¶¶¶¶¶
¶¶¶¶¶¶¶  .  ¶¶¶¶¶¶¶
¶¶¶¶¶    .    ¶¶¶¶¶
¶¶¶...   .      ¶¶¶
¶¶...............¶¶
¶.................¶
¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶¶

2

Java, 712

Input diambil dari baris perintah. Menangani nilai negatif untuk waktu dan juga lebih besar dari yang dibutuhkan.

enum H{;public static void main(String[]r){int x=0,y=0,z=0,l,t=Integer.parseInt(r[0]);String b="",d="█",e=" ",f=".",n="\n",j,k,a="███████████████████"+n;int[]w={17,15,13,9,5},v;int[][]h=new int[10][];for(;x<5;){l=w[x];v=(h[x++]=new int[l--]);l/=2;v[l]=++z;for(y=0;y++<l;){v[l-y]=++z;v[l+y]=++z;}}for(z=0;x>0;){l=w[--x];v=(h[9-x]=new int[l--]);v[l/2]=++z;}for(;x<5;){l=(w[x]-1)/2;v=h[9-x++];for(y=0;y++<l;){v[l-y]=++z;v[l+y]=++z;}}p(a);for(x=0;x<5;x++){l=w[x];j=b;for(y=0;y++*2<19-l;)j+=d;k=b;for(y=0;y<l;)k+=t<h[x][y++]?f:e;p(j+k+j+n);}j="█████████";p(j+f+j+n);for(;x>0;){l=w[--x];j=b;for(y=0;y++*2<19-l;)j+=d;k=b;for(y=0;y<l;)k+=t<h[9-x][y++]?e:f;p(j+k+j+n);}p(a);}static void p(String s){System.out.print(s);}}

keluaran:

time: 0
███████████████████
█.................█
██...............██
███.............███
█████.........█████
███████.....███████
█████████.█████████
███████     ███████
█████         █████
███             ███
██               ██
█                 █
███████████████████

time: 1
███████████████████
█........ ........█
██...............██
███.............███
█████.........█████
███████.....███████
█████████.█████████
███████  .  ███████
█████         █████
███             ███
██               ██
█                 █
███████████████████

time: 9
███████████████████
█....         ....█
██...............██
███.............███
█████.........█████
███████.....███████
█████████.█████████
███████  .  ███████
█████    .    █████
███      .      ███
██       .       ██
█      .....      █
███████████████████

time: 41
███████████████████
█                 █
██               ██
███..         ..███
█████.........█████
███████.....███████
█████████.█████████
███████  .  ███████
█████    .    █████
███   .......   ███
██...............██
█.................█
███████████████████

Mengisi pasir dari pusat yang mengembang ke arah luar. Saya mungkin bisa bermain golf lebih banyak jika saya malas bagaimana mengisi bagian bawah dan mengosongkan bagian atas. Tetapi untuk saat ini saya sangat menyukainya.


2

Haskell 512 Bytes

h=[17,15,13,9,5,1];b=drop 1$reverse h;n#s=[1..n]>>[s];s='.';n =' ';c q=(q#n)++(60-q)#s;f q|q<=5=q#s|3>2=g#s where{g=foldl i 5 (scanl (+) 0 h);i x y=if q>x+y then x-1 else x};e q=j#s++(59-length(k q)-(j))#n where{j=q-length(f q)};l q=c q++k q++(reverse$e q);p _ []=[];p x y=reverse(z++take q x++z):p (drop (head y) x) (tail y)where{q=head y;z=replicate(div (19-q) 2) '|'};k q= (concat.map(\x -> z x ++ "." ++ z x).take (length.f$q)$b)where{z x=(div x 2)#n};m n=mapM_ putStrLn $ t ++ p (l n) (h++b) ++ t;t=[19#'|'] 

Memasukkan m 55

Keluaran

|||||||||||||||||||
|                 |
||               ||
|||             |||
|||||         |||||
|||||||.... |||||||
|||||||||.|||||||||
|||||||  .  |||||||
|||||.........|||||
|||.............|||
||...............||
|.................|
|||||||||||||||||||

Input m 48 Output

    |||||||||||||||||||
    |                 |
    ||               ||
    |||             |||
    |||||......   |||||
    |||||||.....|||||||
    |||||||||.|||||||||
    |||||||  .  |||||||
    |||||..       |||||
    |||.............|||
    ||...............||
    |.................|
    |||||||||||||||||||

1
Selamat datang di PPCG! Saya melihat Anda sudah bermain golf ini cukup banyak, tetapi masih ada banyak ruang putih yang tidak perlu, misalnya. sekitarnya ++. Anda juga dapat menyimpan beberapa byte dengan menentukan sebelumnya lengthdan reverse.
Laikoni

1

C #, 382 410

masih mungkin untuk mengurangi beberapa byte ...

class Program{static void Main(){int u=60-22,d=u,i,j,k,l,m;var c=new char[260];var r=new int[]{0,17,15,13,9,5,1,5,9,13,15,17,0,54,45,32,17,0};for(i=0;i<13;){m=0;l=(19-r[i])/2-1;for(j=19;j>=0;){k=i*20+j--;var b=j>=l&&j<r[i]+l;if(i>6&b)c[k-r[i]+m++ +m]=r[i+6]<d&&d-->0||j==8&r[i+6]>d&&d-->0?'.':' ';else c[k]=i<7&b?u-->1?' ':'.':'█';}c[++i*20-1]='\n';}System.Console.WriteLine(c);}}

Fiddler - 38sec


Keren! Bagaimana cara input?
isaacg

@isaacg, oops, saya akan memperbaikinya hari ini, saya tahu itu terlalu pendek untuk solusi ac # :-) untuk saat ini ubah u = 60-22 untuk sesuatu yang lain
Fredou

Mengapa Anda menggunakan int u=60-22dan bukan hanya int u=38??
dev-masih

1

Ruby: 196 190 186 185 184 karakter

u=[0,17,15,13,9,5].map{|i|(?.*i).center 19,?#}*$/
(?1..$*[0]).map{u[?.]=' '}
l=u.reverse
5.times{|i|l[p=i*20+9]==?.&&l[' ']&&(l[p]=?|)&&l[' ']=?.}
puts u,?#*9+?.+?#*9,l.tr('. | ',' .')

CW karena tidak sesuai persis dengan sampel yang diposting karena ini mengkonsumsi pasir mulai dari kiri. Sebagian besar hanya demonstrasi String.[]=metode.

Contoh dijalankan:

bash-4.3$ ruby hg.rb 38
###################
#                 #
##               ##
###      .......###
#####.........#####
#######.....#######
#########.#########
#######  .  #######
#####    .    #####
###      .   ...###
##...............##
#.................#
###################

Ruby: 215 karakter

Ini menghasilkan output yang dibutuhkan tepat:

u=[0,17,15,13,9,5].map{|i|(?.*i).center 19,?#}*$/
(?1..$*[0]).map{u[?.]=' '}
l=u.reverse
5.times{|i|l[p=i*20+9]==?.&&l[' ']&&(l[p]=?|)&&l[' ']=?.}
puts ([u,?#*9+?.+?#*9,l.tr('. | ',' .')]*$/).split($/).map &:reverse

Contoh dijalankan:

bash-4.3$ ruby hg.rb 38
###################
#                 #
##               ##
###.......      ###
#####.........#####
#######.....#######
#########.#########
#######  .  #######
#####    .    #####
###...   .      ###
##...............##
#.................#
###################
Dengan menggunakan situs kami, Anda mengakui telah membaca dan memahami Kebijakan Cookie dan Kebijakan Privasi kami.
Licensed under cc by-sa 3.0 with attribution required.