指定された文字列で最長の回文を返す関数を記述します


101

たとえば、文字列「abaccddccefe」の「ccddcc」

私は解決策を考えましたが、O(n ^ 2)時間で実行されます

アルゴ1:

手順:そのブルートフォースメソッド


  1. i = 1からiまでのarray.length未満の2 forループがある
    j = i + 1からarray.length未満のjまでの -1
  2. このようにして、配列から可能なすべての組み合わせの部分文字列を取得できます
  3. 文字列が回文かどうかをチェックする回文関数を持っている
  4. したがって、すべての部分文字列(i、j)についてこの関数を呼び出します。回文式の場合は、文字列変数に格納します。
  5. 次の回文の部分文字列が見つかり、それが現在のものよりも大きい場合は、現在のものと置き換えます。
  6. 最後に、文字列変数に答えがあります

問題:1.このアルゴはO(n ^ 2)時間で実行されます。

アルゴ2:

  1. 文字列を逆にして別の配列に格納します
  2. 次に、両方の配列の間で最大の一致する部分文字列を見つけます
  3. しかし、これもO(n ^ 2)時間で実行されます

より良い時間に実行されるアルゴを想像できますか?可能であればO(n)時間


42
最初のものはO(n^2)、部分文字列*を取得して、O(n)それらがパリンドロームかどうかをチェックすることO(n^3)です。
Skylar Saveland

回文を使用していて、文字列を2つの半分として保存していることを知っていて、Javaを使用している場合は、関数のO(1)チェックがあるとしたらどうでしょうか。
viki.omega9 2013年

10
secongアルゴは正しいですか?文字列「abcdecba」についてはどうでしょう。最大の一致する部分文字列は( "abcdecba"と "abcedcba")です: "abc"または "cba"。ただし、どちらも回文ではありません。
Yarneo 2013

@Learner、好奇心旺盛ですが、あなたの手順の中で、forループでどの配列を参照しているのですか?配列で文字列を参照していますか?string.length?
Zolt、2013年

1
O(n ^ 2)で回答を検索する
ユーザー向け

回答:


76

あなたは使用して最長の回文を見つけることができますManacherのアルゴリズムO(n)時間を!その実装はここここにあります

入力のString s = "HYTBCABADEFGHABCDEDCBAGHTFYW1234567887654321ZWETYGDE"場合、正しい出力であるを見つけます1234567887654321


3
これがどのように線形かわかりません。にwhile埋め込まれてforいるのが、外側のループに似ているように見える境界で表示されます。
v.oddou 2016


9

Algo 2はすべての文字列に対して機能するとは限りません。このような文字列「ABCDEFCBA」の例を次に示します。

文字列のサブストリングとして「ABC」と「CBA」が含まれているわけではありません。元の文字列を逆にすると、「ABCFEDCBA」になります。最長の一致する部分文字列は「ABC」であり、回文ではありません。

さらに、この最長一致部分文字列が、実行時間がO(n ^ 3)の回文かどうかをさらに確認する必要がある場合があります。


2
Algo 2は、サブシーケンス文字も文字列内で分離できる一般的なアルゴリズムの問​​題である「最長一致サブシーケンスパリンドローム」で機能することに注意することが重要です。たとえば、上記の2つの文字列の間で最長の一致するサブシーケンス(文字の分離を含む)は「ABCFCBA」で、これも回文です:)ここにLCS問題を説明するリンクがあります: ics.uci.edu/~eppstein/161/960229.html
Jake Drew、

5

私が問題を理解している限り、センターインデックスの周りに回文を見つけ、検索をセンターの左右に広げることができます。それがあり、入力のコーナーにパリンドロームがないことがわかっている場合、境界を1に、長さ1に設定できます。文字列の最小と最大の境界に注意を払いながら、対称インデックス(右と左)の位置にある文字が、最大の上限の中心に到達するまで各中央位置で同じかどうかを確認します。

外側のループはO(n)(最大n-2反復)、内側のwhileループはO(n)(最大((n / 2)-1反復))

これが、他のユーザーから提供された例を使用した私のJava実装です。

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);
    }
}

この出力は次のとおりです。

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

6
私は「HYTBCABADEFGHABCDEDCBAGHTFYW1234567887654321ZWETYGDE」を与えた場合には動作しません。しかしanwerは1234567887654321でなければなりません
Elbek

1
@j_random_hackerいいえ、それは二次解の1つです。ここではとして扱われますexpandAroundCenter
v.oddou 2016

@ v.oddou:まったくその通りで、ネストされたループが2つしかないので、O(n ^ 3)をどのように結論付けたのかわかりません!この誤ったコメントを削除します...しかし、この解決策に問題があることにも気付きました。別のコメントを付けて、作者が気付くようにします。
j_random_hacker

以前のO(n ^ 3)時間の複雑さの主張は間違っていました(これを指摘してくれた@ v.oddouに感謝します)。しかし、別の問題があります。このコードは偶数長の回文を考慮していません。 これは、n-1の各位置の周囲のパリンドローム膨張第二、非常に類似したアウターループ(さらにはO(n ^ 2)それはO(N ^ 2)の時間計算量に影響を与えないように)添加することによって固定することができるとの間のそれぞれを文字のペア。+2修正した場合:)
j_random_hacker 16/07/14

2

正規表現とルビーを使用すると、次のような短い回文をスキャンできます。

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

私は好奇心、シンプルで自明のHTHから次のJavaプログラムを作成しました。ありがとう。

/**
 *
 * @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

最近この質問をされました。これが私が[最終的に]思いついた解決策です。JavaScriptでそれを行ったのは、その言語ではかなり単純だからです。

基本的な概念は、可能な限り最小の複数文字パリンドローム(2文字または3文字のいずれか)を探して文字列を歩くことです。それができたら、回文構造でなくなるまで両側の境界を広げます。その長さが現在の最長のものよりも長い場合は、それを保管して移動してください。

// 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;
}

これは間違いなくクリーンアップされ、もう少し最適化される可能性がありますが、最悪の場合のシナリオ(同じ文字の文字列)を除くすべてにおいて、かなり良いパフォーマンスを発揮するはずです。


1
私は当初、OPのアルゴリズム#1はO(n ^ 2)時間であると思っていましたが、実際にはO(n ^ 3)なので、アルゴリズムが達成可能なO(n)限界まで到達していなくても、それはまだ改善です。
j_random_hacker 2012

1
あなたはこれを「まっすぐ」と呼んでいますが、それは完全なi j l s if状態維持です。マルチリターンポイント、エッジケース...
v.oddou 2016

1

こんにちはここに私の文字列で最長の回文を見つけるコードがあります。アルゴリズムを理解するには、次のリンクを参照してくださいhttp://stevekrenzel.com/articles/longest-palnidrome

使用されるテストデータは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;     
  }

これが均一な長さの回文で機能するかどうかはわかりません...確認できますか?
st0le

これはパリンドロームでも機能します。このプログラムを実行して、機能しない場合はお知らせください。アルゴリズムを理解するには、次のリンクを
Mohit Bhandari

@ st0le:このロジックはパリンドロームでも機能しませんが、パリンドロームでも調整できます。先ほどのコメントで後悔してしまいます。ロジックは入手できましたが、時間があれば数日で更新します。
Mohit Bhandari、2011年

今日まであなたの以前のコメントを決して読まないでください...あなたは私に最後にアドレスしませんでした...あなたの時間をかけてください、それは単なる観察でした。
st0le 2009年

2
私は当初、OPのアルゴリズム#1はO(n ^ 2)時間であると思っていましたが、実際にはO(n ^ 3)なので、アルゴリズムが達成可能なO(n)限界まで到達していなくても、それはまだ改善です。
j_random_hacker 2012

1

このトピックに関するウィキペディアの記事を参照してください。以下の記事からの線形O(n)ソリューションのためサンプルManacherのアルゴリズム Java実装:

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

Regexpブルートフォースを回避する効率的なソリューション

文字列全体から始まり、2文字まで下方向に機能し、一致するとすぐに存在します

"abaccddccefe"正規表現テストの場合、戻る前に7件一致し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

関数

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あなたは私のコメントをdailydoseofexcel.com/archives/2016/01/14/…で私のコメントを更新することができます。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

「HYTBCABADEFGHABCDEDCBAGHTFYW123456789987654321ZWETYGDE」という文字列を試してください。それは偶数と奇数の仲間のために働くべきです。Mohitに感謝します!

名前空間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
これはほとんど O(n ^ 2)時間で物事を行います。なぜ構築isPal-O(n)操作-その長さを測定するためだけ!?また、パリンドロームでさえもバグのある試みをしています。偶数回のバギーについて:else if(input_str[i] == input_str[j])同じテストが前のifステートメントで失敗したに違いないため、成功することはありません。とにかくバギーだ。偶数回文と奇数回のどちらを見ているのか(検討中AAAAAAA)、2桁離れた2文字を見ただけでは分からない。
j_random_hacker 2012

0

次のコードは、偶数長と奇数長の文字列のパリドロムを計算します。

最善の解決策ではありませんが、両方のケースで機能します

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. 区切り文字を使用して各文字を区切るように文字列を変更します[これは奇数と偶数の回文を組み込むことです]
  2. それを中心として、各キャラクターの周りの回文を見つけます

これを使用して、あらゆる長さのすべての回文を見つけることができます。

サンプル :

単語= abcdcbc

modifiedString = a#b#c#d#c#b#c

palinCount = 1010105010301

最長回文の長さ= 5;

最長回文= bcdcb

パブリッククラス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

これは、指定された文字列から最長の回文文字列を返します

-(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"];
}

==出力===

入力:abcccde出力:ccc

入力:abcccbd出力:bcccb

入力:abedccde出力:edccde

入力:abcccdeed出力:証書

入力:abcccbadeed出力:abcccba


0

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');


これが反対投票された理由がわからない-魅力のように動作します。CAテクノロジへのインタビューで問題はありませんでした。
アレックスベネット

0

線形解の場合、Manacherのアルゴリズムを使用できます。Gusfieldのアルゴリズムを呼び出す別のアルゴリズムがあり、以下は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();  
    }  
}  

自分のブログから最高のO(n ^ 2)ソリューションやManacherのアルゴリズムなど、他のソリューションの詳細を見つけることができます。


0

ここで私はそれを試してロジックを書いています:)

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"));
}

}

これは、すべての文字列の回文だけでなく、最長与えます
ヴィヴェック・ミシュラ

0

このソリューションはO(n ^ 2)複雑です。O(1)はスペースの複雑さです。

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

これが私のアルゴリズムです:

1)現在の中心を最初の文字に設定する

2)現在の中心の周りに最大の回文構造が見つかるまで、同時に左右に拡張します

3)見つかった回文が以前の回文よりも大きい場合は、更新する

4)現在の中心を次の文字に設定する

5)文字列内のすべての文字について、ステップ2)から4)を繰り返します。

これはO(n)で実行されます。

それが役に立てば幸い。


5
文字列「aaaaaa」を考えてみましょう。これは、アルゴリズムを使用してO(n ^ 2)で実行されます。
paislee

1
私は当初、OPのアルゴリズム#1はO(n ^ 2)時間であると思っていましたが、実際にはO(n ^ 3)なので、アルゴリズムが達成可能なO(n)限界まで到達していなくても、それはまだ改善です。
j_random_hacker 2012

それが古典的なN2拡張ソリューションです。しかし、このビデオで説明したように実際に近いManacherのソリューションへのだという。youtube.com/watch?v=V-sEwsca1ak違いは、既にスキャン手紙を再スキャンすることを避けることができるように点4 Manacher再利用情報です。
v.oddou 2016

-2

リファレンス:Wikipedia.com

複雑さ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

私の解決策は:

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
とstring-equality()演算のため、これには文字列の長さの立方時間がかかります。OPのアルゴリズム#1と基本的に同じです。Substring()==
j_random_hacker 2012
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.