Знаходження «субліндіндромів».


24

Найкоротший код, який знаходить усі унікальні "підпаліндри" рядка, тобто будь-яку підрядку довжиною> 1, що є паліндром.

напр.1

input: "12131331"
output: "33", "121", "131", "313", "1331"

напр.2

input: "3333"
output: "33", "333", "3333"

1
Чи може бути струна, це власний підпаліндром? Оскільки рядок - це власна підрядка.
JPvdMerwe

@JPvdMerwe: Так, поза курсом.
Ельвекс

Насправді важливіше: яким повинен 333бути результат? Наївно ви би закінчили друк 33двічі
JPvdMerwe

@JPvdMerwe: '333' -> '33', '333'. Я відповідно відредагую питання. Спасибі.
Ельвакс

Як визначається вихід? Коми, розмежовані комами, цитатами звучать кожен підпаліндром, як ви демонструєте тут? Один суб-р на рядок?
Joey

Відповіді:


11

J, 24 31 40

~.(#~(1<#*]-:|.)&>),<\\.

Використання зразка:

   ~.(#~(1<#*]-:|.)&>),<\\. '12131331'
┌───┬───┬───┬────┬──┐
│121│131│313│1331│33│
└───┴───┴───┴────┴──┘
   ~.(#~(1<#*]-:|.)&>),<\\. '3333'
┌──┬───┬────┐
│33│333│3333│
└──┴───┴────┘

Візьміть це, GolfScript!


Зізнайся, ти просто поставив /dev/randomсюди дамп, щоб нас обдурити ;-)
Джої,

@Joey спробуй сам; p (TBH, я не вірив, що спочатку це може спрацювати)
JB

Я впевнений, що це фактичний код. Я провів вихідні, намагаючись обернути голову навколо J, але зазнав невдач. Я все-таки розпізнаю код; Я просто не розумію, що це робить ;-)
Джої,

2
Не можна цього скоротити до ~.(#~(1<#*]-:|.)&>),<\\.(24 символів)?
ефемія

@ephemient Це дійсно так. (Схоже, я застряг у ментальному режимі "відповідь має бути функцією", який тут не застосовується.) Відредаговано, дякую!
JB

7

Пітон 124

r=raw_input()
l=range(len(r))
print', '.join(set('"'+r[i:j+1]+'"'for i in l for j in l if i<j and r[i:j+1]==r[i:j+1][::-1]))

5

Haskell 98, 88 91 96

import List
main=interact$show.filter(\x->length x>1&&x==reverse x).nub.(tails=<<).inits

3

Пітон - 138 136

Цей код не дублює підпаліндром.

r=raw_input()
i,l=0,len(r)
j=l
a=[]
while i<l-1:
 t=r[i:j];j-=1
 if t==t[::-1]:a+=['"'+t+'"']
 if j<i+2:i+=1;j=l
print", ".join(set(a))

1
Змініть '"'+t+'"'на, tщоб заощадити місце, хоча воно використовує одиничні лапки.
Thomas O

3

Рубі - 126 102 97 символів

s=gets
*m=*0..s.size
puts m.product(m).map{|h,j|(c=s[h,j+1]).size>1&&c==c.reverse ? c:0}.uniq-[0]

3

Гольфскрипт, 48 символів

subpalindrome.gs

{,}{(;}/{{,}{);}/}%{+}*{.,1>\.-1%=*},.&{`}%", "*

Використання:

echo "12131331" | ruby golfscript.rb subpalindrome.gs

Перша операція {,}{(;}/перетворює рядок у список трейлінг-підрядків. Потім аналогічне перетворення провідних підрядів відображається над результатом. Потім вирівняйте {+}*, відфільтруйте паліндром за допомогою присудка .,1>\.-1%=*, захопіть унікальні значення .&, а потім симпатичний друк.

Було б акуратніше витягнути перетворення трейлінг-підрядів як блок і повторно використовувати його як заміну для ведучих підрядів після зміни кожної трейлінг-підрядки, але я не можу знайти чіткий спосіб зробити це.


2

Хаскелл - 170 , 153

import Data.List
import Data.Set
p a=fromList$[show x|x<-subsequences a,x==reverse x,length x>1]
main=getLine>>=(\x->putStrLn$intercalate", "$toList$p x)

Замініть main=getLine>>=(\x->putStrLn$intercalate", "$toList$p x)на main=getLine>>=putStrLn.intercalate", ".toList.p. Я б також замінив заклик pсвоїм тілом.
Ясір Арсанукаєв

Підрядки / = subsequences! Ваша програма повідомляє більше субпаліндром, ніж контрольний вихід, наприклад 1. (наприклад, "1111")
JB


2

Пролог, 92

f(S,P):-append([_,X,_],S),X=[_,_|_],reverse(X,X),atom_codes(P,X).
p(S,R):-setof(P,f(S,P),R).

Використання зразка:

?- p("12131331",R).
R = ['121', '131', '1331', '313', '33'].

?- p("3333",R).
R = ['33', '333', '3333'].

2

Windows PowerShell, 104 109 111

0..($l=($s="$input").length-1)|%{($a=$_)..$l|%{-join$s[$a..$_]}}|sort -u|?{$_[1]-and$_-eq-join$_[$l..0]}

Це очікує введення на stdin і викине всі знайдені паліндроми по одному на рядок у stdout:

PS Home:\SVN\Joey\Public\SO\CG183> '12131331'| .\subp.ps1
33
121
131
313
1331

(Коли запуск з cmdнього стає echo 12131331|powershell -file subp.ps1- це просто те, що $inputприймає дещо інше значення залежно від того, як називався сценарій, але це може бути stdin, тільки не інтерактивно.)

2011-01-30 13:57 (111) - Перша спроба.

2011-01-30 13:59 (109) - Вкладене оголошення змінної.

2011-06-02 13:18 (104) - Знаходження підрядків Redone, приєднавшись до масиву char замість виклику .Substring()та накресливши ще трохи.


2

Q, 78

{a::x;(?)(,/)b@'(&:')({x~(|:)x}'')b:-1_1_'({(sublist[;a]')x,'1+c}')c::(!)(#)a}

використання

q){a::x;(?)(,/)b@'(&:')({x~(|:)x}'')b:-1_1_'({(sublist[;a]')x,'1+c}')c::(!)(#)a}"12131331"
"121"
"131"
"313"
"1331"
"33"
q){a::x;(?)(,/)b@'(&:')({x~(|:)x}'')b:-1_1_'({(sublist[;a]')x,'1+c}')c::(!)(#)a}"3333"
"33"
"333"
"3333"

2

Сітківка , 34 27 байт

&@!`(.)+.?(?<-1>\1)+(?(1)^)

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

Тестовий набір потрібен, Mоскільки після нього слід ще один етап, щоб вставити порожні рядки між тестовими прикладами.

Пояснення

&@!`(.)+.?(?<-1>\1)+(?(1)^)

Роздрукуйте ( !) всі унікальні ( @), що збігаються ( &) збіги регулярного виразів (.)+.?(?<-1>\1)+(?(1)^). Це відповідає паліндрому довжиною 2 або більше, використовуючи балансуючі групи. Існує застереження до частини "всі збіги, що перекриваються": ми можемо отримати максимум один матч за стартову позицію. Однак якщо два паліндроми різної довжини починаються в одному положенні, коротший паліндром знову з’явиться в кінці більш тривалого паліндрому. А оскільки жадібність +пріоритетів довше збігається, все одно ми отримуємо всі паліндроми.


2

05AB1E , 11 10 байт

ŒÙʒÂQ}žQSK

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



@scottinet не вдається для синглів, EG1234142141410010101000
Magic Octopus Urn

1
Твоє теж, але не тим самим. o_O Щось відбувається, що потребує розслідування. Тим часом, ось 10-
байтна

Була помилка з унікальним, я її виправив. Тепер і ваша відповідь на 11 байт, і моя робота в 9 байтах :-)
scottinet

@scottinet Ваш 10-байтний також може бути 9-байтовим, змінивши 1›на . :)
Кевін Круїссен

1

Перл, 112

$_=<>;chop;s/./$&$' /g;
map{/../&&$_ eq reverse&&$h{$_}++}split/ /
  for grep{s/./$`$& /g}split/ /;
print for keys %h

1

JavaScript (ES6), 120 байт

a=>{for(b=0,c=d=a.length,e=[];b<d;--c<b+2?(b++,c=d):1)(f=a.slice(b,c))==f.split``.reverse().join``&&e.push(f);return e}

Ця функція приймає рядок як вхід і виводить масив.


1

Clojure, 81 байт

#(set(for[i(range 2(+(count %)1))p(partition i 1 %):when(=(reverse p)(seq p))]p))

forІдеальне поєднання тут :) Можливо, :when(=(reverse p)p)якщо введення було списком символів АБО ціла рядок не вважалася паліндром, насправді в цьому випадку iможе бути (count %)також і максимальний діапазон .

Найбільш компактний чохол для довідки:

#(set(for[i(range 2(count %))p(partition i 1 %):when(=(reverse p)p)]p))

1

Пітона, 83 102 символи

s=lambda t:(t[1:]or())and(t,)*(t==t[::-1])+s(t[1:])+s(t[:-1])
print set(s(input()))

Фраза (t[1:]or())and...рівнозначна (...)if t[1:]else()і рятує один символ! Я надто пишаюся цим, враховуючи заощадження.

Приклад:

python x
"51112232211161"
set(['11', '22', '11122322111', '161', '111', '112232211', '1223221', '22322', '232'])

1

Scala 127

object p extends App{val s=args(0);print(2.to(s.size).flatMap(s.sliding(_).toSeq.filter(c=>c==c.reverse)).toSet.mkString(" "))}

Щоб зберегти це порівняння з яблуками та яблуками порівняно з іншими відповідями Скали, я також зробив шахту об'єктом, який розширює додаток. Замість того, щоб повторювати вхідний рядок вручну та використовувати підрядку, я скористався ковзанням (), щоб створити послідовність усіх підрядків для мене.


1

Scala 156 170

object o extends App{val l=args(0).length-2;val r=for(i<-0 to l;j<-i to l;c=args(0).substring(i,j+2);if(c==c.reverse))yield c;print(r.toSet.mkString(" "))}

object o{def main(s:Array[String]){val l=s(0).length-2;val r=for(i<-0 to l;j<-i to l;c=s(0).substring(i,j+2);if(c==c.reverse)) yield c;println(r.distinct.mkString(" "))}}


Привіт Лалит, я трохи скоротив ваш код: Немає порожнього перед виходом та розширенням програми замість перезапису головного, println => друку та різного => toSet
невідомого користувача

1

Perl 6 ,  35  32 байти

{unique m:ex/(.+).?<{$0.flip}>/}

Перевірте це

{set m:ex/(.+).?<{$0.flip}>/}

Перевірте це

Розширено:

{  # bare block lambda with implicit parameter 「$_」

  set             # turn into a Set object (ignores duplicates)

  \             # stringify 「~」 all of these 「«」 (possibly in parrallel)
                  # otherwise it would be a sequence of Match objects

  m               # match
  :exhaustive     # in every way possible
  /
    ( .+ )        # at least one character 「$0」
    .?            # possibly another character (for odd sized sub-palindromes)
    <{ $0.flip }> # match the reverse of the first grouping
  /
}



1

APL (Dyalog Classic) , 27 байт

{∪⍵/⍨≡∘⌽¨⍨⍵}∘⊃(,/1↓⍳∘≢,/¨⊂)

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

{∪⍵/⍨≡∘⌽¨⍨⍵}∘⊃(,/1↓⍳∘≢,/¨⊂)    Monadic train:
                                Enclose the input, '12131331'
                     ⍳∘≢          Range from 1 to length of input
                     ⍳∘≢,/¨⊂      List of list of substrings of each length
                   1            Remove the first list (length-1 substrings)
                ,/              Put the rest of the substrings into a single list.
{∪⍵/⍨≡∘⌽¨⍨⍵}                   To the result, apply this function which
                                   keeps all palindromes from a list:
      ≡∘⌽¨⍨⍵                    Boolean value of whether each (¨) string in argument
      ≡∘⌽                      is equal to its own reverse

  ⍵/⍨                           Replicate (filter) argument by those values.
                                 This yields the length >1 palindromes.
                                Remove duplicates from the list of palindromes.

Через те, що ОП вимагає "коду", фрагмент ∪w/⍨≡∘⌽¨⍨w←⊃,/1↓(⍳∘≢,/¨⊂)дійсний.
Адам

@ Adám, я думаю, що я дотримуватимусь такої відповіді, як це робиться заради сучасних стандартів сайту, тим більше, що він не отримує загальної виграші.
lirtosiast



1

Брахілог , 11 байт

{s.l>1∧.↔}ᵘ

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

(Заголовок у посиланні розбивається під час публікації, тому ось присудок (еквівалент функції в Брахілозі) лише для першого тестового випадку, а wв кінці фактично друкується вихід.)

               The output is
{        }ᵘ    a list containing every possible unique
 s.            substring of
               the input
   l           the length of which
    >          is greater than
     1         one
      ∧        and
       .       which
        ↔      reversed
               is itself. (implicit output within the inline sub-predicate)

Я відчуваю, що існує коротший спосіб перевірити, чи довжина більша за 1. (Якби не відфільтрували тривіальні паліндроми, це було б просто {s.↔}ᵘ.)


1

APL (NARS), 65 символів, 130 байт

{0=≢m←∪b/⍨{1≥≢⍵:0⋄∧/⍵=⌽⍵}¨b←↑∪/{x[⍵;]⊂y}¨⍳≢x←11 1‼k k⊢k←≢y←⍵:⍬⋄m}

тест:

  r←{0=≢m←∪b/⍨{1≥≢⍵:0⋄∧/⍵=⌽⍵}¨b←↑∪/{x[⍵;]⊂y}¨⍳≢x←11 1‼k k⊢k←≢y←⍵:⍬⋄m}
  o←⎕fmt
  o r '1234442'
┌2───────────┐
│┌2──┐ ┌3───┐│
││ 44│ │ 444││
│└───┘ └────┘2
└∊───────────┘
  o r '3333'
┌3───────────────────┐
│┌4────┐ ┌3───┐ ┌2──┐│
││ 3333│ │ 333│ │ 33││
│└─────┘ └────┘ └───┘2
└∊───────────────────┘
  o r  "12131331"
┌5─────────────────────────────────┐
│┌4────┐ ┌3───┐ ┌2──┐ ┌3───┐ ┌3───┐│
││ 1331│ │ 121│ │ 33│ │ 313│ │ 131││
│└─────┘ └────┘ └───┘ └────┘ └────┘2
└∊─────────────────────────────────┘
  o r '1234'
┌0─┐
│ 0│
└~─┘


{0=≢m←∪b/⍨{1≥≢⍵:0⋄∧/⍵=⌽⍵}¨b←↑∪/{x[⍵;]⊂y}¨⍳≢x←11 1‼k k⊢k←≢y←⍵:⍬⋄m}
 y←⍵  assign the argument to y (because it has to be used inside other function)
 x←11 1‼k k⊢k←≢y   assign the lenght of y to k, call the function 11 1‼k k
                   that seems here find all partition of 1 2 ..k
 {x[⍵;]⊂y}¨⍳≢      make partition of arg ⍵ using that set x
 ∪/                set union with precedent to each element of partition y (i don't know if this is ok)
 b←↑               get first assign to b
 {1≥≢⍵:0⋄∧/⍵=⌽⍵}¨ for each element of b return 1 only if the argument ⍵ is such that 
                   "∧/⍵=⌽⍵" ⍵ has all subset palindrome, else return 0
 b/⍨               get the elements in b for with {1≥≢⍵:0⋄∧/⍵=⌽⍵} return 1
 m←∪               make the set return without ripetition element, and assign to m
 0=≢               if lenght of m is 0 (void set) than 
 :⍬⋄m              return ⍬ else return m

хтось краще знає, чому, і може пояснити це краще, без змін все це ... Я не настільки впевнений у цьому коді, можливо, якщо тестових прикладів буде більше, щось піде не так ...


1

Japt , 9 байт

ã â fÅfêU

Спробуй це

ã â fÅfêU     :Implicit input of string
ã             :Substrings
  â           :Deduplicate
    f         :Filter elements that return truthy
     Å        :  Slice off first character
       f      :Filter elements that return true
        êU    :  Test for palindrome

0

Java 8, 202 201 199 байт

import java.util.*;s->{Set r=new HashSet();String x;for(int l=s.length(),i=0,j;i<l;i++)for(j=i;++j<=l;)if((x=s.substring(i,j)).contains(new StringBuffer(x).reverse())&x.length()>1)r.add(x);return r;}

Спробуйте тут.

Якщо функція заборонена і потрібна повна програма, це замість 256 255 253 байтів :

import java.util.*;interface M{static void main(String[]a){Set r=new HashSet();String x;for(int l=a[0].length(),i=0,j;i<l;i++)for(j=i;++j<=l;)if((x=a[0].substring(i,j)).contains(new StringBuffer(x).reverse())&x.length()>1)r.add(x);System.out.print(r);}}

Спробуйте тут.

Пояснення:

import java.util.*;      // Required import for Set and HashSet

s->{                     // Method with String parameter and Set return-type
  Set r=new HashSet();   //  Return-Set
  String t;              //  Temp-String
  for(int l=s.length(),  //  Length of the input-String
          i=0,j;         //  Index-integers (start `i` at 0)
      i<l;i++)           //  Loop (1) from `0` to `l` (exclusive)
    for(j=i;++j<=l;)     //   Inner loop (2) from `i+1` to `l` (inclusive)
      if((t=s.substring(i,j) 
                         //    Set `t` to the substring from `i` to `j` (exclusive)
         ).contains(new StringBuffer(t).reverse())
                         //    If this substring is a palindrome,
         &t.length()>1)  //    and it's length is larger than 1:
        r.add(t);        //     Add the String to the Set
                         //   End of inner loop (2) (implicit / single-line body)
                         //  End of loop (1) (implicit / single-line body)
  return r;              //  Return the result-Set
}                        // End of method

0

JavaScript (ES6), 107 байт

Повертає набір .

s=>new Set((g=(s,r=[...s].reverse().join``)=>s[1]?(r==s?[s]:[]).concat(g(s.slice(1)),g(r.slice(1))):[])(s))

Тестові справи

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