Pisin palindromi-alijono etsiminen

Kokeneet kirjoittajat eivät ole vielä tarkistaneet sivun nykyistä versiota, ja se voi poiketa merkittävästi 16.7.2020 tarkistetusta versiosta . tarkastukset vaativat 2 muokkausta .

Tietojenkäsittelytieteessä pisin palindrominen osamerkkijono -  ongelma on ongelma löytää pisin osamerkkijonosta , joka on palindromi . Esimerkiksi "banaanin" pisin palindrominen osajono on "anana". Pisin palindrominen osajono ei välttämättä ole ainutlaatuinen; esimerkiksi merkkijonossa "abracadabra" ei ole palindromista osamerkkijonoa, joka on pidempi kuin kolme merkkiä, vaan on sellaisia, jotka koostuvat täsmälleen kolmesta merkistä, nimittäin "aka" ja "ada". Jotkut sovellukset haluavat löytää kaikki maksimaaliset palindromiset osamerkkijonot (eli kaikki alimerkkijonot, jotka ovat itse palindromeja ja joita ei voida täyttää pidempiin palindromisiin alimerkkijonoihin) sen sijaan, että palautettaisiin vain yksi osamerkkijono tai palindromisen osamerkkijonon enimmäispituus.

Manacher (1975 ) keksi aikalineaarisen algoritmin luettelemaan kaikki palindromit tietyn merkkijonon alussa. Kuitenkin, kuten Apostolico, Breslauer & Galil (1995 ) osoittavat, samaa algoritmia voidaan käyttää kaikkien pisimpien palindromisten alijonojen löytämiseen mistä tahansa annetusta merkkijonosta, jälleen lineaarisessa ajassa. Siksi se tarjoaa ratkaisun ongelmaan löytää suurin palindromiosamerkkijono lineaarisessa ajassa. Vaihtoehtoisia lineaarisia aikaratkaisuja ovat ehdottaneet Jeuring (1994 ) ja Gusfield (1997 ), jotka kuvasivat ratkaisun, joka perustuu päätepuiden käyttöön . Tehokkaat rinnakkaisalgoritmit tämän ongelman ratkaisemiseksi tunnetaan myös. [yksi]

Pisimmän palindromisen alijonon löytämisen ongelmaa ei pidä sekoittaa pisimmän palindromisen osajonon löytämisen ongelmaan .

Manakerin algoritmi

Löytääkseen pisimmän palindromin merkkijonosta lineaarisessa ajassa algoritmi voi käyttää seuraavia palindromien ja alipalindromien ominaisuuksia:

  1. Palindromin vasen puoli on sen oikean puolen peilikuva.
  2. (Tapaus 1) Kolmannella palindromilla, jonka keskipiste on ensimmäisen palindromin oikealla puolella, on täsmälleen sama pituus kuin toisella palindromilla, jonka keskipiste peilataan vasemmalle puolelle, jos toinen palindromi sijaitsee ensimmäisen sisällä ja vetäytyy rajalta ainakin yhdelle hahmolle.
  3. (Tapaus 2) Jos toisella palindromilla on yhteinen raja ensimmäisen palindromin kanssa tai se ulottuu sen yli, niin kolmannen palindromin pituus on taatusti vähintään etäisyys sen keskustasta ensimmäisen palindromin oikeaan reunaan. Tämä pituus on etäisyys toisen palindromin keskustasta ensimmäisen palindromin vasemmanpuoleiseen merkkiin.
  4. Kolmannen palindromin pituuden löytämiseksi tapauksessa 2, on tarpeen verrata ensimmäisen palindromin oikeanpuoleisimman merkin jälkeen olevia merkkejä peilikuvaansa kolmannen palindromin keskustasta, kunnes joko merkkijono on kulunut loppuun tai hahmoja löytyy.
  5. (Tapaus 3) Ensimmäinen tai toinen palindromi eivät anna tietoa sellaisen neljännen palindromin pituuden määrittämiseksi, jonka keskipiste on ensimmäisen palindromin rajan ulkopuolella.
  6. Siksi merkkijonon alimerkkijonojen palindromipituuksien määrittämiseksi vasemmalta oikealle on toivottavaa, että sinulla on viitepalindromi (toimii ensimmäisenä palindromina), jonka merkit ovat merkkijonon oikeanpuoleisimmilla paikoilla (ja siten kolmas palindromi tapauksessa 2 ja neljäs palindromi tapauksessa 3 voi korvata ensimmäisen palindromin, jolloin siitä tulee uusi vertailupalindromi).
  7. Mitä tulee ajan arvioimiseen merkkijonon kunkin merkin palindromipituuden määrittämiseen: tapauksessa 1 merkkien vertailua ei tehdä, tapauksissa 2 ja 3 vain ne merkkijonon merkit, jotka ovat viitepalindromin oikeanpuoleisimman merkin takana, ovat ehdokkaita. vertailu (ja siksi tapaus 3 johtaa aina muutokseen vertailupalindromissa, kun tapaus 2 muuttaa vertailupalindromia vain, jos käy ilmi, että kolmannen palindromin pituus on itse asiassa suurempi kuin sen luvattu vähimmäispituus).
  8. Parillisen asteen palindromeille keskus on palindromin kahden keskisymbolin välissä.

Toteutus

Päästää:

Tulos: Pisin palindromi eli merkkijonon ensimmäinen merkki

#sisällytä <merkkijono> #sisällytä <vektori> käyttäen nimiavaruutta std ; string longestPalindromi ( const string & s ){ vektori < char > s2 ( s . koko ( ) * 2 + 1 , '#' ); //luo pseudomerkkijono, jonka #-raja on ( int i = 0 ; i != s . size (); ++ i ){ s2 [ i * 2 + 1 ] = s [ i ]; } intp [ s2 . _ koko ()]; int r , c , indeksi , i_mir ; int maxLen = 1 ; i_mir = indeksi = r = c = 0 ; for ( int i = 1 ; i != s2 . koko () - 1 ; ++ i ){ i_mir = 2 * c - i ; //Jos osumme edellisen tuloksen säteeseen, //niin nykyisen säteen alkuarvo voidaan ottaa peilituloksesta p [ i ] = r > i ? min ( p [ i_mir ], r - i ) : 0 ; while ( i > p [ i ] && ( i + p [ i ] + 1 ) < s2 . koko ( ) && s2 [ i - p [ i ] - 1 ] == s2 [ i + p [ i ] + 1 ] ) ++ p [ i ]; if ( p [ i ] + i > r ){ c = i ; r = i + p [ i ]; } if ( maxLen < p [ i ]){ maxLen = p [ i ]; indeksi = i ; } } //Yhdistä löydetyt paikat alkuperäiseen merkkijonon palautukseen s . substr (( indeksi - maxLen ) / 2 , maxLen ); }

Muistiinpanot

  1. Crochemore & Rytter (1991 ), Apostolico, Breslauer & Galil (1995 ).

Kirjallisuus

Linkit

Tämä artikkeli sisältää tekstiä PEGWikin pisimmästä palindromisesta osamerkkijonosta Creative Commons Attribution ( CC-BY-3.0 ) -lisenssillä.