Покоління лінійки ASCII


11

Завдання полягає у створенні лінійки лічильників ASCII символів у такому форматі:

+-------------------------------------------------------------------------+
||    |    |    |    |    |    |    |    |    |    |    |    |    |    |  |
|0         10        20        30        40        50        60        70 |
|                                                                         |
+-------------------------------------------------------------------------+

Довжина лінійки повинна бути масштабованою до будь-якого кратного 10.

Правила:

  • Лінійка повинна бути горизонтальною.
  • Скажімо, для спрощення речей, що лінійка повинна працювати до позначки 1000, усі числа вирівняні вліво, один пробіл після остаточного числа .
  • Просити лінійку з нульовою довжиною слід надрукувати NaR
  • Негативні цифри повинні друкувати перевернуту лінійку (-90, -80, -70, ...) з правильними вирівняними номерами в точно такому ж форматі, що і вище

І друк на консолі - це добре

Готовий, набір, гольф!


Як програма знає, як довго зробити лінійкою? Чи надається вона як номер на STDIN?
PhiNotPi

Так, як зазвичай я здогадуюсь!
user2398029

1
Чи повинна лінійка бути горизонтальною чи вертикальною? якщо він горизонтальний, чи можемо ми припустити максимум 80 символів, чи виводимо це у файл, щоб уникнути обгортання? чи нам потрібно змінити розмір консолі / терміналу під час запису до stdout?
Блейзер

5
Одним з головних моментів коду-гольфу є те, що проблема повинна бути на 100% однозначно визначена (див. Faq ). У фактичному виході мало місця для творчості, адже творчість полягає в самому коді. Надалі будь ласка, спробуйте продумати всі ці можливі випадки, перш ніж
надсилати

3
Чи можете ви відредагувати завдання, щоб включити правила, зазначені в коментарях?
флеск

Відповіді:


8

Пітон - 227 232

Підтримує всю специфікацію

редагувати: покращене вираження генератора.

Підтримка правильних вирівняних від’ємних чисел додає дивовижної кількості коду.

b,p,d,w,='|+- '
g=input
s=''.join(('%'+d[:i>0]+'10s')%i+['',9*w][i==0] for i in range(g(),g()+1,10)).strip()+w
m,n=s[0]==d and s.find(w)-1,len(s)
t=p+n*d+p
print['\n'.join([t,b+(w*m+'|    '*n)[:n]+b,b+s+b,b+n*w+b,t]),'NaR'][n<9]

Приклади виходів:

-30 30

+-----------------------------------------------------------------+
|  |    |    |    |    |    |    |    |    |    |    |    |    |  |
|-30       -20       -10         0         10        20        30 |
|                                                                 |
+-----------------------------------------------------------------+

-30 -30

NaR

100 150

+------------------------------------------------------+
||    |    |    |    |    |    |    |    |    |    |   |
|100       110       120       130       140       150 |
|                                                      |
+------------------------------------------------------+

-1000 -950

+--------------------------------------------------------+
|    |    |    |    |    |    |    |    |    |    |    | |
|-1000      -990      -980      -970      -960      -950 |
|                                                        |
+--------------------------------------------------------+

10

Не збираюся бити динамічні мови сьогодні, але все одно ...

Хаскелл, 341

import Data.List
main=interact$unlines.m.map read.words
m[l,r]|r>l=ᴛ.("┌│││└":).(++["┬   ─","┐│││┘"]).ʀ.t.ʀ.t.takeWhile((>4).length).ᴛ$[c"┬",c"│    ",[l,l+10..r]>>=h.show,c" ",c"─"]|True=["NaR"]
h s=p s$length s;p s@('-':_)l=r(6-l)ꜱ++s++r 4ꜱ;p s l=r 5ꜱ++s++r(5-l)ꜱ
ᴛ=transpose;ʀ=reverse;r=replicate;c=cycle
ꜱ=' ';t l@(c:o)|c!!2==ꜱ=t o|True=l

Я взяв на себе можливість обмінятись фактичними символами ASCII з кращими символами для малювання коробки Unicode.

$ echo "-50 30" | runhaskell  def0.hs
┌┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┬┐
│  │    │    │    │    │    │    │    │    │    │    │    │    │    │    │    │    │  │
│-50       -40       -30       -20       -10         0         10        20        30 │
│                                                                                     │
└─────────────────────────────────────────────────────────────────────────────────────┘

2
Лінійка виглядає дуже, дуже приємно.
user2398029

3

Python 2.7, 342 266 260 символів

a,b,c,d,m='+|- \n'
def f(y):x=map(str,(range(0,y+1,10)if y>0 else range(y,1,10)));h,g=len(x[-1])+1,len(x)-1;u=a+(c*10)*g+c*h+a;return'NaR'if y==0 else u+m+b+(b+d*4)*2*g+b+d*(h-1)+b+m+b+''.join(i.ljust(10)for i in x[:-1])+x[-1].ljust(h)+b+m+b+(d*10)*g+d*h+b+m+u

повертає кордон кожного рядка рядка, який ви можете потім роздрукувати або зберегти у файл (я віддаю перевагу останній довжиною більше 70, оскільки консоль буде jsut, щоб вона виглядала переплутаною під час обгортання)

Передбачається, yщо це рядок (raw_input () n python або sys.argv [1], якщо ви хочете викликати через cmd) ціле число (наприклад, з введенням () в 2.x або int (вхід ()) у 3.x )

Я зробив цю функцію більш гнучким

редагування: зменшено до 266 символів. більше не повертає кортеж, а натомість рядок. Тепер як аргумент бере ціле число замість рядка

edit2: зменшено до 260 знаків, функція одного рядка

Примітка: чи обробляє негативні цифри, але не виправдовує обґрунтування (я дійсно не забуваю, що це виправдання занадто важливо


Або просто використовувати нерозривну консоль (наприклад, Terminator ).
перестали повертати проти годинника,

@leftaroundabout Я не був впевнений, що таке існує
Blazer

незважаючи на те, що встановити окрему платформу просто для того, щоб лінійка видалася безглуздою, де, оскільки створення зворотного значення, яке хто-небудь міг би використовувати (записувати у файл тощо), є більш універсальним
Blazer

@Blazer, ваша оболонка не має труб та переспрямувань?
Пітер Тейлор

1

PowerShell , 256 253 233 225 222 байти

param($a,$b)('NaR',$($d=@{}
$a..$b|%{$d.$_=' ';0..($l=($s="$($_-$_%10)").Length-1)|%{$d[$_+$s-$l*(0-gt$s)]=$s[$_]}}
$d.Keys|sort|%{$e+='-';$p+='|    '[$_%5];$r+=$d.$_;$w+=' '}
@"
+$e-+
|$p |
|$r |
|$w |
+$e-+
"@))[$a-lt$b]

Спробуйте в Інтернеті!

Менше гольфу:

param($a,$b)
(
    'NaR',
    $(
        # {key, value} := {position, digit|minus|space}
        $d=@{}
        $a..$b|%{
            $d.$_=' '

            # draw a nearest left number
            $n=$_-$_%10
            $s="$n"
            $l=$s.Length-1
            0..$l|%{
                $d[$_+$s-$l*(0-gt$s)]=$s[$_]
            }
        }

        # edge, points, ruler, whitespaces
        $d.Keys|sort|%{
            $e+='-'
            $p+='|    '[$_%5]
            $r+=$d.$_
            $w+=' '
        }

        # output the multiline string
@"
+$e-+
|$p |
|$r |
|$w |
+$e-+
"@
    )
)[$a-lt$b]

0

Пітон, 291 241 персонаж

Досить простий підхід. Я впевнений, що це можна вдосконалити досить багато.
Я намагався дотримуватися вказівок у коментарях, але я не підтримую негативні цифри (сподіваюся, це був жарт).
Це програма, яка друкує лінійку на стандартний вихід. Якщо ваш екран досить широкий, він повинен підтримувати досить довгі лінійки.

import sys
def p(b,a="|"):print a+b+a
j="".join
l=int(sys.argv[1])//10*10
if l:
    d=j(["%-10d"%n for n in range(0,l,10)])+"%d "%l
    L=len(d)
    h="-"*L
    p(h,"+")
    p(j(["|    "[n%5] for n in range(L)]))
    p(d)
    p(" "*L)
    p(h,"+")
else: print "NaR"

Ви можете пограти ще декількома персонажами з цього місця. Я поголив тут 12 символів
Гордон Бейлі,

Дякую @GordonBailey, але все одно вже є переможець, який коротший, ніж мій код, і підтримує повну специфікацію.
угорен

0

C ++, 392

Це запрограмовано за допомогою консолі Windows, тому я просто вибрав максимальний розмір лінійки - 70, він просто вийде з ладу для чогось більшого. Негативні цифри (до -70) та 0 обробляються правильно.

#include<ios>
#define q s[2][i
#define v ,memset(&s
char g,o,i,n,s[5][80];int main(int a,char**b){a=atoi(b[1])v,32,400);n=0<a;for(a=abs(a)v[0][1],45,a+3)v[4][1],45,a+3);i<a+4;++i)o=i-n,!(o%5)?s[1][n?i:i+3]='|',(o%2-1?(n?q]=i/10+48,i>9?q+1]=48:0:((a-i)>9?q+2]=(a-i)/10+48,q+1]=45:0,q+3]=48)):0):0;for(;g-5;g++)for(s[g][a+4]=s[g][i=0]=g&g-4?'|':43;i-80;i++)printf(a?"%c":g+i?"":"NaR",s[g][i]);}

0

Пітон - 208

(не підтримує вирівняні відмінні цифри)

 l,u=map(int,raw_input().split())
 n=u-l
 q="+%s+\n"
 q=q+"|%s|\n"*3+q
 print q%('-'*n,(''.join("|    "for i in range(n)))[:n],(''.join("{:<10d}".format(i)for i in range(l,u,10)))[:n],' '*n,'-'*n)if n>0 else"NaR"

Я думаю, що мій улюблений трюк полягав у тому, щоб створити рядки, довші ніж необхідні, а потім обрізати їх, наприклад:

 ''.join("|    "for i in range(n)))[:n]

Я просто хотів би, щоб був вигідніший варіант форматування рядків для вирівнювання вліво (якщо тільки немає такого, про який я просто не знаю)


0

Perl 5.14, 198 224 символів

Можливо, це може бути скорочено набагато далі, але ось перша спроба (з розривами рядків, вставлених для читабельності):

$l=shift||die"NaR\n";
@n=grep!($_%10),$l>0?0..$l:$l..0;
$l=9-length$n[$#n];
@f=('-'x10,'|    'x2,'@'.'<'x9,' 'x10);
@f=map$_ x@n,@f;
map$_=~s/.{$l}$//,@f;
eval"format=\n+$f[0]+\n|$f[1]|\n|$f[2]|\n\@n\n|$f[3]|\n+$f[0]+\n.\n";
write

EDIT: Відредаговано для відмирання з "NaR", коли вхід є 0та підтримує негативний діапазон.

EDIT2: У мене не було можливості більше працювати над цим, і я лише щойно помітив правильне вирівнювання за правилом від'ємних чисел, яке мій код не підтримує, тому я думаю, що інше рішення слід позначити як відповідь, якщо граничний термін досягнуто


@louism: моя відповідь не відповідає всім правилам. По можливості слід вибрати іншу відповідь.
флеск
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.