使用PHP怎么在兩個(gè)文件中找出相同的記錄

使用PHP怎么在兩個(gè)文件中找出相同的記錄?很多新手對(duì)此不是很清楚,為了幫助大家解決這個(gè)難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來(lái)學(xué)習(xí)下,希望你能有所收獲。

讓客戶滿意是我們工作的目標(biāo),不斷超越客戶的期望值來(lái)自于我們對(duì)這個(gè)行業(yè)的熱愛(ài)。我們立志把好的技術(shù)通過(guò)有效、簡(jiǎn)單的方式提供給客戶,將通過(guò)不懈努力成為客戶在信息化領(lǐng)域值得信任、有價(jià)值的長(zhǎng)期合作伙伴,公司提供的服務(wù)項(xiàng)目有:空間域名、網(wǎng)站空間、營(yíng)銷軟件、網(wǎng)站建設(shè)、遼陽(yáng)縣網(wǎng)站維護(hù)、網(wǎng)站推廣。

引言

給定a,b兩個(gè)文件, 分別有x,y行數(shù)據(jù), 其中(x, y均大于10億), 機(jī)器內(nèi)存限制100M,該如何找出其中相同的記錄?<!--more-->

思路

  • 處理該問(wèn)題的困難主要是無(wú)法將這海量數(shù)據(jù)一次性讀內(nèi)內(nèi)存中.

  • 一次性讀不進(jìn)內(nèi)存中,那么是否可以考慮多次呢?如果可以,那么多次讀入要怎么計(jì)算相同的值呢?

  • 我們可以用分治思想, 大而化小。相同字符串的值hash過(guò)后是相等的, 那么我們可以考慮使用hash取模, 將記錄分散到n個(gè)文件中。這個(gè)n怎么取呢? PHP 100M內(nèi)存,數(shù)組大約可以存100w的數(shù)據(jù), 那么按a,b記錄都只有10億行來(lái)算, n至少要大于200。

  • 此時(shí)有200個(gè)文件,相同的記錄肯定在同一個(gè)文件中,并且每個(gè)文件都可以全部讀進(jìn)內(nèi)存。那么可以依次找出這200個(gè)文件中各自相同的記錄,然后輸出到同一個(gè)文件中,得到的最終結(jié)果就是a, b兩個(gè)文件中相同的記錄。

  • 找一個(gè)小文件中相同的記錄很簡(jiǎn)單了吧,將每行記錄作為hash表的key, 統(tǒng)計(jì)key的出現(xiàn)次數(shù)>=2就可以了。

實(shí)操

10億各文件太大了,實(shí)操浪費(fèi)時(shí)間,達(dá)到實(shí)踐目的即可。

問(wèn)題規(guī)??s小為:  1M內(nèi)存限制, a, b各有10w行記錄, 內(nèi)存限制可以用PHP的ini_set('memory_limit', '1M');來(lái)限制。

生成測(cè)試文件

生成隨機(jī)數(shù)用于填充文件:

/**
 * 生成隨機(jī)數(shù)填充文件
 * Author: ClassmateLin
 * Email: classmatelin.site@gmail.com
 * Site: https://www.classmatelin.top
 * @param string $filename 輸出文件名
 * @param int $batch 按多少批次生成數(shù)據(jù)
 * @param int $batchSize 每批數(shù)據(jù)的大小
 */
function generate(string $filename, int $batch=1000, int $batchSize=10000)
{
    for ($i=0; $i<$batch; $i++) {
        $str = '';
        for ($j=0; $j<$batchSize; $j++) {
            $str .= rand($batch, $batchSize) . PHP_EOL; // 生成隨機(jī)數(shù)
        }
        file_put_contents($filename, $str, FILE_APPEND);  // 追加模式寫入文件
    }
}

generate('a.txt', 10);
generate('b.txt', 10);

分割文件

  • a.txt, b.txt通過(guò)hash取模的方式分割到n個(gè)文件中.

/**
 * 用hash取模方式將文件分散到n個(gè)文件中
 * 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='files')
{
    if (!is_dir($dir)){
        mkdir($dir);
    }

    $fp = fopen($filename, 'r');

    while (!feof($fp)){
        $line = fgets($fp);
        $n = crc32(hash('md5', $line)) % $mod; // hash取模
        $filepath = $dir . '/' . $n . '.txt';  // 文件輸出路徑
        file_put_contents($filepath, $line, FILE_APPEND); // 追加模式寫入文件
    }

    fclose($fp);
}

spiltFile('a.txt');
spiltFile('b.txt');
  • 執(zhí)行splitFile函數(shù), 得到如下圖files目錄的20個(gè)文件。

查找重復(fù)記錄

現(xiàn)在需要查找20個(gè)文件中相同的記錄, 其實(shí)也就是找一個(gè)文件中的相同記錄,操作個(gè)20次。

  • 找一個(gè)文件中的相同記錄:

/**
 * 查找一個(gè)文件中相同的記錄輸出到指定文件中
 * Author: ClassmateLin
 * Email: classmatelin.site@gmail.com
 * Site: https://www.classmatelin.top
 * @param string $inputFilename 輸入文件路徑
 * @param string $outputFilename 輸出文件路徑
 */
function search(string $inputFilename, $outputFilename='output.txt')
{
    $table = [];
    $fp = fopen($inputFilename, 'r');

    while (!feof($fp))
    {
        $line = fgets($fp);
        !isset($table[$line]) ? $table[$line] = 1 : $table[$line]++; // 未設(shè)置的值設(shè)1,否則自增
    }

    fclose($fp);

    foreach ($table as $line => $count)
    {
        if ($count >= 2){ // 出現(xiàn)大于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='files', $outputFilename='output.txt')
{
    $files = scandir($dirs);

    foreach ($files as $file)
    {
        $filepath = $dirs . '/' . $file;
        if (is_file($filepath)){
            search($filepath, $outputFilename);
        }
    }
}
  • 到這里已經(jīng)解決了大文件處理的空間問(wèn)題,那么時(shí)間問(wèn)題該如何處理? 單機(jī)可通過(guò)利用CPU的多核心處理,不夠的話通過(guò)多臺(tái)服務(wù)器處理。

完整代碼

<?php
ini_set('memory_limit', '1M'); // 內(nèi)存限制1M

/**
 * 生成隨機(jī)數(shù)填充文件
 * Author: ClassmateLin
 * Email: classmatelin.site@gmail.com
 * Site: https://www.classmatelin.top
 * @param string $filename 輸出文件名
 * @param int $batch 按多少批次生成數(shù)據(jù)
 * @param int $batchSize 每批數(shù)據(jù)的大小
 */
function generate(string $filename, int $batch=1000, int $batchSize=10000)
{
    for ($i=0; $i<$batch; $i++) {
        $str = '';
        for ($j=0; $j<$batchSize; $j++) {
            $str .= rand($batch, $batchSize) . PHP_EOL; // 生成隨機(jī)數(shù)
        }
        file_put_contents($filename, $str, FILE_APPEND);  // 追加模式寫入文件
    }
}




/**
 * 用hash取模方式將文件分散到n個(gè)文件中
 * 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='files')
{
    if (!is_dir($dir)){
        mkdir($dir);
    }

    $fp = fopen($filename, 'r');

    while (!feof($fp)){
        $line = fgets($fp);
        $n = crc32(hash('md5', $line)) % $mod; // hash取模
        $filepath = $dir . '/' . $n . '.txt';  // 文件輸出路徑
        file_put_contents($filepath, $line, FILE_APPEND); // 追加模式寫入文件
    }

    fclose($fp);
}




/**
 * 查找一個(gè)文件中相同的記錄輸出到指定文件中
 * Author: ClassmateLin
 * Email: classmatelin.site@gmail.com
 * Site: https://www.classmatelin.top
 * @param string $inputFilename 輸入文件路徑
 * @param string $outputFilename 輸出文件路徑
 */
function search(string $inputFilename, $outputFilename='output.txt')
{
    $table = [];
    $fp = fopen($inputFilename, 'r');

    while (!feof($fp))
    {
        $line = fgets($fp);
        !isset($table[$line]) ? $table[$line] = 1 : $table[$line]++; // 未設(shè)置的值設(shè)1,否則自增
    }

    fclose($fp);

    foreach ($table as $line => $count)
    {
        if ($count >= 2){ // 出現(xiàn)大于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='files', $outputFilename='output.txt')
{
    $files = scandir($dirs);

    foreach ($files as $file)
    {
        $filepath = $dirs . '/' . $file;
        if (is_file($filepath)){
            search($filepath, $outputFilename);
        }
    }
}

// 生成文件
generate('a.txt', 10);
generate('b.txt', 10);

// 分割文件
spiltFile('a.txt');
spiltFile('b.txt');

// 查找記錄
searchAll('files', 'output.txt');

看完上述內(nèi)容是否對(duì)您有幫助呢?如果還想對(duì)相關(guān)知識(shí)有進(jìn)一步的了解或閱讀更多相關(guān)文章,請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對(duì)創(chuàng)新互聯(lián)的支持。

分享標(biāo)題:使用PHP怎么在兩個(gè)文件中找出相同的記錄
文章位置:http://muchs.cn/article34/ihpope.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站導(dǎo)航、外貿(mào)建站面包屑導(dǎo)航、定制開(kāi)發(fā)、營(yíng)銷型網(wǎng)站建設(shè)品牌網(wǎng)站設(shè)計(jì)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

營(yíng)銷型網(wǎng)站建設(shè)