sql >> データベース >  >> RDS >> Mysql

間違ったバイトカウント長によって破損したシリアル化された文字列を修復するにはどうすればよいですか?

    unserialize() [function.unserialize]: Error at offset invalid serialization dataが原因でした 長さが無効なため

    クイックフィックス

    できることは、recalculating the lengthです。 シリアル化された配列の要素の数

    現在シリアル化されているデータ

    $data = 'a:10:{s:16:"submit_editorial";b:0;s:15:"submit_orig_url";s:13:"www.bbc.co.uk";s:12:"submit_title";s:14:"No title found";s:14:"submit_content";s:12:"dnfsdkfjdfdf";s:15:"submit_category";i:2;s:11:"submit_tags";s:3:"bbc";s:9:"submit_id";b:0;s:16:"submit_subscribe";i:0;s:15:"submit_comments";s:4:"open";s:5:"image";s:19:"C:fakepath100.jpg";}';
    

    再計算なしの例

    var_dump(unserialize($data));
    

    出力

    Notice: unserialize() [function.unserialize]: Error at offset 337 of 338 bytes
    

    再計算

    $data = preg_replace('!s:(\d+):"(.*?)";!e', "'s:'.strlen('$2').':\"$2\";'", $data);
    var_dump(unserialize($data));
    

    出力

    array
      'submit_editorial' => boolean false
      'submit_orig_url' => string 'www.bbc.co.uk' (length=13)
      'submit_title' => string 'No title found' (length=14)
      'submit_content' => string 'dnfsdkfjdfdf' (length=12)
      'submit_category' => int 2
      'submit_tags' => string 'bbc' (length=3)
      'submit_id' => boolean false
      'submit_subscribe' => int 0
      'submit_comments' => string 'open' (length=4)
      'image' => string 'C:fakepath100.jpg' (length=17)
    

    推奨事項 ..私

    この種のクイックフィックスを使用する代わりに...私はあなたが

    で質問を更新することをアドバイスします
    • データのシリアル化方法

    • どのように保存しているか..

    ================================編集1===============================

    エラー

    二重引用符"を使用したため、エラーが生成されました 代わりに一重引用符' そのため、C:\fakepath\100.png C:fakepath100.jpgに変換されました

    エラーを修正するには

    $h->vars['submitted_data']を変更する必要があります From(単一の'に注意してください )

    交換

     $h->vars['submitted_data']['image'] = "C:\fakepath\100.png" ;
    

    あり

     $h->vars['submitted_data']['image'] = 'C:\fakepath\100.png' ;
    

    追加のフィルター

    シリアル化を呼び出す前に、この単純なフィルターを追加することもできます

    function satitize(&$value, $key)
    {
        $value = addslashes($value);
    }
    
    array_walk($h->vars['submitted_data'], "satitize");
    

    UTF文字を使用している場合は、

    を実行することもできます。
     $h->vars['submitted_data'] = array_map("utf8_encode",$h->vars['submitted_data']);
    

    将来のシリアル化されたデータで問題を検出する方法

      findSerializeError ( $data1 ) ;
    

    出力

    Diffrence 9 != 7
        -> ORD number 57 != 55
        -> Line Number = 315
        -> Section Data1  = pen";s:5:"image";s:19:"C:fakepath100.jpg
        -> Section Data2  = pen";s:5:"image";s:17:"C:fakepath100.jpg
                                                ^------- The Error (Element Length)
    

    findSerializeError 機能

    function findSerializeError($data1) {
        echo "<pre>";
        $data2 = preg_replace ( '!s:(\d+):"(.*?)";!e', "'s:'.strlen('$2').':\"$2\";'",$data1 );
        $max = (strlen ( $data1 ) > strlen ( $data2 )) ? strlen ( $data1 ) : strlen ( $data2 );
    
        echo $data1 . PHP_EOL;
        echo $data2 . PHP_EOL;
    
        for($i = 0; $i < $max; $i ++) {
    
            if (@$data1 {$i} !== @$data2 {$i}) {
    
                echo "Diffrence ", @$data1 {$i}, " != ", @$data2 {$i}, PHP_EOL;
                echo "\t-> ORD number ", ord ( @$data1 {$i} ), " != ", ord ( @$data2 {$i} ), PHP_EOL;
                echo "\t-> Line Number = $i" . PHP_EOL;
    
                $start = ($i - 20);
                $start = ($start < 0) ? 0 : $start;
                $length = 40;
    
                $point = $max - $i;
                if ($point < 20) {
                    $rlength = 1;
                    $rpoint = - $point;
                } else {
                    $rpoint = $length - 20;
                    $rlength = 1;
                }
    
                echo "\t-> Section Data1  = ", substr_replace ( substr ( $data1, $start, $length ), "<b style=\"color:green\">{$data1 {$i}}</b>", $rpoint, $rlength ), PHP_EOL;
                echo "\t-> Section Data2  = ", substr_replace ( substr ( $data2, $start, $length ), "<b style=\"color:red\">{$data2 {$i}}</b>", $rpoint, $rlength ), PHP_EOL;
            }
    
        }
    
    }
    

    データベースに保存するためのより良い方法

    $toDatabse = base64_encode(serialize($data));  // Save to database
    $fromDatabase = unserialize(base64_decode($data)); //Getting Save Format 
    


    1. クエリを使用してSQLデータベースの疑わしいモードを通常モードに変更する

    2. データベースのコピーをダウンロードします

    3. PythonとSQLiteの警告

    4. Oracleでのテーブル変更の監視