Назовите покерную комбинацию

22

Назовите покерную комбинацию

Учитывая пять карт, выведите название покерной руки, которая будет одной из:

High card
One pair
Two pair
Three of a kind
Straight
Flush
Full house
Four of a kind
Straight flush
Royal Flush

Если вы сомневаетесь, обратитесь к правилам на http://en.wikipedia.org/wiki/List_of_poker_hands .

вход

5 карт из аргументов стандартного ввода или командной строки. Карта представляет собой двухбуквенную строку в форме RS, где R - ранг, а S - масть. В рядах являются 2- 9(номер карты), T(десять), J(Jack), Q(Queen), K(король), A(Ace). В костюмы являются S, D, H, Cдля лопаты, бубны, червы и клубов соответственно.

Пример карт

5H - five of hearts
TS - ten of spades
AD - ace of diamonds

Пример ввода => желаемый вывод

3H 5D JS 3C 7C => One pair
JH 4C 2C JD 2H => Two pair
7H 3S 7S 7D 7C => Four of a kind
8C 3H 8S 8H 3S => Full house

правила

Самый короткий код выигрывает

редактировать

Выглядит великолепно до сих пор! Я не могу на самом деле проверить все ответы, так как я не очень хорошо знаю эти языки и у меня нет компиляторов / интерпретаторов для всех из них, но я подозреваю, что не все думали о том, что тузы могут быть как самыми высокими, так и самые низкие карты стрита (флеш) .

daniero
источник
2
На переполнении стека есть смутно родственные вещицы .
dmckee
Разрешено ли использовать заглавные буквы (или нет), как нам угодно?
Mr.Wizard
Мистер Волшебник, конечно.
Даниеро

Ответы:

3

GolfScript ( 209 208 207 206 200 199 197 196 символов)

3/zip:^0={10,''*"TJQKA"+?}/]:?15,{?\{=}+,,}%2,-$6,14.),++@$/):|;[!!2*^1=.&,(!+5+]or{/}*'Full house
Two pair
One pair
ThreeKFourKHigh card
Flush
Straight''K'/' of a kind
'*n/~|1$"Royal"if" "+2$+](=

Я использую предлагаемую свободу настройки капитализации: мой Стрит-Флеш и Роял-Флэш оба используют Флэш, чтобы повторно использовать слово из простого флеша.

Примечание: некоторые более ранние версии были глючными: они поддерживали аншлаг только тогда, когда пара имела меньшее значение, чем королевская пара. Их можно исправить, заменив пробел, разделяющий - 0на $.

демонстрация

Питер Тейлор
источник
Теперь это программа для гольфа! Я искал способы сократить его, но ничего не могу придумать. Использование .&для поиска различных символов в строке - очень полезный трюк.
Кристиан Лупаску
@ w0lf, это довольно стандартный трюк. Говард тоже использует это в своем решении.
Питер Тейлор
8

Придумал свой ответ :)

Питон - 312 301 298

R,K,F,S,g=' 23456789TJQKA2345A',' of a Kind','Flush','Straight ',sorted
s,r=''.join(g(raw_input(),key=R.find)).split()
n,m=g(map(r.count,set(r)))[-2:]
print[[F,[0,'High Card','TOwnoe'[n&1::2]+' Pair',['Full House','Three'+K][n&1],'Four'+K][m]],[[S,'Royal '][r[0]=='T']+F,S]][r in R][len(set(s))>1]

Создает список 2x2, где индексы двух измерений являются логическими проверками для флеша и стрита. В случае и того и другого мы проверяем, является ли это флеш-роялем или просто стрит-флешем. Если нет флеша и не стрита, мы проверяем другие руки: mи nдержим наибольшее и второе наибольшее количество карт одного ранга; имена рук хранятся в списке с индексами в соответствии с m. Субтитры в значениях этого списка выполняются для того, nчтобы отделить одну пару от двух пар и три вида от дома.

Редактировать: Спасибо Нолен Роялти за спасение в общей сложности 20 символов!

daniero
источник
1
... и побил мой.
Mr.Wizard
Люблю новое решение, 312 персонажей довольно крошечные. Очень умный метод борьбы с одной парой против двух: D
Nolen Royalty
Спасибо :) Вы можете попробовать, если хотите. Но вы, возможно, не используете аналогичные переменные, как мои m и n. Проверяя это и снова просматривая ваш код, я только что понял, что могу побриться еще немного на оригинальном ^^
daniero
1
Не могли бы вы потерять еще 8 символов, переключившись m,n=g([c.count(x)for x in set(r)])на m,n=g(map(c.count,set(r)))?
Nolen Royalty
Вау, ты чертовски прав, я мог бы: D Не знаю, почему это ускользнуло от меня. Хороший улов, спасибо!
Даниеро
5

Рубин 1.9 (427 359 348 338 296 292)

РЕДАКТИРОВАТЬ : Исправлено для работы с низкими тузами.

o,p=%w(flush straight)
f=/1{5}|1{4}0+1$/
s=[0]*13
puts Hash[*$*.map{|c|s['23456789TJQKA'.index c[0]]+=1;c[1]}.uniq[1]?[f,p,?4,'four'+a=' of a kind',/3.*2|2.*3/,'full house',?3,'three'+a,/2.*2/,'two pair',?2,'one pair',0,'high card']:[/1{5}$/,'royal '+o,f,p+' '+o,0,o]].find{|r,y|s.join[r]}[1]

Основная идея состоит в том, чтобы создать массив количества карт в каждом ранге, объединить цифры в строку, а затем запустить регулярные выражения, чтобы увидеть, какая форма руки подходит. Мы подсчитываем количество разных мастей, чтобы определить, проверять ли их по разным флешам (флеш, стрит-флеш, рояль-флеш) или другим формам (все остальное).

Принимает карты как отдельные аргументы командной строки, например так:

>ruby poker-hand-golf.rb 3H 5D JS 3C 7C
one pair
Пол Престиж
источник
4

C, 454 символа

#define L for(a=1;a<6;a++)for(b=0;b<13;b++)
#define U u[b+6]
#define R(x,y) if(x)puts(#y);else
b,f,r,h=0,s=0,u[20]={0};main(int a,char**v){L U+=v[a][0]=="23456789TJQKA"[b];f=v[1][1];L{if(v[a][1]!=f)f=0;u[a]+=a==U;if(b>7)h+=U;if(a*13+b<64||!U)r=0;else if(++r==5)s=1;}R(f&&h==25,Royal flush)R(f&&s,Straight flush)R(u[4],Four of a kind)R(u[3]&&u[2],Full house)R(f,Flush)R(s,Straight)R(u[3],Three of a kind)R(u[2]==2,Two pair)R(u[2],One pair)R(h,High card);}

Запуск из командной строки с карточками в качестве аргументов, например ./a.out 8C 3H 8S 8H 3S

Расширенная версия с комментариями:

#define L for(a=1;a<6;a++)for(b=0;b<13;b++)
#define R(x,y) if(x)puts(#y);else
#define U u[b+6]
b,f,r,h=0,s=0,u[20]={0};
main(int a,char**v){
    // card usage - u[6..]
    L U+=v[a][0]=="23456789TJQKA"[b];
    // NOTE: lets expand the inner body of the loop in the answer so this looks more sane:
    // flush
    f=v[1][1];L if(v[a][1]!=f)f=0;
    // count of usages - u[0..5] 
    L u[a]+=a==U;
    // high cards x5
    L if(b>7)h+=U;
    // straights
    L if(a*13+b<64||!U)r=0;else if(++r==5)s=1;        
    // display
    R(f&&h==25,Royal flush)
    R(f&&s,Straight flush)
    R(u[4],Four of a kind)
    R(u[3]&&u[2],Full house)
    R(f,Flush)
    R(s,Straight)
    R(u[3],Three of a kind)
    R(u[2]==2,Two pair)
    R(u[2],One pair)
    R(h,High card);    
}

Редактирование:

  1. Сохранено 12 символов путем объединения и повторного использования циклов.
  2. Сохранено 9 символов, введя строковую константу.
  3. Сохранено 19 символов с помощью строкового преобразования в макросе
ребенок кролика
источник
3

Математика , 365

Вот мой взгляд на ответ Дэвида Каррахера.

Показано с пробелами для удобства чтения.

If[
  a = Characters;
  x = Thread;
  r = Range;
  d = Sort[a@StringSplit@# /. x[a@"23456789TJQKA" -> 2~r~14]];
  {t, u} = Sort[Last /@ Tally@#] & /@ x@d;
  c = First /@ d;
  f = u == {5};
  S = "Straight";
  c == r[b = d[[1, 1]], b + 4],
  If[f,
   If[c == 10~r~14, "Royal Flush", S <> " flush"], S],
  If[f, "Flush",
   Switch[t,
    {_, 4},    "Four of a kind",
    {2, 3},    "Full house",
    {__, 3},   "Three of a kind",
    {_, 2, 2}, "Two pair",
    {__, 2},   "One pair",
    _,         "High card"]
  ]
] &

Однострочная версия:

If[a=Characters;x=Thread;r=Range;d=Sort[a@StringSplit@#/.x[a@"23456789TJQKA"->2~r~14]];{t,u}=Sort[Last/@Tally@#]&/@x@d;c=First/@d;f=u=={5};S="Straight";c==r[b=d[[1,1]],b+4],If[f,If[c==10~r~14,"Royal Flush",S<>" flush"],S],If[f,"Flush",Switch[t,{_,4},"Four of a kind",{2,3},"Full house",{__,3},"Three of a kind",{_,2,2},"Two pair",{__,2},"One pair",_,"High card"]]]&
Mr.Wizard
источник
Ницца. Вы даже нашли место для сохранения в сопоставлении с образцом. Например, _вместо{_,_,_,_}
DavidC
Хорошие решения, вы оба. Ради подсчета символов, я думаю, что «Пара» должна называться «Одна пара», хотя это звучит немного плохо, поскольку это то, что я опубликовал, и другие реализовали.
Даниеро
@Daniero Спасибо. Я исправлю имя.
Мистер Волшебник
3

К, 294 295

d:{F:"Flush";S:"Straight ";P:" Pair";K:" of a kind";$[(f:1=#?,/-1#'c)&("AJKQT")~a@<a:,/j:1#'c:" "\:x;"Royal ",F;f&s:(4#1)~1_-':a@<a:,/(("A23456789TJQKA")!1+!14)@j;S,F;4=p:|/#:'=j;"Four",K;(2;3)~u:a@<a:,/#:'=j;"Full House";f;F;s;S;3=p;"Three",K;(1;2;2)~u;"Two",P;(1;1;1;2)~u;"One",P;"High Card"]}

,

k)d'("TS JS QS KS AS";"3S 4S 5S 7S 6S";"JC JH KS JD JS";"JC JH 2S JD 2C";"2C 9C TC QC 6C";"8C 5D 9H 6C 7D";"8C 8D 9H 8S 7D";"8C 8D 9H 2S 9D";"8C 8D 4H 2S 9D";"3C 8D 4H 2S 9D")
"Royal Flush"
"Straight Flush"
"Four of a kind"
"Full House"
"Flush"
"Straight "
"Three of a kind"
"Two Pair"
"One Pair"
"High Card"

редактировать: добавлен 1 символ для Ace-low straights

tmartin
источник
3

Python 334 , 326 322 персонажа

p,f,l,t,o=" pair"," of a kind"," Flush","Straight","A23456789TJQK"
v,u=zip(*raw_input().split())
s=''.join(sorted(v,key=o.find))
print{5:"High card",7:"One"+p,9:"Two"+p,11:"Three"+f,13:"Full house",17:"Four"+f,23:t,24:l[1:],25:t,42:t+l,44:"Royal"+l}[(sum(map(v.count,v)),24)[len(set(u))<2]+((0,20)[s=="ATJQK"],18)[s in o]]

Я знаю, что последний лайнер становится довольно нечитабельным, я выложу версию без игры в гольф, когда буду доволен своим решением.

Нолен Роял
источник
2

GolfScript, 258 250 символов

3/zip~;.&,(!\{"23456789TJQKA"?}%$.(\{.@- 8%}%\;"\1"-!\.1/.&{1$\-,}%1.$?)"Four"" of a kind":k+{.,2="Full house"{.2\?)"Three"k+{.3-,({.3\?)"One pair"{;"Straight":?;2$2$&{(8="Royal"?if" flush"+}{;?{"Flush""High card"if}if}if}if}"Two pair"if}if}if}if])\;

Программа ожидает ввода на STDIN, как указано выше, и выводит на STDOUT. Вы можете проверить код самостоятельно .

> 8C 3H 8S 8H 3S
Full house

> 8C 7H 6S TH 9S
Straight

> AH 3H 4S 2H 6S
High card

Изменить: Включенные предложения w0lf.

Говард
источник
Отличное решение! Вы можете сохранить 3 символа, введя " of a kind"переменную, потому что она используется дважды.
Кристиан Лупаску
это также работает с"Straight"
Cristian Lupascu
@ w0lf Спасибо. Я добавил ваши предложения в код.
Говард
Я думаю, что есть небольшая ошибка с обнаружением стритов: AH KH 2C 3H 4Hсчитается стритом, но должна быть High картой.
Кристиан Лупаску
@ w0lf Хммм, я должен подумать об этом ...
Говард
2

Mathematica - 500 494 465 символов

Это решение основано на демонстрации покера Эдом Пеггом-младшим. В этой версии карты обрабатываются внутренне как числа вRange[2,14]

v[x_] := Block[{d, t, c, f, s},
 d = Sort@ToExpression[Characters[ImportString[x, "Table"][[1]]] /. {"T" -> 10, "J" -> 11, "Q" -> 12, "K" -> 13, "A" -> 14}];t = Sort /@ Map[Length, Split /@ Sort /@ Transpose@d, {2}];c = d[[All, 1]];f = (t[[2]] == {5});s = c == Range[b = d[[1, 1]], b + 4];
If[s,
 If[f, If[c == 10~Range~14, "royal flush", "straight flush"],"straight"],
 If[ f, "flush",
Switch[t[[1]],
 {1, 4}, "four of a kind",
 {2, 3}, "full house",
 {1, 1, 3}, "three of a kind",
 {1, 2, 2}, "two pair",
 {1, 1, 1, 2}, "one pair",
 {1, 1, 1, 1, 1}, "high card"]]]]

Образцы входов, выходов:

данные

Заметки:

ф: флеш

с: карты (без масти)

s: прямой

t: {карты, люксы}

д:

DavidC
источник
Хорошо, но как вы получаете две пары из JH 4C 2C JD TH?
Даниеро
Ты прав. Возникла ошибка, когда я соединил некоторые компоненты в чистую функцию. Я разыщу это.
DavidC
@Daniero Проблема, которую вы подняли, была решена.
DavidC
Дэвид, есть много места, чтобы сжать это. Могу ли я?
Mr.Wizard
@ Mr.Wizard Будь моим гостем. Я буду смотреть и учиться.
DavidC