Scrivi una funzione che restituisca il palindromo più lungo in una data stringa


102

ad es. "ccddcc" nella stringa "abaccddccefe"

Ho pensato a una soluzione ma funziona in tempo O (n ^ 2)

Algo 1:

Passi: è un metodo di forza bruta

  1. Avere 2 cicli
    for per i = 1 a i minore di array.length -1
    per j = i + 1 aj minore di array.length
  2. In questo modo puoi ottenere sottostringa di ogni possibile combinazione dall'array
  3. Avere una funzione palindromo che controlla se una stringa è palindromo
  4. quindi per ogni sottostringa (i, j) chiamate questa funzione, se è un palindromo memorizzatela in una variabile stringa
  5. Se trovi la successiva sottostringa palindromo e se è maggiore di quella attuale, sostituiscila con quella attuale.
  6. Finalmente la tua variabile stringa avrà la risposta

Problemi: 1. Questo algoritmo viene eseguito in tempo O (n ^ 2).

Algo 2:

  1. Invertire la stringa e memorizzarla in un array diverso
  2. Ora trova la sottostringa corrispondente più grande tra entrambi gli array
  3. Ma anche questo viene eseguito in tempo O (n ^ 2)

Ragazzi, riuscite a pensare a un algoritmo che funziona in tempi migliori. Se possibile O (n) tempo


42
Penso che il primo sia O(n^2)ottenere le sottostringhe * O(n)per verificare se sono palindromi, per un totale di O(n^3)?
Skylar Saveland

E se sapessi che stavo lavorando con palindrome e salvassi le mie stringhe come due metà e poi se usassi Java avrei fatto il controllo O (1) per la funzione?
viki.omega9

10
Il secong algo è corretto? E la stringa: "abcdecba". La più grande sottostringa corrispondente è ("abcdecba" vs. "abcedcba"): "abc" o "cba". Tuttavia, entrambi non sono palindromi.
Yarneo

@Learner, solo curioso, nei tuoi passaggi sopra a quale array ti riferisci nei tuoi cicli for? Per array ti riferisci alla stringa? String.length?
Zolt

1
per chi cerca la risposta con O (n ^ 2) - geeksforgeeks.org/longest-palindrome-substring-set-1
Shirish Herwade

Risposte:


76

Puoi trovare il palindromo più lungo usando l'algoritmo di Manacher in O(n)tempo! La sua implementazione può essere trovata qui e qui .

Per l'input String s = "HYTBCABADEFGHABCDEDCBAGHTFYW1234567887654321ZWETYGDE"trova l'output corretto che è 1234567887654321.


3
Non capisco come sia lineare. vedo un whileincorporato nel forcon un limite che sembra simile al ciclo esterno.
v.oddou


9

L'Algo 2 potrebbe non funzionare per tutte le corde. Di seguito è riportato un esempio di tale stringa "ABCDEFCBA".

Non che la stringa abbia "ABC" e "CBA" come sottostringa. Se si inverte la stringa originale, sarà "ABCFEDCBA". e la sottostringa corrispondente più lunga è "ABC" che non è un palindromo.

Potrebbe essere necessario controllare inoltre se questa sottostringa corrispondente più lunga è effettivamente un palindromo che ha il tempo di esecuzione di O (n ^ 3).


2
È importante notare che Algo 2 dovrebbe funzionare per il "palindromo di sottosequenza di corrispondenza più lunga" che è un problema comune degli algoritmi in cui i caratteri di sottosequenza possono anche essere separati all'interno della stringa. Ad esempio, la sottosequenza di corrispondenza più lunga (comprese le separazioni di caratteri) tra le due stringhe sopra è "ABCFCBA" che è anche un palindromo :) Qui un collegamento che descrive il problema LCS: ics.uci.edu/~eppstein/161/960229.html
Jake Drew

5

Per quanto ho capito il problema, possiamo trovare palindromi attorno a un indice centrale e estendere la nostra ricerca in entrambe le direzioni, a destra ea sinistra del centro. Detto questo e sapendo che non c'è palindromo agli angoli dell'input, possiamo impostare i limiti a 1 e lunghezza-1. Prestando attenzione ai limiti minimo e massimo della stringa, verifichiamo se i caratteri nelle posizioni degli indici simmetrici (destro e sinistro) sono gli stessi per ogni posizione centrale fino a raggiungere il nostro centro limite superiore massimo.

Il ciclo esterno è O (n) (max n-2 iterazioni) e il ciclo while interno è O (n) (max intorno a (n / 2) - 1 iterazioni)

Ecco la mia implementazione Java utilizzando l'esempio fornito da altri utenti.

class LongestPalindrome {

    /**
     * @param input is a String input
     * @return The longest palindrome found in the given input.
     */
    public static String getLongestPalindrome(final String input) {
        int rightIndex = 0, leftIndex = 0;
        String currentPalindrome = "", longestPalindrome = "";
        for (int centerIndex = 1; centerIndex < input.length() - 1; centerIndex++) {
            leftIndex = centerIndex - 1;  rightIndex = centerIndex + 1;
            while (leftIndex >= 0 && rightIndex < input.length()) {
                if (input.charAt(leftIndex) != input.charAt(rightIndex)) {
                    break;
                }
                currentPalindrome = input.substring(leftIndex, rightIndex + 1);
                longestPalindrome = currentPalindrome.length() > longestPalindrome.length() ? currentPalindrome : longestPalindrome;
                leftIndex--;  rightIndex++;
            }
        }
        return longestPalindrome;
    }

    public static void main(String ... args) {
        String str = "HYTBCABADEFGHABCDEDCBAGHTFYW12345678987654321ZWETYGDE";
        String longestPali = getLongestPalindrome(str);
        System.out.println("String: " + str);
        System.out.println("Longest Palindrome: " + longestPali);
    }
}

L'output di questo è il seguente:

marcello:datastructures marcello$ javac LongestPalindrome
marcello:datastructures marcello$ java LongestPalindrome
String: HYTBCABADEFGHABCDEDCBAGHTFYW12345678987654321ZWETYGDE
Longest Palindrome: 12345678987654321

6
Se do "HYTBCABADEFGHABCDEDCBAGHTFYW1234567887654321ZWETYGDE" non funziona Ma la risposta dovrebbe essere 1234567887654321
Elbek

1
@j_random_hacker no, questa è una delle soluzioni quadratiche. È coperto qui come expandAroundCenter.
v.oddou

@ v.oddou: Hai perfettamente ragione, e non so come ho concluso O (n ^ 3) dato che ci sono solo 2 loop annidati! Eliminerò quel commento sbagliato ... Ma ho anche notato che questa soluzione ha un problema, che inserirò in un commento separato in modo che l'autore se ne accorga.
j_random_hacker

La mia precedente affermazione sulla complessità temporale O (n ^ 3) era sbagliata (grazie a @ v.oddou per averlo sottolineato!) Ma c'è un altro problema: questo codice non considera palindromi di lunghezza pari. Questo potrebbe essere risolto aggiungendo un secondo anello esterno molto simile (anche O (n ^ 2) in modo da non influire sulla complessità temporale O (n ^ 2)) che espande i palindromi attorno a ciascuna delle n-1 posizioni tra ciascuna coppia di personaggi. +2 se
risolvi

2

con regex e ruby ​​puoi scansionare palindromi brevi come questo:

PROMPT> irb
>> s = "longtextwithranynarpalindrome"
=> "longtextwithranynarpalindrome"
>> s =~ /((\w)(\w)(\w)(\w)(\w)\6\5\4\3\2)/; p $1
nil
=> nil
>> s =~ /((\w)(\w)(\w)(\w)\w\5\4\3\2)/; p $1
nil
=> nil
>> s =~ /((\w)(\w)(\w)(\w)\5\4\3\2)/; p $1
nil
=> nil
>> s =~ /((\w)(\w)(\w)\w\4\3\2)/; p $1
"ranynar"
=> nil

2

Ho scritto il seguente programma Java per curiosità, HTH semplice e autoesplicativo. Grazie.

/**
 *
 * @author sanhn
 */
public class CheckPalindrome {

    private static String max_string = "";

    public static void checkSubString(String s){
        System.out.println("Got string is "+s);
        for(int i=1;i<=s.length();i++){
            StringBuilder s1 = new StringBuilder(s.substring(0,i));
            StringBuilder s2 = new StringBuilder(s.substring(0,i));
            s2.reverse();
            if(s1.toString().equals(s2.toString())){
                if(max_string.length()<=s1.length()){
                    max_string = s1.toString();
                    System.out.println("tmp max is "+max_string);
                }

            }
        }
    }

    public static void main(String[] args){
        String s="HYTBCABADEFGHABCDEDCBAGHTFYW1234567887654321ZWETYGDE";

        for(int i=0; i<s.length(); i++)
            checkSubString(s.substring(i, s.length()));

        System.out.println("Max string is "+max_string);
    }
}

1

Di recente mi è stata posta questa domanda. Ecco la soluzione che [alla fine] ho trovato. L'ho fatto in JavaScript perché è piuttosto semplice in quella lingua.

Il concetto di base è che cammini sulla corda cercando il più piccolo palindromo multi-carattere possibile (uno a due o tre caratteri). Una volta che lo hai, espandi i bordi su entrambi i lati finché non smette di essere un palindromo. Se quella lunghezza è più lunga di quella attuale, memorizzala e spostati.

// This does the expanding bit.
function getsize(s, start, end) {
    var count = 0, i, j;
    for (i = start, j = end; i >= 0 && j < s.length; i--, j++) {
        if (s[i] !== s[j]) {
            return count;
        }
        count = j - i + 1; // keeps track of how big the palindrome is
    }
    return count;
}

function getBiggestPalindrome(s) {
    // test for simple cases
    if (s === null || s === '') { return 0; }
    if (s.length === 1) { return 1; }
    var longest = 1;
    for (var i = 0; i < s.length - 1; i++) {
        var c = s[i]; // the current letter
        var l; // length of the palindrome
        if (s[i] === s[i+1]) { // this is a 2 letter palindrome
            l = getsize(s, i, i+1);
        }
        if (i+2 < s.length && s[i] === s[i+2]) { // 3 letter palindrome
            l = getsize(s, i+1, i+1);
        }
        if (l > longest) { longest = l; }
    }
    return longest;
}

Questo potrebbe sicuramente essere ripulito e ottimizzato un po 'di più, ma dovrebbe avere prestazioni piuttosto buone in tutti tranne lo scenario peggiore (una stringa della stessa lettera).


1
Inizialmente pensavo che l'algoritmo # 1 dell'OP fosse O (n ^ 2) tempo, ma in realtà è ossessivamente O (n ^ 3), quindi anche se il tuo algoritmo non arriva fino al limite O (n) raggiungibile, è ancora un miglioramento.
j_random_hacker

1
lo chiami "semplice" ma è pieno di i j l s ifmanutenzione e stato. multi punti di ritorno, casi limite ...
v.oddou

1

Ciao Ecco il mio codice per trovare il palindromo più lungo nella stringa. Si prega di fare riferimento al seguente collegamento per comprendere l'algoritmo http://stevekrenzel.com/articles/longest-palnidrome

I dati dei test utilizzati sono HYTBCABADEFGHABCDEDCBAGHTFYW12345678987654321ZWETYGDE

 //Function GetPalindromeString

public static string GetPalindromeString(string theInputString)
 { 

        int j = 0;
        int k = 0;
        string aPalindrome = string.Empty;
        string aLongestPalindrome = string.Empty ;          
        for (int i = 1; i < theInputString.Length; i++)
        {
            k = i + 1;
            j = i - 1;
            while (j >= 0 && k < theInputString.Length)
            {
                if (theInputString[j] != theInputString[k])
                {
                    break;
                }
                else
                {
                    j--;
                    k++;
                }
                aPalindrome = theInputString.Substring(j + 1, k - j - 1);
                if (aPalindrome.Length > aLongestPalindrome.Length)
                {
                    aLongestPalindrome = aPalindrome;
                }
            }
        }
        return aLongestPalindrome;     
  }

Non sono sicuro che funzioni con palindromi di lunghezza pari ... puoi confermare?
inizio

Questo funziona anche per i palindromi, puoi eseguire questo programma e fammi sapere se non funziona per te.Per la comprensione dell'algoritmo si prega di
Mohit Bhandari

@ st0le: questa logica non funzionerà nemmeno per i palindromi, ma potrebbe essere modificata anche per i palindromi. Mi dispiace per il precedente commento. Ho ottenuto la logica e la aggiornerò in pochi giorni se e quando avrò un tempo.
Mohit Bhandari

mai letto il tuo commento precedente fino ad oggi ... l'ultima volta non ti sei rivolto a me .... prenditi il ​​tuo tempo, era solo un'osservazione.
st0le

2
Inizialmente pensavo che l'algoritmo # 1 dell'OP fosse O (n ^ 2) tempo, ma in realtà è ossessivamente O (n ^ 3), quindi anche se il tuo algoritmo non arriva fino al limite O (n) raggiungibile, è ancora un miglioramento.
j_random_hacker

1

Vedi l' articolo di Wikipedia su questo argomento. Esempio di implementazione Java dell'algoritmo di Manacher per la soluzione O (n) lineare dall'articolo seguente:

import java.util.Arrays; public class ManachersAlgorithm {public static String findLongestPalindrome (String s) {if (s == null || s.length () == 0) return "";

char[] s2 = addBoundaries(s.toCharArray());
int[] p = new int[s2.length]; 
int c = 0, r = 0; // Here the first element in s2 has been processed.
int m = 0, n = 0; // The walking indices to compare if two elements are the same
for (int i = 1; i<s2.length; i++) {
  if (i>r) {
    p[i] = 0; m = i-1; n = i+1;
  } else {
    int i2 = c*2-i;
    if (p[i2]<(r-i)) {
      p[i] = p[i2];
      m = -1; // This signals bypassing the while loop below. 
    } else {
      p[i] = r-i;
      n = r+1; m = i*2-n;
    }
  }
  while (m>=0 && n<s2.length && s2[m]==s2[n]) {
    p[i]++; m--; n++;
  }
  if ((i+p[i])>r) {
    c = i; r = i+p[i];
  }
}
int len = 0; c = 0;
for (int i = 1; i<s2.length; i++) {
  if (len<p[i]) {
    len = p[i]; c = i;
  }
}
char[] ss = Arrays.copyOfRange(s2, c-len, c+len+1);
return String.valueOf(removeBoundaries(ss));   }
private static char[] addBoundaries(char[] cs) {
if (cs==null || cs.length==0)
  return "||".toCharArray();

char[] cs2 = new char[cs.length*2+1];
for (int i = 0; i<(cs2.length-1); i = i+2) {
  cs2[i] = '|';
  cs2[i+1] = cs[i/2];
}
cs2[cs2.length-1] = '|';
return cs2;   }
private static char[] removeBoundaries(char[] cs) {
if (cs==null || cs.length<3)
  return "".toCharArray();

char[] cs2 = new char[(cs.length-1)/2];
for (int i = 0; i<cs2.length; i++) {
  cs2[i] = cs[i*2+1];
}
return cs2;   }     }

1

Una Regexpsoluzione efficiente che evita la forza bruta

Inizia con l'intera lunghezza della stringa e funziona fino a 2 caratteri, esiste non appena viene effettuata una corrispondenza

Per "abaccddccefe"la regexp prova 7 corrispondenze prima di tornare ccddcc.

(.) (.) (.) (.) (.) (.) (\ 6) (\ 5) (\ 4) (\ 3) (\ 2) (\ 1)
(.) (.) (. ) (.) (.) (.) (\ 5) (\ 4) (\ 3) (\ 2) (\ 1)
(.) (.) (.) (.) (.) (\ 5) ( \ 4) (\ 3) (\ 2) (\ 1)
(.) (.) (.) (.) (.) (\ 4) (\ 3) (\ 2) (\ 1)
(.) ( .) (.) (.) (\ 4) (\ 3) (\ 2) (\ 1)
(.) (.) (.) (.) (\ 3) (\ 2) (\ 1)
(. ) (.) (.) (\ 3) (\ 2) (\ 1)

Dim strTest
wscript.echo Palindrome("abaccddccefe")

Sub Test()
Dim strTest
MsgBox Palindrome("abaccddccefe")
End Sub

funzione

Function Palindrome(strIn)

Set objRegex = CreateObject("vbscript.regexp")

For lngCnt1 = Len(strIn) To 2 Step -1
    lngCnt = lngCnt1 \ 2
    strPal = vbNullString

    For lngCnt2 = lngCnt To 1 Step -1
        strPal = strPal & "(\" & lngCnt2 & ")"
    Next

    If lngCnt1 Mod 2 = 1 Then strPal = "(.)" & strPal

    With objRegex
        .Pattern = Replace(Space(lngCnt), Chr(32), "(.)") & strPal
        If .Test(strIn) Then
            Palindrome = .Execute(strIn)(0)
            Exit For
        End If
    End With
Next

End Function

@DickKusleika puoi aggiornare il mio commento su dailydoseofexcel.com/archives/2016/01/14/… con il codice rivisto sopra. Thx
brettdj

1
public static void main(String[] args) {
         System.out.println(longestPalindromeString("9912333321456")); 
}

    static public String intermediatePalindrome(String s, int left, int right) {
        if (left > right) return null;
        while (left >= 0 && right < s.length()
                && s.charAt(left) == s.charAt(right)) {
            left--;
            right++;
        }
        return s.substring(left + 1, right);
    }


    public static String longestPalindromeString(String s) {
        if (s == null) return null;
        String longest = s.substring(0, 1);
        for (int i = 0; i < s.length() - 1; i++) {
            //odd cases like 121
            String palindrome = intermediatePalindrome(s, i, i);
            if (palindrome.length() > longest.length()) {
                longest = palindrome;
            }
            //even cases like 1221
            palindrome = intermediatePalindrome(s, i, i + 1);
            if (palindrome.length() > longest.length()) {
                longest = palindrome;
            }
        }
        return longest;
    }

0

Prova la stringa - "HYTBCABADEFGHABCDEDCBAGHTFYW123456789987654321ZWETYGDE"; Dovrebbe funzionare per amici pari e dispari. Molte grazie a Mohit!

using namespace std;

string largestPal(string input_str)
{
  string isPal = "";
  string largest = "";
  int j, k;
  for(int i = 0; i < input_str.length() - 1; ++i)
    {
      k = i + 1;
      j = i - 1;

      // starting a new interation                                                      
      // check to see if even pal                                                       
      if(j >= 0 && k < input_str.length()) {
        if(input_str[i] == input_str[j])
          j--;
        else if(input_str[i] == input_str[j]) {
          k++;
        }
      }
      while(j >= 0 && k < input_str.length())
        {
          if(input_str[j] != input_str[k])
            break;
          else
            {
              j--;
              k++;
            }
          isPal = input_str.substr(j + 1, k - j - 1);
            if(isPal.length() > largest.length()) {
              largest = isPal;
            }
        }
    }
  return largest;
}

2
Questo fa quasi le cose in O (n ^ 2) tempo. Perché costruire isPal- un'operazione O (n) - solo per misurarne la lunghezza !? Inoltre ha un tentativo buggy di gestire anche i palindromi. Sulla bugginess palindromo: else if(input_str[i] == input_str[j])non può mai avere successo perché lo stesso test deve essere fallito nell'affermazione precedente if; ed è comunque bacato perché non puoi dire solo guardando 2 caratteri distanziati di 2 posizioni l'uno dall'altro se stai guardando un palindromo pari o uno strano (considera AAAe AAAA).
j_random_hacker

0

Il codice seguente calcola Palidrom per stringhe di lunghezza pari e dispari.

Non è la soluzione migliore ma funziona per entrambi i casi

HYTBCABADEFGHABCDEDCBAGHTFYW12345678987654321ZWETYGDE HYTBCABADEFGHABCDEDCBAGHTFYW1234567887654321ZWETYGDE

private static String getLongestPalindrome(String string) {
    String odd = getLongestPalindromeOdd(string);
    String even = getLongestPalindromeEven(string);
    return (odd.length() > even.length() ? odd : even);
}

public static String getLongestPalindromeOdd(final String input) {
    int rightIndex = 0, leftIndex = 0;
    String currentPalindrome = "", longestPalindrome = "";
    for (int centerIndex = 1; centerIndex < input.length() - 1; centerIndex++) {
        leftIndex = centerIndex;
        rightIndex = centerIndex + 1;
        while (leftIndex >= 0 && rightIndex < input.length()) {
            if (input.charAt(leftIndex) != input.charAt(rightIndex)) {
                break;
            }
            currentPalindrome = input.substring(leftIndex, rightIndex + 1);
            longestPalindrome = currentPalindrome.length() > longestPalindrome
                    .length() ? currentPalindrome : longestPalindrome;
            leftIndex--;
            rightIndex++;
        }
    }
    return longestPalindrome;
}

public static String getLongestPalindromeEven(final String input) {
    int rightIndex = 0, leftIndex = 0;
    String currentPalindrome = "", longestPalindrome = "";
    for (int centerIndex = 1; centerIndex < input.length() - 1; centerIndex++) {
        leftIndex = centerIndex - 1;
        rightIndex = centerIndex + 1;
        while (leftIndex >= 0 && rightIndex < input.length()) {
            if (input.charAt(leftIndex) != input.charAt(rightIndex)) {
                break;
            }
            currentPalindrome = input.substring(leftIndex, rightIndex + 1);
            longestPalindrome = currentPalindrome.length() > longestPalindrome
                    .length() ? currentPalindrome : longestPalindrome;
            leftIndex--;
            rightIndex++;
        }
    }
    return longestPalindrome;
}

0
  1. Modifica la stringa per separare ogni carattere utilizzando un separatore [questo per incorporare palindromi pari e dispari]
  2. Trova palindromi attorno a ciascun personaggio trattandolo come un centro

Possiamo trovare tutti i palindromi di tutta la lunghezza usando questo.

Campione :

parola = abcdcbc

Stringa modificata = a # b # c # d # c # b # c

palinCount = 1010105010301

lunghezza del palindromo più lungo = 5;

palindromo più lungo = bcdcb

public class MyLongestPalindrome {

static String word;
static int wordlength;
static int highestcount = 0;
static int newlength;
static char[] modifiedString; // stores modified string
static int[] palinCount; // stores palindrome length at each position
static char pound = '#';

public static void main(String[] args) throws IOException {
    // TODO Auto-generated method stub
    System.out.println("Enter String : ");
    InputStreamReader isr = new InputStreamReader(System.in);
    BufferedReader bfr = new BufferedReader(isr);
    word = bfr.readLine();
    wordlength = word.length();
    newlength = (wordlength * 2) - 1;
    convert();
    findpalindrome();
    display();
}

// Inserting # in string
public static void convert() {

    modifiedString = new char[newlength];
    int j = 0;
    int i;
    for (i = 0; i < wordlength - 1; i++) {
        modifiedString[j++] = word.charAt(i);
        modifiedString[j++] = pound;
    }
    modifiedString[j] = word.charAt(i);
}

// display all palindromes of highest length
public static void display() {
    String palindrome;
    String s = new String(modifiedString);
    System.out.println("Length of longest palindrome = " + highestcount);
    for (int i = 0; i < newlength; i++) {
        if (palinCount[i] == highestcount) {
            palindrome = s.substring(i - (highestcount - 1), i
                    + (highestcount));
            i = i + (highestcount - 1);
            palindrome = palindrome.replace("#", "");
            System.out.println(palindrome);
        }
    }
}

// populate palinCount with length of palindrome string at each position
public static void findpalindrome() {
    int left, right, count;
    palinCount = new int[newlength];
    palinCount[0] = 1;
    palinCount[newlength - 1] = 1;
    for (int i = 1; i < newlength - 1; i++) {
        count = 0;
        left = i - 1;
        right = i + 1;
        ;
        if (modifiedString[i] != pound)
            count++;
        while (left >= 0 && right < newlength) {
            if (modifiedString[left] == modifiedString[right]) {
                if (modifiedString[left] != pound)
                    count = count + 2;
                left--;
                right++;
            } else
                break;
        }

        palinCount[i] = count;
        highestcount = count > highestcount ? count : highestcount;

    }

}

}


0

Questo restituirà la stringa palindromo più lunga dalla stringa data

-(BOOL)isPalindromString:(NSString *)strInput
{
    if(strInput.length<=1){
        return NO;
    }
    int halfLenth = (int)strInput.length/2;

    BOOL isPalindrom = YES;
    for(NSInteger i=0; i<halfLenth; i++){

        char a = [strInput characterAtIndex:i];
        char b = [strInput characterAtIndex:(strInput.length-1)-i];

        if(a != b){
            isPalindrom = NO;
            break;
        }
    }
    NSLog(@"-%@- IS Plaindrom %@",strInput,(isPalindrom ? @"YES" : @"NO"));
    return isPalindrom;
}


-(NSString *)longestPalindrom:(NSString *)strInput
{
    if(strInput.length<=1){
        return @"";
    }

    NSString *strMaxPalindrom = @"";

    for(int i = 0; i<strInput.length ; i++){

        for(int j = i; j<strInput.length ; j++){

            NSString *strSub = [strInput substringWithRange:NSMakeRange(i, strInput.length-j)];

            if([self isPalindromString:strSub]){

                if(strSub.length>strMaxPalindrom.length){

                    strMaxPalindrom = strSub;
                }
            }
        }
    }
    NSLog(@"-Max - %@",strMaxPalindrom);
    return strMaxPalindrom;
}

-(void)test
{
    [self longestPalindrom:@"abcccbadeed"];
}

== USCITA ===

Ingresso: abcccde Uscita: ccc

Input: abcccbd Output: bcccb

Input: abedccde Output: edccde

Input: abcccdeed Output: rogito

Input: abcccbadeed Output: abcccba


0

Ecco un'implementazione in javascript:

var longestPalindromeLength = 0;
var longestPalindrome = ''

function isThisAPalidrome(word){
  var reverse = word.split('').reverse().join('')
  return word == reverse
}

function findTheLongest(word){ // takes a word of your choice
  for(var i = 0; i < word.length; i++){ // iterates over each character
    var wordMinusOneFromBeginning = word.substr(i, word.length) // for each letter, create the word minus the first char
    for(var j = wordMinusOneFromBeginning.length; j > 0; j--){ // for the length of the word minus the first char
      var wordMinusOneFromEnding = wordMinusOneFromBeginning.substr(0, j) // create a word minus the end character
      if(wordMinusOneFromEnding <= 0) // make sure the value is more that 0,
      continue // if more than zero, proced to next if statement
      if(isThisAPalidrome(wordMinusOneFromEnding)){ // check if the word minus the first character, minus the last character = a plaindorme
        if(wordMinusOneFromEnding.length > longestPalindromeLength){ // if it is
          longestPalindromeLength = wordMinusOneFromEnding.length; // save its length
          longestPalindrome = wordMinusOneFromEnding // and save the string itself
        } // exit the statement that updates the longest palidrome
      } // exit the stament that checks for a palidrome
    } // exit the loop that goes backwards and takes a letter off the ending
  } // exit the loop that goes forward and takes off the beginning letter
  return console.log('heres the longest string: ' + longestPalindrome
  + ' its ' + longestPalindromeLength + ' charachters in length'); // return the longest palidrome! :)
}
findTheLongest('bananas');


Non so perché questo è stato votato contro: funziona come un fascino. Mi ha fatto passare un colloquio con le tecnologie CA perfettamente.
alex bennett

0

Per una soluzione lineare, puoi utilizzare l'algoritmo di Manacher. C'è un altro algoritmo chiamato Algorithm di Gusfield, e sotto c'è il codice in java:

public class Solution {  
    char[] temp;   
    public int match(int a, int b,int len){   
        int i = 0;   
        while (a-i>=0 && b+i<len && temp[a-i] == temp[b+i]) i++;   
        return i;   
    }  

    public String longestPalindrome(String s) {  

        //This makes use of the assumption that the string has not more than 1000 characters.  
        temp=new char[1001*2];  
        int[] z=new int[1001 * 2];  
        int L=0, R=0;  
        int len=s.length();  

        for(int i=0;i<len*2+1;i++){  
            temp[i]='.';  
        }  

        for(int i=0;i<len;++i){  
            temp[i*2+1] = s.charAt(i);  
        }  

        z[0]=1;  
        len=len*2+1;  

        for(int i=0;i<len;i++){  
            int ii = L - (i - L);     
            int n = R + 1 - i;  
            if (i > R)  
            {  
                z[i] = match(i, i,len);  
                L = i;  
                R = i + z[i] - 1;  
            }  
            else if (z[ii] == n)  
            {  
                z[i] = n + match(i-n, i+n,len);  
                L = i;  
                R = i + z[i] - 1;  
            }  
            else  
            {  
                z[i] = (z[ii]<= n)? z[ii]:n;  
            }   
        }  

        int n = 0, p = 0;  
        for (int i=0; i<len; ++i)  
            if (z[i] > n)  
                n = z[p = i];  

        StringBuilder result=new StringBuilder();  
        for (int i=p-z[p]+1; i<=p+z[p]-1; ++i)  
            if(temp[i]!='.')  
                result.append(String.valueOf(temp[i]));  

        return result.toString();  
    }  
}  

Puoi trovare di più su altre soluzioni come la migliore soluzione O (n ^ 2) o l'algoritmo di Manacher dal mio blog .


0

Qui ho scritto una logica provalo :)

public class palindromeClass{

public  static String longestPalindromeString(String in) {
        char[] input = in.toCharArray();
        int longestPalindromeStart = 0;
        int longestPalindromeEnd = 0;

        for (int mid = 0; mid < input.length; mid++) {
            // for odd palindrome case like 14341, 3 will be the mid
            int left = mid-1;
            int right = mid+1;
            // we need to move in the left and right side by 1 place till they reach the end
            while (left >= 0 && right < input.length) {
                // below check to find out if its a palindrome
                if (input[left] == input[right]) {
                    // update global indexes only if this is the longest one till now
                    if (right - left > longestPalindromeEnd
                            - longestPalindromeStart) {
                        longestPalindromeStart = left;
                        longestPalindromeEnd = right;
                    }
                }
                else
                    break;
                left--;
                right++;
            }
            // for even palindrome, we need to have similar logic with mid size 2
            // for that we will start right from one extra place
            left = mid;
            right = mid + 1;// for example 12333321 when we choose 33 as mid
            while (left >= 0 && right < input.length)
            {
                if (input[left] == input[right]) {
                    if (right - left > longestPalindromeEnd
                            - longestPalindromeStart) {
                        longestPalindromeStart = left;
                        longestPalindromeEnd = right;
                    }
                }
                else
                    break;
                left--;
                right++;
            }


        }
        // we have the start and end indexes for longest palindrome now
        return in.substring(longestPalindromeStart, longestPalindromeEnd + 1);
    }
public static void main(String args[]){
System.out.println(longestPalindromeString("HYTBCABADEFGHABCDEDCBAGHTFYW12345678987654321ZWETYGDE"));
}

}

questo dà tutto il palindromo nella stringa non solo il più lungo
Vivek Mishra

0

Questa soluzione è di complessità O (n ^ 2). O (1) è la complessità dello spazio.

public class longestPalindromeInAString {

        public static void main(String[] args) {
            String a =  "xyMADAMpRACECARwl"; 
            String res = "";
            //String longest = a.substring(0,1);
            //System.out.println("longest => " +longest);
            for (int i = 0; i < a.length(); i++) {
                String temp = helper(a,i,i);//even palindrome
                if(temp.length() > res.length()) {res = temp ;}
                temp = helper(a,i,i+1);// odd length palindrome
                if(temp.length() > res.length()) { res = temp ;}

            }//for
            System.out.println(res);
            System.out.println("length of " + res + " is " + res.length());

        }

        private static String helper(String a, int left, int right) {
            while(left>= 0 && right <= a.length() -1  &&  a.charAt(left) == a.charAt(right)) {
                left-- ;right++ ;
            }
            String curr = a.substring(left + 1 , right);
            System.out.println("curr =>" +curr);
            return curr ;
        }

    }

0

#longest palindrome
s='HYTBCABADEFGHABCDEDCBAGHTFYW123456789987654321ZWETYGDE'
out1=[]
def substring(x):
    for i in range(len(x)):
        a=x[i:]
        b=x[:-i]
        out1.append(a)
        out1.append(b)
        
    return out1

for i in range(len(s)):
    substring(s[i:])    
final=set([item for item in out1 if len(item)>2])
final
palind={item:len(item) for item in final if item==item[::-1]}
print(palind)
sorted(palind.items(),reverse=True, key=lambda x: x[1])[0]

{"DED": 3, "123456789987654321": 18, "67899876": 8, "ABCDEDCBA": 9, "456789987654": 12, "34567899876543": 14, "BCDEDCB": 7, "ABA": 3, " 5678998765 ': 10,' 2345678998765432 ': 16,' CDEDC ': 5,' 789987 ': 6,' 8998 ': 4} (' 123456789987654321 ', 18)


-1

Ecco il mio algoritmo:

1) imposta il centro corrente come prima lettera

2) espandi simultaneamente a sinistra ea destra fino a trovare il palindromo massimo attorno al centro attuale

3) se il palindromo che trovi è più grande del palindromo precedente, aggiornalo

4) imposta il centro corrente come lettera successiva

5) ripetere i passaggi da 2) a 4) per tutte le lettere nella stringa

Questo viene eseguito in O (n).

Spero che sia d'aiuto.


5
Considera la stringa "aaaaaa". Questo viene eseguito in O (n ^ 2) utilizzando il tuo algoritmo.
paislee

1
Inizialmente pensavo che l'algoritmo # 1 dell'OP fosse O (n ^ 2) tempo, ma in realtà è ossessivamente O (n ^ 3), quindi anche se il tuo algoritmo non arriva fino al limite O (n) raggiungibile, è ancora un miglioramento.
j_random_hacker

Questa è la classica soluzione di espansione N2. MA, in realtà è vicino alla soluzione di Manacher come spiegato in questo video: youtube.com/watch?v=V-sEwsca1ak la differenza è il punto 4. Manacher riutilizza le informazioni per evitare di scansionare nuovamente le lettere già scansionate.
v.oddou

-2

Riferimento: Wikipedia.com

Il miglior algoritmo che abbia mai trovato, con complessità O (N)

 import java.util.Arrays;

 public class ManachersAlgorithm {

  public static String findLongestPalindrome(String s) {
    if (s==null || s.length()==0)
      return "";

    char[] s2 = addBoundaries(s.toCharArray());
    int[] p = new int[s2.length]; 
    int c = 0, r = 0; // Here the first element in s2 has been processed.
    int m = 0, n = 0; // The walking indices to compare if two elements are the same
    for (int i = 1; i<s2.length; i++) {
      if (i>r) {
        p[i] = 0; m = i-1; n = i+1;
      } else {
        int i2 = c*2-i;
        if (p[i2]<(r-i)) {
          p[i] = p[i2];
          m = -1; // This signals bypassing the while loop below. 
        } else {
          p[i] = r-i;
          n = r+1; m = i*2-n;
        }
      }
      while (m>=0 && n<s2.length && s2[m]==s2[n]) {
        p[i]++; m--; n++;
      }
      if ((i+p[i])>r) {
        c = i; r = i+p[i];
      }
    }
    int len = 0; c = 0;
    for (int i = 1; i<s2.length; i++) {
      if (len<p[i]) {
        len = p[i]; c = i;
      }
    }
    char[] ss = Arrays.copyOfRange(s2, c-len, c+len+1);
    return String.valueOf(removeBoundaries(ss));
  }

  private static char[] addBoundaries(char[] cs) {
    if (cs==null || cs.length==0)
      return "||".toCharArray();

    char[] cs2 = new char[cs.length*2+1];
    for (int i = 0; i<(cs2.length-1); i = i+2) {
      cs2[i] = '|';
      cs2[i+1] = cs[i/2];
    }
    cs2[cs2.length-1] = '|';
    return cs2;
  }

  private static char[] removeBoundaries(char[] cs) {
    if (cs==null || cs.length<3)
      return "".toCharArray();

    char[] cs2 = new char[(cs.length-1)/2];
    for (int i = 0; i<cs2.length; i++) {
      cs2[i] = cs[i*2+1];
    }
    return cs2;
  }    
}

-5

la mia soluzione è:

static string GetPolyndrom(string str)
{
    string Longest = "";

    for (int i = 0; i < str.Length; i++)
    {
        if ((str.Length - 1 - i) < Longest.Length)
        {
            break;
        }
        for (int j = str.Length - 1; j > i; j--)
        {
            string str2 = str.Substring(i, j - i + 1);
            if (str2.Length > Longest.Length)
            {
                if (str2 == str2.Reverse())
                {
                    Longest = str2;
                }
            }
            else
            {
                break;
            }
        }

    }
    return Longest;
}

1
Ciò richiede tempo cubico nella lunghezza della stringa, a causa delle operazioni Substring()e string-equality ( ==). È fondamentalmente identico all'algoritmo # 1 dell'OP.
j_random_hacker
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.