主頁 > 軟體工程 > 在Linux系統上打開具有特定格式的檔案而無需迭代

在Linux系統上打開具有特定格式的檔案而無需迭代

2022-05-09 06:57:05 軟體工程

我有一個包含數千個檔案的目錄,所有檔案都具有一般格式,***-***.txt除了一個具有 format 的檔案***.txt,我保證它恰好有一個 & ,其確切名稱我無法知道。

我想知道是否可以在不遍歷所有目錄、僅使用 C 或 Linux 函式且不呼叫system().

uj5u.com熱心網友回復:

如前所述,沒有辦法避免對檔案名進行迭代。好訊息是這真的沒關系。即使有成千上萬的條目(哇?!)它也不應該花費太多時間,所以如果你只需要做一次,你就很好了。

C

// Life is easy in C  

#include <filesystem>
#include <iostream>

namespace fs = std::filesystem;

fs::path find_magic_file( const fs::path directory )
{
  for (auto & entry : fs::directory_iterator( directory ))
    if (entry.path().stem().string().find( '-' ) == std::string::npos)
      return entry.path();
  return "";
}

int main( int num_directories, char ** directory_names )
{
  for (int n = 1;  n < num_directories;  n  = 1)
  {
    auto filepath = find_magic_file( directory_names[n] );
    if (!filepath.empty())
      std::cout << filepath.string() << "\n";
  }
}

C

// C requires you to recur to the OS facilities directly
// This example handles Windows and Linux/POSIX systems

#ifdef _WIN32
  #define _CRT_SECURE_NO_WARNINGS
#endif

#include <iso646.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#ifdef _WIN32
  #include <windows.h>
  
  char * find_magic_file( const char * directory )
  {
    WIN32_FIND_DATA entry;

    char * dirname = malloc( strlen( directory )   5 );
    if (!dirname) return NULL;

    HANDLE h = FindFirstFile( strcat( strcpy( dirname, directory ), "/*.*" ), &entry );
    if (h == INVALID_HANDLE_VALUE) return NULL;
    
    while ((strcmp( entry.cFileName, "."  ) == 0) 
        or (strcmp( entry.cFileName, ".." ) == 0)
        or (strchr( entry.cFileName, '-' )))
      if (!FindNextFile( h, &entry )) 
        break;
      
    FindClose( h );
    free( dirname );
    if (strchr( entry.cFileName, '-' )) return NULL;
    
    char * filepath = calloc( strlen( directory )   1   strlen( entry.cFileName )   1, 1 );
    if (filepath) strcat( strcat( strcpy( filepath, directory ), "/" ), entry.cFileName );
    return filepath;
  }
  
#else
  #include <dirent.h>
  #include <sys/types.h>

  char * find_magic_file( const char * directory )
  {
    char * filepath = NULL;
    DIR * dir = opendir( directory );
    if (dir)
    {
      struct dirent * entry;
      while ((entry = readdir( dir )))
      {
        if (!strchr( entry->d_name, '-' )
            and (strcmp( entry->d_name, "."  ) != 0)
            and (strcmp( entry->d_name, ".." ) != 0))
        {
          filepath = malloc( strlen( directory )   1   strlen( entry->d_name )   1 );
          if (filepath) strcat( strcat( strcpy( filepath, directory ), "/" ), entry->d_name );
          break;
        }
      }
      closedir( dir );
    }
    return filepath;
  }
#endif

int main( int num_directories, char ** directory_names )
{
  for (int n = 1;  n < num_directories;  n  = 1)
  {
    char * filepath = find_magic_file( directory_names[n] );
    if (filepath)
    {
      puts( filepath );
      free( filepath );
    }
  }
}

您會注意到main()兩種語言之間唯一真正的區別是find_magic_file().

uj5u.com熱心網友回復:

在 Linux 中,您可以使用 POSIX.1 scandir()函式,以及一個過濾器函式,該函式拒絕帶有破折號的檔案名。

這樣,掃描/迭代是在 C 庫中以適合手頭作業系統的方式完成的,并且如果在掃描期間重命名檔案,則不應混淆。如果將無破折號檔案重命名為破折號檔案,而將另一個破折號檔案重命名為無破折號名稱,則恰好在目錄掃描期間,可能找不到任何一個名稱。(并且,如果它們以相反的順序完成,則可以看到兩個名稱。)

這是一個示例程式,具有完整的錯誤檢查:

#define _POSIX_C_SOURCE 200809L
#include <stdlib.h>
#include <dirent.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>

static int dashless_filter(const struct dirent *ent)
{
    /* Reject file names that begin with a dot '.' ("hidden" files) */
    if (ent->d_name[0] == '.')
        return 0;

    /* Reject file names with dash in them */
    if (strchr(ent->d_name, '-'))
        return 0;

    return 1;
}

char *dashless(const char *dirpath)
{
    struct dirent **list = NULL;
    char           *result = NULL;
    int             num, err;

    /* Return NULL if dirpath is null or empty */
    if (!dirpath || !*dirpath) {
        errno = ENOTDIR;
        return NULL;
    }

    do {
        num = scandir(dirpath, &list, dashless_filter, NULL);
        if (num == -1) {
            err = errno;  /* errno set by scandir() */
            break;
        }

        if (num < 1 || !list || !list[0] || list[0]->d_name[0] == '\0') {
            err = ENOENT;  /* No matching files found */
            break;
        } else
        if (num > 1) {
            err = EMFILE;  /* More than one matching file found. */
            break;
        }

        result = strdup(list[0]->d_name);
        if (!result) {
            err = ENOMEM;
        } else {
            err = 0;
        }
    } while (0);

    /* Free all entries in the list, */
    while (num-->0)
        free(list[num]);
    /* and the list itself. */
    free(list);

    errno = err;
    return result;
}

int main(int argc, char *argv[])
{
    int  status = EXIT_SUCCESS;
    int  arg;

    if (argc < 2 || !strcmp(argv[1], "-h") || !strcmp(argv[1], "--help")) {
        const char *arg0 = (argc > 0 && argv && argv[0] && argv[0][0]) ? argv[0] : "(this)";
        fprintf(stderr, "\n");
        fprintf(stderr, "Usage: %s [ -h | --help ]\n", arg0);
        fprintf(stderr, "       %s DIRECTORY [ DIRECTORY ... ]\n", arg0);
        fprintf(stderr, "\n");
        fprintf(stderr, "This program reports if each specified directory contains\n");
        fprintf(stderr, "a single file without a dash '-' in its name.\n");
        fprintf(stderr, "\n");
        return EXIT_SUCCESS;
    }

    for (arg = 1; arg < argc; arg  ) {
        char *filepath = dashless(argv[arg]);
        if (filepath) {
            printf("%s: %s\n", argv[arg], filepath);
            free(filepath);
        } else {
            if (errno == EMFILE)
                fprintf(stderr, "%s: Multiple dashless files exist.\n", argv[arg]);
            else
                fprintf(stderr, "%s: %s.\n", argv[arg], strerror(errno));
            status = EXIT_FAILURE;
        }
    }

    return status;
}

請注意,它顯式檢查單個檔案名。

如果生成檔案的應用程式或腳本也確實創建了一個具有固定、已知名稱的硬鏈接到一個無破折號檔案,不管它可能是什么,那會更好。這樣,人們總是可以使用固定的符號鏈接/硬鏈接名稱,并且如果重命名無破折號檔案,則不會有任何競爭視窗在此期間將發現一個無破折號檔案或兩個無破折號檔案。

轉載請註明出處,本文鏈接:https://www.uj5u.com/gongcheng/470632.html

標籤:C linux 文件

上一篇:c中的tictactoe-預期的表達-怎么了?

下一篇:返回列表

標籤雲
其他(140250) Python(31804) JavaScript(20059) Java(15546) C(13449) 區塊鏈(8221) AI(7469) 基礎類(6313) C#(6102) MySQL(6028) 爪哇(5712) html(5219) 腳本語言(PerlPython)(5129) 非技術區(4971) PHP(4916) sql(4870) 熊猫(4865) Linux(4690) 数组(4377) Android(4321) R(4254) 反应(3923) 数据框(3396) css(3349) C語言(3288) 节点.js(3137) C++語言(3117) Java相關(2746) 疑難問題(2699) json(2613) 單片機工控(2479) 列表(2474) 扑(2424) VBA(2399) 安卓(2387) 打字稿(2263) ASP.NET(2045) iOS(1983) 细绳(1967) Web開發(1951) MongoDB(1864) 網絡通信(1793) 蟒蛇-3.x(1774) 數據庫相關(1767) VB基礎類(1755) 循环(1742) 麻木的(1740) .NETCore(1728) 正则表达式(1719) 字典(1712) 镖(1694) 擅长(1673) 迅速(1651) 開發(1646) C++(1646) 系統維護與使用區(1617) Unity3D(1613) .NET技术(1600) HtmlCss(1590) 基礎和管理(1579)

熱門瀏覽
  • Git本地庫既關聯GitHub又關聯Gitee

    創建代碼倉庫 使用gitee舉例(github和gitee差不多) 1.在gitee右上角點擊+,選擇新建倉庫 ? 2.選擇填寫倉庫資訊,然后進行創建 ? 3.服務端已經準備好了,本地開始作準備 (1)Git 全域設定 git config --global user.name "成鈺" git c ......

    uj5u.com 2020-09-10 05:04:14 more
  • CODING DevOps 代碼質量實戰系列第二課,相約周三

    隨著 ToB(企業服務)的興起和 ToC(消費互聯網)產品進入成熟期,線上故障帶來的損失越來越大,代碼質量越來越重要,而「質量內建」正是 DevOps 核心理念之一。**《DevOps 代碼質量實戰(PHP 版)》**為 CODING DevOps 代碼質量實戰系列的第二課,同時也是本系列的 PHP ......

    uj5u.com 2020-09-10 05:07:43 more
  • 推薦Scrum書籍

    推薦Scrum書籍 直接上干貨,推薦書籍清單如下(推薦有順序的哦) Scrum指南 Scrum精髓 Scrum敏捷軟體開發 Scrum捷徑 硝煙中的Scrum和XP : 我們如何實施Scrum 敏捷軟體開發:Scrum實戰指南 Scrum要素 大規模Scrum:大規模敏捷組織的設計 用戶故事地圖 用 ......

    uj5u.com 2020-09-10 05:07:45 more
  • CODING DevOps 代碼質量實戰系列最后一課,周四發車

    隨著 ToB(企業服務)的興起和 ToC(消費互聯網)產品進入成熟期,線上故障帶來的損失越來越大,代碼質量越來越重要,而「質量內建」正是 DevOps 核心理念之一。 **《DevOps 代碼質量實戰(Java 版)》**為 CODING DevOps 代碼質量實戰系列的最后一課,同時也是本系列的 ......

    uj5u.com 2020-09-10 05:07:52 more
  • 敏捷軟體工程實踐書籍

    Scrum轉型想要做好,第一步先了解并真正落實Scrum,那么我推薦的Scrum書籍是要看懂并實踐的。第二步是團隊的工程實踐要做扎實。 下面推薦工程實踐書單: 重構:改善既有代碼的設計 決議極限編程 : 擁抱變化 代碼整潔代碼 程式員的職業素養 修改代碼的藝術 撰寫可讀代碼的藝術 測驗驅動開發 : ......

    uj5u.com 2020-09-10 05:07:55 more
  • Jenkins+svn+nginx實作windows環境自動部署vue前端專案

    前面文章介紹了Jenkins+svn+tomcat實作自動化部署,現在終于有空抽時間出來寫下Jenkins+svn+nginx實作自動部署vue前端專案。 jenkins的安裝和配置已經在前面文章進行介紹,下面介紹實作vue前端專案需要進行的哪些額外的步驟。 注意:在安裝jenkins和nginx的 ......

    uj5u.com 2020-09-10 05:08:49 more
  • CODING DevOps 微服務專案實戰系列第一課,明天等你

    CODING DevOps 微服務專案實戰系列第一課**《DevOps 微服務專案實戰:DevOps 初體驗》**將由 CODING DevOps 開發工程師 王寬老師 向大家介紹 DevOps 的基本理念,并探討為什么現代開發活動需要 DevOps,同時將以 eShopOnContainers 項 ......

    uj5u.com 2020-09-10 05:09:14 more
  • CODING DevOps 微服務專案實戰系列第二課來啦!

    近年來,工程專案的結構越來越復雜,需要接入合適的持續集成流水線形式,才能滿足更多變的需求,那么如何優雅地使用 CI 能力提升生產效率呢?CODING DevOps 微服務專案實戰系列第二課 《DevOps 微服務專案實戰:CI 進階用法》 將由 CODING DevOps 全堆疊工程師 何晨哲老師 向 ......

    uj5u.com 2020-09-10 05:09:33 more
  • CODING DevOps 微服務專案實戰系列最后一課,周四開講!

    隨著軟體工程越來越復雜化,如何在 Kubernetes 集群進行灰度發布成為了生產部署的”必修課“,而如何實作安全可控、自動化的灰度發布也成為了持續部署重點關注的問題。CODING DevOps 微服務專案實戰系列最后一課:**《DevOps 微服務專案實戰:基于 Nginx-ingress 的自動 ......

    uj5u.com 2020-09-10 05:10:00 more
  • CODING 儀表盤功能正式推出,實作作業資料可視化!

    CODING 儀表盤功能現已正式推出!該功能旨在用一張張統計卡片的形式,統計并展示使用 CODING 中所產生的資料。這意味著無需額外的設定,就可以收集歸納寶貴的作業資料并予之量化分析。這些海量的資料皆會以圖表或串列的方式躍然紙上,方便團隊成員隨時查看各專案的進度、狀態和指標,云端協作迎來真正意義上 ......

    uj5u.com 2020-09-10 05:11:01 more
最新发布