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

11

Соревнование:

В этом вопросе: Назовите покерную комбинацию, в которой вы брали пятикарточную покерную комбинацию, и определите ее. Этот вопрос похож, с двумя поворотами:

Во-первых, выходные данные будут в нижнем регистре. Это позволяет больше играть в гольф, так как вам не нужно беспокоиться о капитализации flushиstraight

high card
one pair
two pair
three of a kind
straight
flush
full house
four of a kind
straight flush
royal flush

Во-вторых, благодаря популярности техасского холдема и 7-карточного стад- покера , мы здесь, в кодовом гольфе, должны иметь возможность выиграть покерную комбинацию из семи карт, я прав? Когда вы набираете комбинацию из семи карт, используйте пять лучших карт для своей руки и игнорируйте две, которые вам не нужны.

Ссылка:

Список покерных рук: http://en.wikipedia.org/wiki/List_of_poker_hands

Ввод (снято непосредственно с предыдущего потока)

7 карт из аргументов стандартного ввода или командной строки. Карта - это двухбуквенная строка в форме 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 AH QS => one pair
JH 4C 2C 9S 4H JD 2H => two pair
7H 3S 7S 7D AC QH 7C => four of a kind
8C 3H 8S 8H 3S 2C 5D => full house
AS KC KD KH QH TS JC => straight

Обратите внимание, что во втором примере фактически есть три пары, но вы можете использовать только пять карт two pair. В пятом примере есть и a, three of a kindи straightвозможное, но a straightлучше, поэтому вывод straight.

счет

Это , поэтому выигрывает самый короткий код!

описки

  1. Вы не можете использовать внешние ресурсы.
  2. Туз и высокий и низкий для прямых.
durron597
источник
Ницца; Я втайне надеялся, что кто-нибудь заберет мяч. Просто хотел отметить, что у меня не было никаких ограничений на использование заглавных букв в исходном вопросе (уточнено в комментарии), поэтому вы могли (и большинство / все сделали) вывести «Стрит-флеш». ИМХО заглавными выглядит лучше.
Даньеро
Вы говорите, ввод (поднял прямо из предыдущего потока) 5 карт. Я думаю, что вы хотели изменить это на 7.
Level River St
@steveverrill Вы можете редактировать сообщения самостоятельно на стеке обмена. Хотя я сделал это для вас здесь
durron597
Разрешены ли внешние ресурсы? Существуют таблицы поиска, которые позволят вам просто посмотреть каждую карту в руке и получить силу руки.
Кендалл Фрей,
Может ли туз быть как низким, так и высоким для стритов?
Ник Т

Ответы:

4

Рубин 353

Это было основано на ответе Крона от исходного вопроса.

Это принимает входные данные в качестве аргументов командной строки. По сути, мы просто перебираем все комбинации размера 5, чтобы определить тип раздачи. Каждый тип руки был изменен, так что он начинается с цифры. («флеш-рояль» -> «0-рояль-флеш», «старшая карта» -> «9-я карта»). Это позволяет нам сортировать строки, которые были возвращены. Первая строка после сортировки - лучшая возможная рука. Таким образом, мы печатаем это после удаления всех чисел из строки.

o,p=%w(4flush 1straight)
f=/1{5}|1{4}0+1$/
puts $*.combination(5).map{|z|s=[0]*13;Hash[*z.map{|c|s['23456789TJQKA'.index c[0]]+=1;c[1]}.uniq[1]?[f,'5'+p,?4,'2four'+a=' of a kind',/3.*2|2.*3/,'3full house',?3,'6three'+a,/2.*2/,'7two pair',?2,'8one pair',0,'9high card']:[/1{5}$/,'0royal '+o,f,p+' '+o,0,o]].find{|r,y|s.join[r]}[1]}.sort[0].gsub(/\d/,'')
FDinoff
источник
Ницца. Gsub в конце может быть просто не так ли?
Bazzargh
@bazzargh нет, нужно удалить все цифры. Код объединяет 4flush с 1straight или 0royal, чтобы получить «0royal 4 flush» или «1straight 4flush». Если мы используем только sub, 4 не будет удален.
FDinoff
Дает неправильный результат для AS QS JS TS 9S 5H 5D. Это будет стоить вам персонажа!
@ WumpusQ.Wumbley Хм, похоже, это ошибка в оригинальном коде. Я постараюсь выяснить, в чем проблема позже.
FDinoff
5

Haskell 618 603 598 525 512 504 480 464

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

import Data.List
m=map
z=take 5
q=m(\x->head[n|(f,n)<-zip"A23456789TJQK"[1..],f==x!!0])
l=m length
v=" of a kind"
w="flush"
y="straight"
c f s p r|f&&r="9royal "++w|f&&s='8':y++" "++w|f='5':w|s||r='4':y|True=case p of 4:_->"7four"++v;3:2:_->"6full house";3:_->"3three"++v;2:2:_->"2two pair";2:_->"1one pair";_->"0high card"
d x=c([5]==l(group$m(!!1)x))(q x==z[head(q x)..])(l$group$q x)$q x==1:[10..13]
k h=tail$maximum$m(d.z)$permutations$words h
main=interact k

Отредактированный для вставки «пары» и использования префиксов чисел после просмотра записи @ FDinoff, он также составлял функции карты, чтобы сбрить еще один символ.

bazzargh
источник
Вы можете спасти себя пару символов (я думаю, около 5), если вы избавитесь от вас. "one pair","two pair"корочеu=" pair" ... "one"++u,"two++u
FDinoff
да, я только что сделал это изменение после прочтения вашего кода. Также техника числового префикса спасает меня еще 5
bazzargh
2

C ++, 622 553 символа

четыре ненужных перевода строки добавлены ниже для ясности.

#include"stdafx.h"
#include"string"
std::string c=" flush",d=" of a kind",e="straight",z[10]={"high card","one pair","two pair","three"+d,e,c,"full house","four"+d,e+c,"royal"+c},
x="CDHSA23456789TJQK";char h[99];int main(){__int64 f,p,t,g,u,v,w,l=1,a=78517370881,b=a+19173960,i,j,q=0;gets_s(h,99);for(i=28;i-->7;){f=p=0;
for(j=7;j--;)if(j!=i%7&j!=(i+i/7)%7){f+=l<<x.find(h[j*3+1])*6;p+=l<<x.find(h[j*3])*3-12;}
v=p&b*2;u=v&v-1;w=p&p/2;g=p*64&p*8&p&p/8&p/64;f&=f*4;t=f&&p==a?9:f&&g?8:p&b*4?7:u&&w?6:f?5:g||p==a?4:w?3:u?2:v?1:0;
q=t>q?t:q;}puts(z[q].c_str());}

Ситуация изменилась в версии для гольфа:

Версия 1: Изменены все числовые переменные __int64для одного объявления.

Rev 1: гольф приращение и состояние forпетель

Rev 0: изменены восьмеричные константы на десятичные.

Rev 0: Изменены ifоператоры для присвоений с условным оператором. Rev 1: переставлен далее в одно выражение для t. Это требует новой переменной vдля одного из промежуточных значений

Rev 0: удален подробный вывод. Только выводит лучшую общую руку.

Rev 0: Сжатие при выводе текста (сложно в C, потому что вы не можете объединять строки с помощью оператора +). Запись "flush" только один раз сэкономила мне 12 символов, но стоила мне 15, что в целом ухудшило меня на 3 символа. Поэтому я просто написал это 3 раза вместо этого. Rev 1: используется std::stringвместо того, char[]что предложено FDinoff, что делает возможным объединение с +.

Ungolfed версия, 714 некомментированных непробельных символов.

Обходит все 21 возможных рук, которые можно составить из 7 карт, и каждый раз отбрасывает 2 карты. Масти и ранг пяти выбранных карт суммируются в переменных f и p с различной восьмеричной цифрой для каждой масти / ранга. Различные битовые операции выполняются для определения типа руки, которая затем сохраняется в t (все 21 возможности выводятся в версии без заглавных букв). Наконец, выводится лучшая возможная рука.

#include "stdafx.h"
#include "string.h"

char x[] = "CDHSA23456789TJQK", h[99], z[10][99] = 
{ "high card", "one pair", "two pair","three of a kind", "straight","flush","full house","four of a kind","straight","royal" };

int main(void)
{
        int i,j,q=0;                  //i,j:loop counters. q:best possible hand of 7 card   
        scanf_s("%s/n", &h, 99); getchar();
        for (i = 7; i < 28; i++){

          //f,p: count number of cards of each suit (2 octal digits) and rank (1 octal digit.)
          //t: best hand for current 5 cards. g:straight flag. u,w: flags for pairs and 3's.   
          //l: constant 1 (64bit leftshift doesn't work on a literal.) 
          //octal bitmasks: a=ace high straight, b=general purpose

            __int64 f=0,p=0,t=0,g,u,w,l=1,a=01111000000001,b=a+0111111110;

           for (j = 0; j < 7; j++){
               if (j != i %7 & j != (i+i/7) %7){

                   f += l << (strchr(x,h[j*3+1])-x)*6;
                   p += l << (strchr(x,h[j*3])-x-4)*3;

                   printf_s("%c%c ",h[j*3], h[j*3+1]);
               }
           }

           w=p&b*2;                          //if 2nd bit set we have a pair
           if (w) t=1;
           u= w & w-1;                       //if there is only one pair w&w-1 evaluates to 0; +ve for 2 pair.
           if (u) t=2;
           w = p & p/2;                      // if 2nd and 1st bit set we have 3 of kind. 
           if (w) t=3;
           g = p*64 & p*8 & p & p/8 & p/64;  // detects all straights except ace high. pattern for ace high in a.
           if (g||p==a) t=4;
           f&=f*4;                           //for a flush we want 5 cards of the same suit, binary 101
           if (f) t=5;
           if (u&&w) t=6;                    //full house meets conditions of 2 pair and 3 of kind
           if (p & b*4) t=7;                 //four of a kind
           if (f && g) t=8;                  //straight flush
           if (f && p==a) t=9;               //royal flush
           printf_s("%s %s \n",z[t],t>7?z[5]:"");
           q=t>q?t:q;
        }   
        printf_s("%s %s",z[q],q>7?z[5]:"");
        getchar();
}

Неуправляемый выход

введите описание изображения здесь

Уровень реки St
источник
Поскольку вы говорите, что используете c ++, вы можете использовать, <string>который поддерживает + для конкатенации строк. Что означает, что вы, вероятно, могли бы использовать <iostream>и использовать. coutОднако на самом деле я не знаю, приведет ли это к меньшему количеству символов.
FDinoff
@FDinoff, что я мог бы сохранить: " pair flush flush straight of a kind"= 35 символов. Как только вы добавляете в #includeсбережения минимальные, то вам нужно учитывать дополнительные ",=+и объявления констант. Кроме того, я новичок в C ++ и борюсь с IDE и настройками компилятора (это вынуждает меня использовать scanf_sи printf_sвместо старых «небезопасных» версий, а помощь по его исправлению идет по кругу.) coutМожет немного помочь, это на мое сделать список, но, вероятно, для другой программы. Что coutменя убивает , так это то, что using namespace stdя не знаю, есть ли способ избежать написания всего этого.
Уровень Река St
Вам почти никогда не понадобятся printf и scanf, так как вы используете c ++. Есть и другие (безопаснее) сделать то же самое. Вы можете использовать, std::coutчтобы обойтиusing namespace std
FDinoff
@FDinoff спасибо за совет. В моем последнем редактировании я сохранил 18 байтов с различной обработкой строк: gets_s& puts, плюс std::stringдля конкатенации, что означает, что я должен преобразовать char*в вывод. Гольф, с которым я писал, работает просто stringили просто iostream.причудливо, я должен включить обоих, чтобы использовать <<>>операторов с cin/cout& std::strings. В целом, использование обоих #includes работает на 5 байт хуже, хотя я могу объявить hкак std::stringи избежать отдельного charобъявления. Как и ожидалось, я не могу найти список того, что namespace stdв помощь (или объяснение об операторе.)
Level River St
@FDinoff Я согласен, я бы обычно не использовал scanfи gets, за исключением игры в гольф, где программы все равно довольно небезопасны. Я мог бы сократить на 5 байт, -s,99если бы мог использовать getsвместо gets_s, но я не могу заставить компилятор позволить мне. Что меня удивляет, так это то, насколько небезопасен C / C ++ в целом! Несколько недель назад я был бы потрясен, обнаружив, что _int64 x=1<<yдает неправильный ответ для y больше 31. Но сейчас я просто слегка раздражен. Увидев, что подписчики массива выходят за границы без сообщений об ошибках, я привык к этому. Есть ли способ включить проверку лучше?
Уровень Река St
2

Perl (> = 5,14), 411 403 400 397 400

Изменить : встроенный саб, который был вызван только один раз, сохраняя 8 символов.
Редактировать 2 : удалить a, .""который был оставлен после ранней попытки.
Редактировать 3 : вместо временной переменной, которая сохраняет оригинал $_, используйте одну, чтобы сделать ее ненужной. Чистая прибыль 3 символа.
Правка 4 : исправлена ​​ошибка с обнаружением переполненного фулл-хауса (2x 3 в своем роде). стоит 3 чарса.

Не совсем победитель, но прямой детектор - интересная концепция, я думаю.

sub
j{join"",sort@_}sub
o{j(map{{A=>10}->{$_},11+index(j(2..9).TJQKA,$_)}$h=~/(.(?=@_))/g)=~/.*(..)(??{j
map$1+$_.'.*',1..4})/?$1:()}$h=$_=<>;if(j(/(\S)\b/g)=~/(.)\1{4}/){$w=$_==19?royal:straight
for
o$f=$1}$_=j(/\b(\S)/g)=~s/(.)\1*/length$&/rge;$k=" of a kind";print$w?"$w flush":/4/?four.$k:/3.*2|[23].*3/?"full house":$f?flush:(o".")?straight:/3/?three.$k:/2.*2/?"two pair":/2/?"one pair":"high card"

Расширенная версия:

# We'll be doing a lot of sorting and joining
sub j {
  return join "", sort @_;
}

# r() expects $_ to contain a rank, and converts it to a numeric code. The
# code starts at 10 so the numbers will sort correctly as strings, and a list
# of 2 values is returned because A is both 10 and 23. All other ranks have
# undef as the first value and their proper 11..22 value as the second value.
sub r {
  return ({A=>10}->{$_}, 11+index(j(2..9).TJQKA,$_));
}

# Sequence-detector. Factored into a sub because it's run twice; once over
# the ranks in the flush suit to find a straight flush and once over all the
# ranks to find a straight. On successful match, returns the lowest rank of
# the straight (in the 10..23 representation).
# Required parameter: the suit to search, or "." for all suits.
sub o {
  j(map r,$h=~/(.(?=@_))/g)          # The list of ranks, in increasing order,
                                     # with ace included at both ends...
    =~                               # ...is matched against...
  /.*(..)(??{j map$1+$_.'.*',1..4})/ # ...a pattern requiring 5 consecutive
                                     # numbers.
  ?$1:()
  # A note about this regexp. The string we're matching is a bunch of numbers
  # in the range 10..23 crammed together like "121314151619" so you might
  # worry about a misaligned match starting on the second digit of one of the
  # original numbers. But since that would make every pair of digits in the
  # match end with a 1 or a 2, there's no way 5 of them will be consecutive.
  # There are no false matches.
  # Another note: if we have a royal flush and also have a 9 in the same
  # suit, we need to return the T rank, not the 9, which is why the regexp
  # starts with a .*
}

# Read a line into $_ for immediate matching with /.../ and also save it into
# $h because $_ will be clobbered later and we'll need the original string
# afterwards.
$h = $_ = <>;

if(j(/(\S)\b/g) =~ /(.)\1{4}/) { # flush detector: sorted list of all suits
                                 # contains 5 consecutive identical chars
  # $f=$1 comes first, so $f will be true later if there's a flush.
  # Then o() is called with the flush suit as arg to detect straight flush.
  # If there's no straight flush, o() returns the empty list and for loop
  # runs 0 times, so $w is not set. If there is a straight flush, the return
  # value of o() is compared to 19 to detect royal flush.
  $w = ($_==19 ? "royal" : "straight")
    for o($f=$1);
}

$_ =
  j(/\b(\S)/g)                 # Get the sorted+joined list of ranks...
    =~ s/(.)\1*/length $&/rge; # ... and turn it into a list of sizes of
                               # groups of the same rank. The /r flag
                               # requires perl 5.14 or newer.

print
  $w             ? "$w flush" :
  /4/            ? "four of a kind" :
  /3.*2|[23].*3/ ? "full house" :
  $f             ? "flush" :
  (o".")         ? "straight" :
  /3/            ? "three of a kind" :
  /2.*2/         ? "two pair" :
  /2/            ? "one pair" :
                   "high card"

источник
1

JavaScript 600

использование с nodeJS: node code.js "7H 3S 7S 7D AC QH 7C"

function a(o){s="";for(k in o)s+=o[k];return s}
b=process.argv[2]
c={S:0,H:0,D:0,C:0}
v={A:0,K:0,Q:0,J:0,T:0,"9":0,"8":0,"7":0,"6":0,"5":0,"4":0,"3":0,"2":0}
d=b.split(" ")
for(i=d.length;i--;){e=d[i];c[e[1]]++;v[e[0]]++}
c=a(c);v=a(v)
f=g=h=j=k=l=m=false
if((st=c.indexOf(5))!=-1)g=!g
if(v.match(/[1-9]{5}/))h=!h
if(st==0)f=!f
if(v.indexOf(4)!=-1)j=!j
if(v.indexOf(3)!=-1)k=!k
if(n=v.match(/2/g))if(n)if(n.length>=2)m=!m;else l=!l
p=" of a kind"
q="Flush"
r="Straight"
console.log(f&&g?"Royal "+q:h&&g?r+" "+q:j?"Four"+p:k&&(l||m)?"Full House":g?q:h?r:k?"Three"+p:m?"Two pairs":l?"Pair":"High card")
guy777
источник