Табло гольфу


11

Гольф лідерів цього питання (id = 111735). Ваша програма повинна надіслати один запит HTTP або HTTPS до StackExchange API, проаналізувати його та подати користувачеві у формі, подібній до типового фрагменту Leaderboard

Вибірка зразка (для питання 47338 замість цього):

Leaderboard
    Author  Language    Size
1.  FryAmTheEggman  Pyth    19
2.  Peter Taylor    CJam    24
3.  Martin Ender    CJam    27
3.  Peter Taylor    GolfScript  27
5.  randomra    J   32
5.  Optimizer   CJam    32
7.  Timtech TI-Basic 83/84  40
8.  mollmerx    k   41
9.  Sp3000  ><> 45
10. Sherlock9   Ruby    47
11. Martin Ender    Mathematica 49
12. Alex A. Julia   57
13. Sp3000  Python 2    58
14. Zgarb   Haskell 68
15. Timtech GML 76
16. Jakube  Python 2    79
16. nimi    Haskell 79
18. coredump    Common Lisp 91
19. Jim Large   Ruby    92
Winners by Language
Language    User    Score
Python 2    Sp3000  58
CJam    Peter Taylor    24
Julia   Alex A. 57
TI-Basic 83/84  Timtech 40
><> Sp3000  45
Haskell Zgarb   68
GolfScript  Peter Taylor    27
Common Lisp coredump    91
Pyth    FryAmTheEggman  19
k   mollmerx    41
J   randomra    32
Ruby    Sherlock9   47
Mathematica Martin Ender    49
GML Timtech 76

Зверніть увагу на повторювані ранги 3, 5 і 16. Можливо, я навіть додам спеціальну неконкурентну відповідь лише для того, щоб примусити правильне, не спрощене вручення дублікатів балів.

Вихід повинен складатися з:

  1. Рядок "Табло"
  2. Рядок "\ tAuthor \ tLanguage \ tSize"
  3. Для кожної відповіді виділено вкладку рядка рангів і а ., потім відповідь ім'я автора, потім ім’я мови, потім оцінка; у порядку зростання за рахунком
  4. Рядок "Переможці за мовою"
  5. Рядок "Мова \ tкористувач \ tScore"
  6. Для кожної використовуваної мови назва мови, відокремлена вкладками, автор відповіді з нижчим балом та оцінка

Іншими словами, щось на зразок того, як якщо б один копіював і вставляв результат фрагменту таблиці цього питання до текстового файлу (без речей "\ tLink"). Дивіться також посилання на реалізацію в Python .

Правила

  • Немає доступу до мережі, крім одного запиту API на api.stackexchange.com
  • Немає використання функцій API або мов, які з'явилися після подання цього питання.
  • Перший рядок відповіді повинен бути сумісним з таблицями. Якщо він порушує сценарій лідерів, приєднаний до питання, відповідь не є конкурентоспроможною.
  • Якщо щойно додана відповідь робить якусь існуючу відповідь зламаною, то автор старої відповіді повинен її виправити (або вона стає неконкурентною).
  • Посилання на мови, закреслені бали тощо повинні оброблятися.
  • Ранги повинні оброблятися, як у фрагменті (наприклад, рівний бал => рівний ранг => розрив у званнях).

Прийнята відповідь - це відповідь з найнижчою оцінкою після достатньої кількості бездіяльності (мінімум 1 місяць).

Гарна ідея

  • Для перевірки з ідентифікаторами питань 47338 (для обробки дублікатів очок + викреслена оцінка балів) та 17005 (для обробки посилань). Це стикається з відповіді "Дійсно" на "Добре" і захищає від перерв від подальших подань.
  • Для включення вихідних прикладів як для цього, так і для перекритих версій ID.

Не Необхідно

  • Обробка понад 100 відповідей (ліміт API для одного запиту)
  • Обробка скасованих коментарів
  • Сортування розділу "Переможці за мовою"
  • Дискримінація конкуруючих та ламаних відповідей

Таблиця лідерів


Дещо пов’язане і це (але вони дуже різні виклики).
Стюі Гріффін

Відповіді:


2

Perl + Mojolicious, 468 456 469 504 байт

Використання бібліотеки Мооліція .

use v5.10;use ojo;while(@i=@{(g("http://api.stackexchange.com/2.2/questions/111735/answers?site=codegolf&filter=withbody&page=".++$p)->json//{})->{items}}){push@r,[$_->{owner}{display_name},(($h=x($_->{body})->at("h1,h2")||next)->at("a")||$h)->text=~/\s*([^,]+)\s*/,$h->text=~/(\d+)[^\d]*$/]for@i}$,="   ";say"Leaderboard
",Author,$l=Language,Size;say+(++$i,$s{@$_[2]}//=$i).".",@$_
for@r=sort{@$a[2]-@$b[2]}@r;%h=map{@$_[1],$_}reverse@r;say"Winners by $l
$l",User,Score;say$_,$h{$_}[0],$h{$_}[2]for keys%h

Безголівки:

use v5.10;
use ojo;

my @r;
while (my @i = @{ (g("http://api.stackexchange.com/2.2/questions/111735/answers?site=codegolf&filter=withbody&page=" . ++$p)->json // {})->{items} }) {
    my $h = x($_->{body})->at("h1,h2") or next;
    push(@r, [$_->{owner}{display_name}, ($h->at("a") || $h)->text =~ /\s*([^,]+)\s*/, $h->text =~ /(\d+)[^\d]*$/]) for @i;
}

$, = "\t";
my %s;
say("Leaderboard\n", "Author", (my $l = "Language"), "Size");
say((++$i, $s{$_->[2]} //= $i) . ".", @$_) for @r = sort { $a->[2] <=> $b->[2] } @r;

my %h = map { $_->[1] => $_ } reverse(@r);
say("Winners by $l\n$l", "User", "Score");
say($_, $h{$_}[0], $h{$_}[2]) for keys(%h);

Can't locate ojo.pm in @INC-> Чи означає це, що це не просто "Perl", а "Perl + Mojolicious"? Чи не використання бібліотек, що не входять до мови, не враховуються як стандартна лазівка?
Ві.

Якщо я виправити ідентифікатор питання на 47338, я не бачу правильного поводження з зв'язками. Замість того, щоб нормальні записи зникли.
Ві.

Виправлення: записи з однаковими балами отримують різний ранг. Хоча це не робить цю відповідь недійсною (поки що), це не добре.
Ві.

@Vi. Фіксовані зв’язки.
Денис Ібаєв

1
Працює. Напевно, найбільш серйозним питанням, що залишився (і, ймовірно, джерелом потоку) є ім'я подання. Може змінити його на "Perl + Mojolicious", щоб уникнути конкуренції з чистими рішеннями Perl? Моолічіс, здається, орієнтований на однолінійну бібліотеку (тому корисна для гольфу) бібліотеку, не встановлену в більшості розгорнень Perl, тому не може вважатися частиною мови.
Ві.

6

Пітон 3, 860 856 байт

Трохи гольфуйте, просто завантажте таблицю лідерів та надайте шаблон для інших гольфістів:

import json,re,html as h,requests as r
p=print
u=h.unescape;a=[];n={}
for i in json.loads(r.get("https://api.stackexchange.com/2.2/questions/111735/answers?page=1&pagesize=100&site=codegolf&filter=!t)IWYnsLAZle2tQ3KqrVveCRJfxcRLe").text)["items"]:
    m=re.match(r'<h\d>\s*([^\n,]*[^\s,]),.*?(\d+)(?=[^\n\d<>]*(?:<(?:s>[^\n<>]*<\/s>|[^\n<>]+>)[^\n\d<>]*)*<\/h\d>)',i["body"].splitlines()[0]);l=u(m.group(1));t=u(i["owner"]["display_name"]);s=m.group(2);a.append((t,l,s))
    if l not in n: n[l]=[]
    n[l].append((t,s))
p("Leaderboard\n\tAuthor\tLanguage\tSize")
z=0;y=None
for i in enumerate(sorted(a,key=lambda x:x[2])):
    if y==i[1][2]:z+=1
    else:z=0;y=i[1][2]
    p("%d.\t%s\t%s\t%s"%(i[0]+1-z,i[1][0],i[1][1],i[1][2]))
p("Winners by Language\nLanguage\tUser\tScore")
for i in n.keys():
    n[i].sort(key=lambda x:x[1])
    print("%s\t%s\t%s"%(i,n[i][0][0],n[i][0][1]))

Відступ з вкладками. Останнє printнавмисно не замінюється, pщоб створити крапку балів з відповіддю Mathematica .

Безголівки:

import json
import re
import html
import requests
url="https://api.stackexchange.com/2.2/questions/111735/answers?page=1&pagesize=100&site=codegolf&filter=!t)IWYnsLAZle2tQ3KqrVveCRJfxcRLe"
data=json.loads(requests.get(url).text)
answers=[]
languages={}
for i in data["items"]:
    header=i["body"].splitlines()[0]
    m=re.match(r'<h\d>\s*([^\n,]*[^\s,]),.*?(\d+)(?=[^\n\d<>]*(?:<(?:s>[^\n<>]*<\/s>|[^\n<>]+>)[^\n\d<>]*)*<\/h\d>)', header)
    lang=html.unescape(m.group(1))
    author=html.unescape(i["owner"]["display_name"])
    score=m.group(2)
    answers.append((author, lang, score))
    if lang not in languages: languages[lang]=[]
    languages[lang].append((author, score))
answers.sort(key=lambda x:x[2])
print("Leaderboard")
print("\tAuthor\tLanguage\tSize")
rankadj=0
prevscore=None
for i in enumerate(answers):
    if prevscore == i[1][2]:
        rankadj+=1
    else:
        rankadj=0
        prevscore=i[1][2]
    print("%d.\t%s\t%s\t%s" % (i[0]+1-rankadj, i[1][0], i[1][1], i[1][2]))
print("Winners by Language")
print("Language\tUser\tScore")
for i in languages.keys():
    w=languages[i]
    w.sort(key=lambda x:x[1])
    print("%s\t%s\t%s" % (i, w[0][0], w[0][1]))

Примітка: він ще не обробляє посилання правильно, тому не вдається, наприклад, питання 17005 .


1
Навіть якщо ви самі на це відповісте, вам доведеться відповісти на питання про гольф з кодом .
NoOneIsHere

@NoOneIsHere, я не впевнений у " доведеться ". Тільки в тому випадку, "якщо ви відповідаєте на питання з кодом про гольф і хочете виграти ".
Ві.

3
@Vi. Потім включіть його у запитання як опорне рішення. Це відповідь, яка не відповідає на (власний) запитання, яке запитує код для гольфу, за кожним кодом - гольф .
NoOneIsHere

1
@Vi. ніпе. Або це повністю гольф, або це заборонено. Якщо це можна легко пограти в гольф, то це не відповідь. Я дійсно сказав би просто поставити це питання як довідкове рішення.
Rɪᴋᴇʀ

4
@Vi. Ні, гольфу це найкраще .
NoOneIsHere

1

Bash + JQ, 399 байт

Зауважте, що це майже напевно можна додатково гольфувати, оптимізуючи jqлогіку вираження.

Гольф

curl api.stackexchange.com/2.2/questions/111735/answers?site=codegolf\&filter=withbody|zcat|jq -r '[.items[]|{o:.owner.display_name}+(.body|capture("^<h1>(?<l>.*?),.*?(?<b>\\d*)\\D*</h"))]|sort_by(.b|tonumber)|("Leaderboard\n\tAuthor\tLanguage\tSize",(keys[] as $i|.[$i]|"\($i+1).\t"+.o+"\t"+.l+"\t"+.b),"Winners by Language\nLanguage\tUser\tScore",(group_by(.l)|.[]|min_by(.b)|.l+"\t"+.o+"\t"+.b))'

Вибірка зразка

Leaderboard
    Author  Language    Size
1.  zeppelin    Bash + JQ   399
2.  Tom JavaScript ES6  454
3.  Denis Ibaev Perl    456
4.  Vi. Python 3    860
Winners by Language
Language    User    Score
Bash + JQ   zeppelin    399
JavaScript ES6  Tom 454
Perl    Denis Ibaev 456
Python 3    Vi. 860

Яка версія jq потрібна? Я отримуюerror: capture is not defined
Ві.

Я не бачу повторних рангів, коли я прив’язую ідентифікатор питання до 47338. Подання може ставати недійсним по мірі їх надходження (і пізніше я можу додати спеціальний неконкуруючий псевдовідповідь лише для того, щоб примусити правильну обробку рангів).
Ві.

Відповідь CJAM Мартіна Ендера має "CJam, <s> 28 </s> 27 байт". Це слід тлумачити як 27, а не 28, як в офіційному фрагменті лідерів. Або 19-го та 22-го байт Піта.
Ві.

Зверніть увагу на пункт "* Посилання на мови, закреслені бали тощо повинні оброблятися". у правилах виклику.
Ві.

@Vi - має бути все встановлено зараз (див. Jqplay.org/s/LuZfAn2Pxr ). Відповідь Pyth - це все-таки 22 байти, оскільки він просто не відформатований правильно (перше число байтів прибуває першим).
цепелін

1

Математика, 852 856 байт

Використовується вбудований JSONToolsпакет. Це не та річ, яку Математика мала використовувати, тому я використала її!

p=Print;S=StringRiffle;L=Length;r=Range;out=Association@JSONTools`FromJSON[Import["http://api.stackexchange.com/2.2/questions/111735/answers?site=codegolf&filter=withbody"]];l={};i=Association/@(out["items"]);
(f=("body"/.i)[[#]];h=StringPosition[f,{"<h1>","</h1>"}];a="display_name"/.("owner"/.i)[[#]];s=StringSplit[StringTake[f,{h[[1]][[2]]+1,h[[2]][[1]]-1}],{",","<a>","</a>",">","<s>","</s>"," bytes","<strike>","</strike>"}];AppendTo[l,{a,s[[1]],ToExpression@s[[-1]]}])&/@r@L["body"/.i];l=SortBy[l,Last];o=r@L@l;If[l[[#]][[3]]==l[[#-1]][[3]],o[[#]]=o[[#-1]]]&/@r[2,L@l];
p@"Leaderboard"
p@"\tAuthor\tLanguage\tSize"
For[i=1,i<=L@l,i++,p[ToString@o[[i]]<>"."<>S[l[[i]][[#]]&/@r@3,"\t"]]]
l=SortBy[l,{#[[2]],#[[3]]}&];l=DeleteDuplicatesBy[l,#[[2]]&];
p@"Winners by Language"
p@"Language\tUser\tScore"
For[i=1,i<=L@l,i++,p[S[l[[i]][[#]]&/@{2,1,3},"\t"]]]

Як це протестувати? Я отримую ReplaceAll::reps: <content of the downloaded Mathematica answer> is neither a list of replacement rules nor a valid dispatch table, and so cannot be used for replacing., після чого відбитки {$Failed, $Failed, $Failed, $Failed, $Failed, bytes, $Failed, $Failed}}.
Ві.

Можливо, рішення порушено власним вихідним кодом (який, очевидно, містить <h1>)?
Ві.

@Vi. Це дивно, це працює для мене. На якому питанні ви його тестували? Я не думаю, що це вихідний код порушує його, оскільки API StackExchange використовує HTML-версії для <і> автоматично.
numbermaniac

Цей, 111735. Але я замінив URL-адресою, /tmp/q.jsonде є попередньо завантажена відповідь JSON.
Ві.

Вихід для 47338: paste.debian.net/918716
Vi.
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.