Anzahl der String-Permutationen, die Palindrome sind

13

Ihre Eingabe wird eine Zeichenfolge sein, die aus kleinen englischen Buchstaben besteht.

Ihre Aufgabe ist es, die Anzahl der unterschiedlichen Permutationen der ursprünglichen Zeichenfolge zu bestimmen, die ein Palindrom sind.

Die Eingabezeichenfolge enthält bis zu 100 Buchstaben. Bei einer längeren Zeichenfolge kann das Ergebnis sehr groß sein, daher sollte die Ausgabe die Anzahl der Permutationen modulo 666013 sein.

Beispielsweise,

cababaa -> 3

Die möglichen Permutationen sind:

aabcbaa
abacaba
baacaab

Das ist , also gewinnt die kürzeste Antwort!

Andrei Mihailescu
quelle
2
Msgstr "Da die Zeichenkette bis zu 100 Stellen hat, muss das Ergebnis% 666013 sein." In diesem Fall ist es sinnvoll, einen entsprechenden Testfall beizufügen.
Martin Ender
4
Ich verstehe die Zeile% 666013 nicht. Dies ist jedoch eine vielversprechende Herausforderung, und ich wäre bereit, für eine Wiedereröffnung zu stimmen, sobald dies erklärt ist.
12
Oh, jetzt wurde das bearbeitet, ich verstehe, worauf du hinaus willst. Ich glaube nicht, dass diese Linie zu der Herausforderung beiträgt. Meistens werden nur Sprachen ohne Ganzzahlen mit willkürlicher Genauigkeit bestraft. Normalerweise tun wir so etwas wie "Die Antwort sollte korrekt sein, wenn sie in einer hypothetischen Version Ihrer Sprache mit unbegrenzten ganzen Zahlen ausgeführt wird".
7
Das könnte wirklich mehr Testfälle gebrauchen.
smls
3
Vorschläge für Testfälle (bitte überprüfen Sie diese): abcdabcddddd -> 120 (keine ungerade Anzahl von Zeichen) , abcdabcdddddd -> 120 (eine ungerade Anzahl von Zeichen) , abcdabcddddddeee -> 0 (zwei ungerade Anzahl von Zeichen) , aabbccddeeffgghhiijj -> 298735 (beeinflusst durch das Modulo) .
smls

Antworten:

5

Brachylog (2), 15 Bytes

{p.↔}ᶠdl%₆₆₆₀₁₃

Probieren Sie es online!

Erläuterung

{p.↔}ᶠdl%₆₆₆₀₁₃
{   }ᶠdl          Count (l) the number of distinct (d) results (ᶠ) obtainable by:
 p                  permuting {the input}
  .                 to produce an output
   ↔                that, if reversed, is still the output
        %₆₆₆₀₁₃   then take that number modulo 666013

quelle
2
Ich muss definitiv dieses "find unique" implementieren ...
Fatalize
2
@Fatalize: Ja! Ich denke, dass sogar "count unique" bei Herausforderungen häufig genug vorkommt, um eine 1-Byte-Darstellung zu ermöglichen. Auf der anderen Seite tut "modulo 666013" dies mit ziemlicher Sicherheit nicht ;-)
5

05AB1E , 17 16 13 Bytes

-1 Byte von Jonathon Allan

-3 Bytes von Emigna und Adnan

œÙvyÂQO•E›j•%

Erläuterung:

œÙ                # Unique permutations of [implicit] input
  vy              # For each permutation...
    ÂQ            # Check if it is a palindrome
      O           # If so, increment the counter
       •E›j•%     # Modulo 666013 (no clue why this number, or even why this rule is in place)

Probieren Sie es online!

Okx
quelle
1
Der Inhalt E›jstellt die Ziffern dar, [14, 116, 45]die von der Basis konvertiert werden 214, und wird 14*214^2 + 116*214 + 45 = 666013. Ich bin nicht ganz sicher, wo sich die Referenz für die Ziffern befindet, aber sie scheinen mit ihrer Bestellung auf der Infoseite übereinzustimmen . @Adnan kann uns aufklären.
Jonathan Allan
1
@Emigna Einfach, wenn Sie die Sprache kennen: D
Jonathan Allan
1
Sie können 2 Bytes sparen, indem Sie die if-Anweisung entfernen, da Sie sowieso nur die erforderlichen Werte auf dem Stack haben:œÙvyÂQ}O•E›j•%
Emigna
2
@JonathanAllan Die vollständige Auswahl an Ziffern (und Zeichen) finden Sie hier :).
Adnan
1
Aufbauend auf @ Emigna Kommentar, können Sie ein anderes Byte speichern , indem der Schließbügel zu entfernen: œÙvyÂQO•E›j•%.
Adnan
4

Perl 6 , 104 108 88 84 Bytes

{my &f={[*] 1..$_ div 2}
.comb.Bag{*}.&{(2>.grep(*%2))*f(.sum)/[*]($_».&f)%666013}}

Probieren Sie es online!

Wie es funktioniert

Ich kann nicht einfach alle Permutationen generieren und filtern, auch wenn astronomische Laufzeiten zulässig sind, da die integrierte permutationsRoutine von Perl 6 es ablehnt, Listen mit mehr als 20 Elementen zu permutieren, und die Aufgabenbeschreibung Eingaben von bis zu 100 erfordert Zeichen.

Deshalb verwende ich stattdessen eine direkte Formel, die auf den Buchstabenhäufigkeiten der Eingabe basiert:

  1. mein & f = {[*] 1 .. $ _ div 2}

    Eine Hilfsfunktion, die eine Zahl halbiert und auf die nächste Ganzzahl abrundet und dann die Fakultät davon aufnimmt.

  2. .comb.Bag {*}. & {};

    Zählen Sie die Buchstabenhäufigkeiten in der Eingabezeichenfolge auf und machen Sie sie zum Thema für den Rest des Codes. ZB zur Eingabe abcdabcddddddwäre dies die Liste (2, 2, 2, 7).

  3. (2> .grep (*% 2)) *

    Wenn es mehr als eine ungerade Buchstabenfrequenz gibt, multiplizieren Sie das Ergebnis mit Null, da in diesem Fall keine Palindrome möglich sind.

  4. f (.sum) / [*] ($ _ ». & f)

    Berechnen Sie die Anzahl der möglichen Permutationen der Zeichen, die sich auf "einer Seite" jedes Palindroms befinden (was einem Multiset mit den Multiplizitäten entspricht, die durch Halbieren und Untermalen der Eingangsbuchstabenfrequenzen erhalten werden) . Die verwendete Formel stammt aus diesem PDF :
    (n 1 + ... + n k )! / (n 1 ! ⋅ ... ⋅n k 1)
    ZB für die Eingabe von Buchstabenfrequenzen (2,2,2,7)bilden die Buchstaben auf einer Seite des Palindroms ein Multiset mit Multiplizitäten (1,1,1,3), und die Anzahl der Permutationen ist somit (1+1+1+3)! / (1!⋅1!⋅1!⋅3!) = 120.

  5. % 666013

    Nehmen Sie das Ergebnis modulo 666013.

smls
quelle
Gut zu sehen, dass meine alternative Methode gültig ist!
Jonathan Allan
3

Python3, 81 80 Bytes

from itertools import*
lambda s:sum(a==a[::-1]for a in{*permutations(s)})%666013

Dies ist das kürzeste, was ich mir vorstellen konnte. Ich bin mir nicht sicher, ob die Permutationen einfacher generiert werden können ...

Erläuterung

lambda s:                       # Anonymous function taking a parameter <s>. 
    sum(                        # Sum the following terms.
        a==a[::-1]              # Check whether the permutation <a> is a palindrome,
        for a in                # for each element <a>,
        {                       # inside a set that can only contain distinct elements.
            *                   # Splat the elements of the following object:
            permutations(s)     # the permutations of the input parameter <s>.
        }                       #
    )%666013                    # Modulo the sum by 666013.

Anmerkungen

  1. Die Prüfung a==a[::-1]gibt einen booleschen Wert zurück, der von der sum(...)Funktion jedoch implizit in eine Ganzzahl (0 oder 1) umgewandelt und entsprechend summiert wird.
  2. Ich muss den ' splat-Operator ' (nicht den richtigen Namen) verwenden, um die Elemente aus dem Objekt zu extrahieren permutations(...). Andernfalls {...}würde set ( ) nur ein Element enthalten, das Objekt selbst.
  3. Ich verwende ein set ( {...}), um nur bestimmte Permutationen in mir zu behalten.

In Floroid ist dies (fast) z(T(a==aDKaIW(cb(L)))%666013), gibt jedoch stattdessen das Ergebnis aus und nimmt Eingaben über die Befehlszeile entgegen.

Vielen Dank an Jonathan Allan für das Speichern eines Bytes! -> importStil geändert

Probieren Sie es online!

Yytsi
quelle
3

Gelee , 13 Bytes

Œ!QŒḂ€S%“µɲ€’

Probieren Sie es online!

Wie?

Ein roher Forcer.

Œ!QŒḂ€S%“µɲ€’ - Main link: string
Œ!            - all permutations
  Q           - unique
     €        - for each
   ŒḂ         - isPalindromic? (yep a built-in!)
      S       - sum
       %      - mod
        “µɲ€’ - base 250 compressed number 666013

Ich glaube, dass dies effizienter sein wird, aber es sind 30 Bytes (Bearbeiten: Dieses PDF scheint es zu bestätigen, dank der Antwort von smls ):

ÑHḞµS!÷!P$ - Link 1, palindrome count: string a    e.g. 'abcabcd'
Ñ          - call the next link (2) as a monad(a)  e.g. [2, 2, 2, 1]
 H         - halve                                 e.g. [1, 1, 1, 0.5]
  Ḟ        - floor (get pair counts)               e.g. [1, 1, 1, 0]
   µ       - start a new monadic chain - call that p
    S      - sum(p)                                e.g. 3
     !     - factorial                             e.g. 6
         $ - last 2 links as a monad:
       !   -     factorial(p) (vectorises)         e.g. [1, 1, 1, 1]
        P  -     product                           e.g. 1
      :    - integer division                      e.g. 6

ĠL€ - Link 2, count characters: string a           e.g. 'abcabcd'
Ġ   - group indexes                                e.g. [[1, 4], [2, 5], [3, 6], 7]
 L€ - length of €ach                               e.g. [2, 2, 2, 1]

ÇḂS⁼LḂ$aÑ%“µɲ€’ - Main link: string a              e.g. 'abcabcd'
                - first check to see if any palindromes will be possible:
Ç               - last link (2) as a monad         e.g. [2, 2, 2, 1]
 Ḃ              - mod 2                            e.g. [0, 0, 0, 1]
  S             - sum                              e.g. 1
      $         - last two links as a monad:
    L           -     length(a string)             e.g. 7
     Ḃ          -     mod 2                        e.g. 1
   ⁼            - equal?                           e.g. 1 (1 when palindromes are possible)
       a        - and
        Ñ       - next link as a monad             e.g. 6
         %“µɲ€’ - mod 666013, as in the brute force version.
Jonathan Allan
quelle
Ja das tut es, das %ist mod.
Jonathan Allan
Gah, ich wollte gerade genau diese Antwort posten, bin aber nicht rechtzeitig dort angekommen, weil ich zuerst die Brachylog-Antwort gepostet habe. Eine Frage der Sekunde, denke ich. Klar sollte ich daran denken, dass Jelly eine populärere Sprache als Brachylog ist und deshalb sollte ich zuerst an diesem Beitrag arbeiten.
Wow, Byte für Byte? Ich habe noch 13, aber denke, es ist etwas weniger effizient :)
Jonathan Allan
Ist die Zahl in einer anderen Basis komprimiert oder was? : P
Yytsi
Von meinem TIO-Tab Œ!QŒḂ€S%“µɲ€’. Das sieht für mich bytewise identisch aus.
2

Mathematica, 46 Bytes

Permutations@#~Count~_?PalindromeQ~Mod~666013&

Nimmt eine Liste von Zeichen als Eingabe.

Schrecklich ineffizient, weil es tatsächlich alle Permutationen der Eingabe generiert und dann die palindromischen zählt.

Martin Ender
quelle
Ich denke, dies gibt falsch eine positive Antwort, anstatt 0, wenn die Zeichenfolge mehrere Buchstaben mit ungerader Multiplizität (wie "abcdabcddddddeee") enthält.
Greg Martin
@ GregMartin Danke, behoben. Das war sowieso unnötig kompliziert.
Martin Ender
2

Mathematica, 68 Bytes

If[i=Floor[t=Last/@Tally@#/2];Tr[t-i]<1,Multinomial@@i,0]~Mod~666013

Reine Funktion, die eine Liste von Zeichen als Eingabe verwendet und eine Ganzzahl zurückgibt. Nicht so kurz wie die Mathematica-Antwort von Martin Ender , aber es ist trotzdem eine nette Herangehensweise, die dieselbe Herangehensweise zu sein scheint wie in der Perl 6-Antwort von smls .

Zuerst t=Last/@Tally@#/2berechnet die Zählungen aller unterschiedlichen Zeichen in der Eingabe, dividiert durch 2; dann i=Floorabrundet irgendwelche Brüche in auftreten t. Beachten Sie, dass palindromische Permutationen der Eingabe genau dann vorliegen, wenn höchstens eine ungerade Zahl unter den ursprünglichen Zählwerten vorhanden ist, dh wenn höchstens ein Bruch in vorhanden ist t. Wir können das testen, indem wir einfach alle Elemente von t-i(using Tr) addieren : Wenn die Antwort kleiner als ist 1, gibt es palindromische Permutationen, andernfalls nicht.

Wenn ija , dann repräsentiert das die Anzahl der verschiedenen Zeichen in der linken Hälfte der Permutationen, die beliebig angeordnet werden können. Die Anzahl der Möglichkeiten, dies zu tun, ist genau der MultinomialKoeffizient (ein Quotient bestimmter Fakultäten), den Mathematica eingebaut hat.

Greg Martin
quelle
1

k, 23 Bytes

{666013!+/{x~|x}'cmb x}

Wenn Sie OK verwenden oder cmbnicht vorhanden sind, verwenden Sie prmanstelle von cmb.

zgrep
quelle
1

C ++ 14, 161 Bytes

Als unbenanntes Lambda wird angenommen, dass die Eingabe wie folgt ist std::stringund über den Referenzparameter zurückgegeben wird.

#import<algorithm>
[](auto s,int&n){auto a=s.begin(),b=s.end();std::sort(a,b);n=0;do n=(n+std::equal(a,b,s.rbegin()))%666013;while(std::next_permutation(a,b));}

Ungolfed und Nutzung:

#include<iostream>
#include<string>

#import<algorithm>
auto f=
[](auto s,int&n){
 auto a=s.begin(),b=s.end();
 std::sort(a,b);
 n=0;
 do
  n=(n+std::equal(a,b,s.rbegin()))%666013;
 while(std::next_permutation(a,b));
}
;

using namespace std;


int main(){
 string s;
 s = "cababaa";
 s = "abcdabcddddd";
 int n;
 f(s,n);
 cout << n << endl;
}
Karl Napf
quelle
1

Ruby, 67 57 52 59 Zeichen

->s{s.chars.permutation.uniq.count{|t|t.reverse==t}%666013}
Dorian
quelle
Codegolf-Einsendungen sollten richtige Programme / Funktionen / Lambdas sein, keine Schnipsel . Ich bin kein Ruby-Programmierer, aber ich denke, Sie können daraus ein Lambda machen, indem Sie es einpacken ->s{ }, nicht wahr?
smls
Ist das Argument laut Dokumentation nicht (s.size)überflüssig?
smls
1
Ich habe es auf Ruby 2.4 getestet und es funktioniert auch ohne .to_a.
smls
@smls Funktioniert nicht auf Ruby 2.3.3 ( undefined method uniq 'für # <Enumerator`), aber richtig, es funktioniert auf Ruby 2.4, danke :)
Dorian
Muss das Ergebnis genommen werden mod 666013?
NonlinearFruit
1

Japt , 20 bis 18 Bytes

á f_¥Zw} l %666013

2 Bytes gespart dank ETHproductions.

Probieren Sie es online!

Tom
quelle
Schön, das hätte ich getan. Sie können mit zwei Bytes speichern f_¥Zw}, wie _es kurz fürZ{Z
ETHproductions
á fêS â l %666013würde dir ein byte sparen.
Powelles
0

MATL, 13 Bytes

Y@Xu!"@tP=Avs

Probieren Sie es bei MATL Online aus

Erläuterung

        % Implicitly grab input as a string
Y@      % Compute all permutations of the inputs (one per row)
Xu      % Determine the unique rows
!       % Take the transpose so each permutation is a column
"       % For each unique permutation
  @     % Take this permutation
  tP=A  % Duplicate it, reverse it, and compare (yields 1 for palindrome and 0 otherwise)
  v     % Vertically concatenate the entire stack
  s     % Compute the sum of all elements
        % Implicitly end for loop and display the result
Suever
quelle
0

CJam , 19 Bytes

qe!{_W%=}%:+666013%

Probieren Sie es online!

Erläuterung:

qe! {_ W% =}%: + 666013% e # Volles Programm.
qe # Alle Eingaben abrufen.
 e! e # Holen Sie sich alle eindeutigen Permutationen.
   {_W% =} e # Funktion zum Überprüfen, ob eine Liste ein Palindrom ist.
    _ e # ToS duplizieren.
     W% e # ToS umkehren (Push -1, Modularer Index von ToS).
       = e # Überprüfe, ob ToS gleich SToS ist.
         % e # Map.
          : + e # Summe (durch Addition reduzieren).
            666013 e # Drücken Sie 666013.
                  % e # Modulo.

Erik der Outgolfer
quelle
0

Ohm, 17 Bytes

I⌐:_₧?¡;;¼,

Erläuterung:

I⌐:_₧?¡;;¼,  ■Main wire
I⌐:     ;    ■for _ in permutations(input()){
   _₧? ;     ■  if(palindrome(_)){
      ¡      ■    counter++;
       ;     ■  }
        ;    ■}
         ¼,  ■print(counter)
Roman Gräf
quelle
0

PHP, 182 Bytes

function f($a,$b=1){return$a?f($a-1,bcmul($a,$b)):$b;}$a=count_chars($argv[1],$r=1);foreach($a as$v){$c+=$v%2?:0;$c>1?die("0"):$z+=$f=$v/2^0;$r=bcmul(f($f),$r);}echo bcdiv(f($z),$r);

Online Version

Nervenzusammenbruch

function f($a,$b=1){  #Factorial
    return$a?f($a-1,bcmul($a,$b)):$b;
}
$a=count_chars($argv[1],$r=1); # Array count for every char
foreach($a as$v){
    $c+=$v%2?:0; # counter mod 2 ==1
    $c>1?die("0"):$z+=$f=$v/2^0; # end program if there are 2 chars which cannot divide by 2
    $r=bcmul(f($f),$r);
}
echo bcdiv(f($z),$r);
Jörg Hülsermann
quelle