目次
データの x 行と y 行がそれぞれ含まれる 2 つのファイル a と b があるとします。ここで (x、y は両方とも 1 より大きい) 10 億 )、マシンのメモリは 100M に制限されています。同じレコードを見つけるにはどうすればよいですか? この問題を解決する際の主な困難は、この膨大なデータを一度にメモリに読み込むことができないことです。メモリに読み込めない場合は、一度に、複数回考えることができますか? 毛織物?一緒に議論しましょう" >データの x 行と y 行がそれぞれ含まれる 2 つのファイル a と b があるとします。ここで (x、y は両方とも 1 より大きい) 10 億 )、マシンのメモリは 100M に制限されています。同じレコードを見つけるにはどうすればよいですか? この問題を解決する際の主な困難は、この膨大なデータを一度にメモリに読み込むことができないことです。メモリに読み込めない場合は、一度に、複数回考えることができますか? 毛織物?一緒に議論しましょう
はじめに" >はじめに
考察" >考察
実際の操作" >実際の操作
次に、20 個のファイルで同じレコードを検索する必要があります。実際には、1 つのファイルで同じレコードを検索する必要があります。 20回動作します。
ホームページ バックエンド開発 PHPチュートリアル PHP で 2 つの大きなファイル内の同じレコードを見つけるにはどうすればよいですか?

PHP で 2 つの大きなファイル内の同じレコードを見つけるにはどうすればよいですか?

Jun 23, 2021 am 11:32 AM
php

データの x 行と y 行がそれぞれ含まれる 2 つのファイル a と b があるとします。ここで (x、y は両方とも 1 より大きい) 10 億 )、マシンのメモリは 100M に制限されています。同じレコードを見つけるにはどうすればよいですか? この問題を解決する際の主な困難は、この膨大なデータを一度にメモリに読み込むことができないことです。メモリに読み込めない場合は、一度に、複数回考えることができますか? 毛織物?一緒に議論しましょう

はじめに

それぞれ x 行と y 行のデータを含む 2 つのファイル a と b が与えられます。ここで (x、y はどちらも大きいです) 10 億以上)、マシンのメモリ制限は 100M、同じレコードを見つけるにはどうすればよいでしょうか?

考察

  • 対処する際の主な困難この問題は、この大量のデータを一度にメモリに読み込むことはできません。

  • #一度にメモリに読み込むことができない場合、複数回考えることができますか?可能であれば、複数回読み取った後、同じ値を計算するにはどうすればよいですか?

  • 分割統治思考を使用して、大きなものを小さなものに減らすことができます。ハッシュ後の同じ文字列の値が等しい場合は、ハッシュモジュロを使用してレコードを n 個のファイルに分散することを検討できます。これを取得するにはどうすればよいですか? PHP には 100M のメモリがあり、配列には約 100 万のデータを格納できるため、レコード a と b の行数が 10 億行しかないことを考慮すると、n は少なくとも 200 より大きくなければなりません。

  • 現時点では 200 個のファイルがあります。同じレコードが同じファイル内に存在する必要があり、各ファイルはメモリに読み込むことができます。次に、これら 200 個のファイルから同じレコードを順番に検索し、同じファイルに出力すると、最終的に 2 つのファイル a と b に同じレコードが含まれます。

  • 小さなファイル内で同じレコードを見つけるのは非常に簡単です。レコードの各行をハッシュ テーブルのキーとして使用し、キーの出現回数 >= 2 を数えます。 。

実際の操作

10 億ファイルは大きすぎます。実際の操作は時間の無駄です。実用的な目的を達成するだけです。

問題のサイズは次のように削減されます: 1M メモリ制限、a と b にはそれぞれ 100,000 行のレコードがあります。メモリ制限は PHP の ini_set('memory_limit', '1M');# によって制限できます。 ##。

テスト ファイルの生成

ファイルを満たす乱数の生成:

/**
 * 生成随机数填充文件
 * Author: ClassmateLin
 * Email: classmatelin.site@gmail.com
 * Site: https://www.classmatelin.top
 * @param string $filename 输出文件名
 * @param int $batch 按多少批次生成数据
 * @param int $batchSize 每批数据的大小
 */function generate(string $filename, int $batch=1000, int $batchSize=10000){
    for ($i=0; $i<$batch; $i++) {
        $str = &#39;&#39;;
        for ($j=0; $j<$batchSize; $j++) {
            $str .= rand($batch, $batchSize) . PHP_EOL; // 生成随机数
        }
        file_put_contents($filename, $str, FILE_APPEND);  // 追加模式写入文件
    }}generate(&#39;a.txt&#39;, 10);generate(&#39;b.txt&#39;, 10);
ログイン後にコピー

ファイルの分割

##ハッシュ係数を使用して
    a.txt
  • b.txt を n 個のファイルに分割します。
    /**
     * 用hash取模方式将文件分散到n个文件中
     * Author: ClassmateLin
     * Email: classmatelin.site@gmail.com
     * Site: https://www.classmatelin.top
     * @param string $filename 输入文件名
     * @param int $mod 按mod取模
     * @param string $dir 文件输出目录
     */
    function spiltFile(string $filename, int $mod=20, string $dir=&#39;files&#39;)
    {
        if (!is_dir($dir)){
            mkdir($dir);
        }
    
        $fp = fopen($filename, &#39;r&#39;);
    
        while (!feof($fp)){
            $line = fgets($fp);
            $n = crc32(hash(&#39;md5&#39;, $line)) % $mod; // hash取模
            $filepath = $dir . &#39;/&#39; . $n . &#39;.txt&#39;;  // 文件输出路径
            file_put_contents($filepath, $line, FILE_APPEND); // 追加模式写入文件
        }
    
        fclose($fp);
    }
    
    spiltFile(&#39;a.txt&#39;);
    spiltFile(&#39;b.txt&#39;);
    ログイン後にコピー
splitFile

関数を使用して、以下に示すように files ディレクトリに 20 個のファイルを取得します。

重複レコードの検索

次に、20 個のファイルで同じレコードを検索する必要があります。実際には、1 つのファイルで同じレコードを検索する必要があります。 20回動作します。

#ファイル内の同じレコードの検索:
    /**
     * 查找一个文件中相同的记录输出到指定文件中
     * Author: ClassmateLin
     * Email: classmatelin.site@gmail.com
     * Site: https://www.classmatelin.top
     * @param string $inputFilename 输入文件路径
     * @param string $outputFilename 输出文件路径
     */
    function search(string $inputFilename, $outputFilename=&#39;output.txt&#39;)
    {
        $table = [];
        $fp = fopen($inputFilename, &#39;r&#39;);
    
        while (!feof($fp))
        {
            $line = fgets($fp);
            !isset($table[$line]) ? $table[$line] = 1 : $table[$line]++; // 未设置的值设1,否则自增
        }
    
        fclose($fp);
    
        foreach ($table as $line => $count)
        {
            if ($count >= 2){ // 出现大于2次的则是相同的记录,输出到指定文件中
                file_put_contents($outputFilename, $line, FILE_APPEND);
            }
        }
    }
    ログイン後にコピー
  • #すべてのファイル内の同じレコードの検索:

    /**
     * 从给定目录下文件中分别找出相同记录输出到指定文件中
     * Author: ClassmateLin
     * Email: classmatelin.site@gmail.com
     * Site: https://www.classmatelin.top
     * @param string $dirs 指定目录
     * @param string $outputFilename 输出文件路径
     */
    function searchAll($dirs=&#39;files&#39;, $outputFilename=&#39;output.txt&#39;)
    {
        $files = scandir($dirs);
    
        foreach ($files as $file)
        {
            $filepath = $dirs . &#39;/&#39; . $file;
            if (is_file($filepath)){
                search($filepath, $outputFilename);
            }
        }
    }
    ログイン後にコピー
  • ##大容量ファイル処理の容量問題はここで解決しましたが、時間の問題はどうすればよいでしょうか? CPU のマルチコアを利用すれば 1 台のマシンで処理できます。複数のサーバーで処理されます。

  • 完全なコード

    ログイン後にコピー

    推奨学習:「PHPビデオチュートリアル」

    以上がPHP で 2 つの大きなファイル内の同じレコードを見つけるにはどうすればよいですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

CakePHP プロジェクトの構成 CakePHP プロジェクトの構成 Sep 10, 2024 pm 05:25 PM

この章では、CakePHP の環境変数、一般設定、データベース設定、電子メール設定について理解します。

Ubuntu および Debian 用の PHP 8.4 インストールおよびアップグレード ガイド Ubuntu および Debian 用の PHP 8.4 インストールおよびアップグレード ガイド Dec 24, 2024 pm 04:42 PM

PHP 8.4 では、いくつかの新機能、セキュリティの改善、パフォーマンスの改善が行われ、かなりの量の機能の非推奨と削除が行われています。 このガイドでは、Ubuntu、Debian、またはその派生版に PHP 8.4 をインストールする方法、または PHP 8.4 にアップグレードする方法について説明します。

CakePHP の日付と時刻 CakePHP の日付と時刻 Sep 10, 2024 pm 05:27 PM

Cakephp4 で日付と時刻を操作するには、利用可能な FrozenTime クラスを利用します。

CakePHP ファイルのアップロード CakePHP ファイルのアップロード Sep 10, 2024 pm 05:27 PM

ファイルのアップロードを行うには、フォーム ヘルパーを使用します。ここではファイルアップロードの例を示します。

CakePHP ルーティング CakePHP ルーティング Sep 10, 2024 pm 05:25 PM

この章では、ルーティングに関連する次のトピックを学習します。

CakePHP について話し合う CakePHP について話し合う Sep 10, 2024 pm 05:28 PM

CakePHP は、PHP 用のオープンソース フレームワークです。これは、アプリケーションの開発、展開、保守をより簡単にすることを目的としています。 CakePHP は、強力かつ理解しやすい MVC のようなアーキテクチャに基づいています。モデル、ビュー、コントローラー

PHP 開発用に Visual Studio Code (VS Code) をセットアップする方法 PHP 開発用に Visual Studio Code (VS Code) をセットアップする方法 Dec 20, 2024 am 11:31 AM

Visual Studio Code (VS Code とも呼ばれる) は、すべての主要なオペレーティング システムで利用できる無料のソース コード エディター (統合開発環境 (IDE)) です。 多くのプログラミング言語の拡張機能の大規模なコレクションを備えた VS Code は、

CakePHP バリデータの作成 CakePHP バリデータの作成 Sep 10, 2024 pm 05:26 PM

Validator は、コントローラーに次の 2 行を追加することで作成できます。

See all articles