C语言二叉树 遍历目录树

简介: C语言二叉树 遍历目录树
#include "stdio.h"
#include "windows.h"
#include <iostream>
using namespace std;
unsigned long sum = 0;
//
// 目录树链表结点定义
typedef struct _tFileTreeItem
{
    struct _tFileTreeItem* pPrevItem;   // 前一个单元
    struct _tFileTreeItem* pNextItem;   // 后一个单元
    struct _tFileTreeItem* pParentItem; // 父单元
    struct _tFileTreeItem* pChildItem;  // 子单元
    char FilePath[MAX_PATH];
    char FileName[MAX_PATH];
    unsigned long dwHashIndex;          // 索引序号
    int level;                          // 树的深度
    int nNameLength;                    // 名称长度
    char *strItemName;                  // 单元名称,不包含路径.(可变长度)
}FileTreeItem, *PFileTreeItem;
//
//查找文件或文件夹,保存到树中
void FindDir( FileTreeItem* TreeNode )
{
    WIN32_FIND_DATA fd;
    HANDLE hSearch;
    char filePathName[MAX_PATH];
    FileTreeItem* temp;
    FileTreeItem* tempnode;
    ZeroMemory( &fd, sizeof(WIN32_FIND_DATA) );
    ZeroMemory( filePathName, MAX_PATH );
    strcpy_s( filePathName, MAX_PATH, TreeNode->FilePath );
    if( filePathName[strlen(filePathName) - 1] != '\\' )
    {
        strcat_s( filePathName, MAX_PATH, "\\" );
        strcat_s( TreeNode->FilePath, MAX_PATH, "\\" );
    }
    strcat_s( filePathName, MAX_PATH, "*" );
    char* wcFilePathName = filePathName;
    hSearch = FindFirstFile( wcFilePathName, &fd );
    if ( hSearch != INVALID_HANDLE_VALUE )
    {
        if( (fd.dwFileAttributes == FILE_ATTRIBUTE_DIRECTORY)
            && lstrcmp(fd.cFileName, ".") && lstrcmp(fd.cFileName,"..") )      
        {
            tempnode= new FileTreeItem;
            char *tempBuffer = fd.cFileName ;
            //保存临时节点的信息
            tempnode->level = TreeNode->level + 1;
            strcpy_s( tempnode->FilePath, MAX_PATH, TreeNode->FilePath );
            strcat_s( tempnode->FilePath, MAX_PATH, tempBuffer );
            strcpy_s( tempnode->FileName, MAX_PATH, tempBuffer );
            tempnode->pPrevItem = NULL;
            tempnode->pNextItem = NULL;
            tempnode->pParentItem = NULL;
            tempnode->pChildItem = NULL;
            if( NULL == TreeNode->pChildItem )
            {// 根结点没有孩子结点,这个就是第一个孩子结点了
                TreeNode->pChildItem = tempnode;
                tempnode->pPrevItem = tempnode->pNextItem = tempnode;// 双向循环链表指针指向自身
            }
            else
            {// 将此结点插入到父结点指向的双向循环链表中去
                temp = TreeNode->pChildItem;// 指向父结点的第一个孩子用来插入结点
                // 将此结点插入到第一个结点之前,即本级目录树的尾部【小技巧】
                tempnode->pNextItem = temp;
                tempnode->pPrevItem = temp->pPrevItem;
                temp->pPrevItem->pNextItem = tempnode;
                temp->pPrevItem = tempnode;
            }
            tempnode->pParentItem = TreeNode;// 指向父结点
            FindDir(tempnode);
        }
        while( FindNextFile(hSearch, &fd) )
        {
            if ( !lstrcmp(fd.cFileName, ".") || !lstrcmp(fd.cFileName,"..") )
            {
                continue;
            }
            else if ( (fd.dwFileAttributes == FILE_ATTRIBUTE_DIRECTORY) || (fd.dwFileAttributes ==48)
                && lstrcmp(fd.cFileName, ".") && lstrcmp(fd.cFileName,"..") )
            { 
                tempnode= new FileTreeItem;
                char *tempBuffer = fd.cFileName ;
                //保存临时节点的信息
                tempnode->level = TreeNode->level + 1;
                strcpy_s( tempnode->FilePath, MAX_PATH, TreeNode->FilePath );
                strcat_s( tempnode->FilePath, MAX_PATH, tempBuffer );
                strcpy_s( tempnode->FileName, MAX_PATH, tempBuffer );
                tempnode->pPrevItem = NULL;
                tempnode->pNextItem = NULL;
                tempnode->pParentItem = NULL;
                tempnode->pChildItem = NULL;
                if( NULL == TreeNode->pChildItem )
                {// 根结点没有孩子结点,这个就是第一个孩子结点了
                    TreeNode->pChildItem = tempnode;
                    tempnode->pPrevItem = tempnode->pNextItem = tempnode;// 双向循环链表指针指向自身
                }
                else
                {// 将此结点插入到父结点指向的双向循环链表中去
                    temp = TreeNode->pChildItem;// 指向父结点的第一个孩子用来插入结点
                    // 将此结点插入到第一个结点之前,即本级目录树的尾部【小技巧】
                    tempnode->pNextItem = temp;
                    tempnode->pPrevItem = temp->pPrevItem;
                    temp->pPrevItem->pNextItem = tempnode;
                    temp->pPrevItem = tempnode;
                }
                tempnode->pParentItem = TreeNode;// 指向父结点
                FindDir(tempnode);
            }
            else if ( lstrcmp( fd.cFileName, ".." ) && fd.dwFileAttributes != 48 )
            {// 此时存放的是文件
                tempnode= new FileTreeItem;
                char *tempBuffer = fd.cFileName;
                //保存临时节点的信息
                tempnode->level = TreeNode->level + 1;
                strcpy_s( tempnode->FilePath, MAX_PATH, TreeNode->FilePath );
                strcat_s( tempnode->FilePath, MAX_PATH, tempBuffer );
                strcpy_s( tempnode->FileName, MAX_PATH, tempBuffer );
                tempnode->pPrevItem = NULL;
                tempnode->pNextItem = NULL;
                tempnode->pParentItem = NULL;
                tempnode->pChildItem = NULL;
                if( NULL == TreeNode->pChildItem )
                {// 根结点没有孩子结点,这个就是第一个孩子结点了
                    TreeNode->pChildItem = tempnode;
                    tempnode->pPrevItem = tempnode->pNextItem = tempnode;// 双向循环链表指针指向自身
                }
                else
                {// 将此结点插入到父结点指向的双向循环链表中去
                    temp = TreeNode->pChildItem;// 指向父结点的第一个孩子用来插入结点
                    // 将此结点插入到第一个结点之前,即本级目录树的尾部【小技巧】
                    tempnode->pNextItem = temp;
                    tempnode->pPrevItem = temp->pPrevItem;
                    temp->pPrevItem->pNextItem = tempnode;
                    temp->pPrevItem = tempnode;
                }
                tempnode->pParentItem = TreeNode;// 指向父结点
            }
        }
        FindClose(hSearch);
    }
}
//
//深度遍历
void SearchTree(FileTreeItem* treeRoot)
{
    printf(treeRoot->FilePath);
    printf("\tlevel:%d.",treeRoot->level );
    printf("\n");
    sum++;
    if( (treeRoot->pChildItem != NULL) )
    {
        SearchTree( treeRoot->pChildItem );
    }
    if ( treeRoot->pNextItem != treeRoot->pParentItem->pChildItem )
    {
        SearchTree( treeRoot->pNextItem );
    }
}
void main()
{
    char* input_path= "e:\\";
    //初始化树根节点
    DWORD dw_time_start  = GetTickCount();
    FileTreeItem *DirTreeRoot= new FileTreeItem;
    DirTreeRoot->level = 0; 
    strcpy_s( DirTreeRoot->FileName, strlen(DirTreeRoot->FileName), input_path );
    strcpy_s(DirTreeRoot->FilePath, strlen(DirTreeRoot->FilePath), input_path);
    DirTreeRoot->pPrevItem = NULL;
    DirTreeRoot->pNextItem = NULL;
    DirTreeRoot->pParentItem = DirTreeRoot;
    DirTreeRoot->pChildItem = NULL;     
    FindDir( DirTreeRoot );
    DWORD dw_time_end  = GetTickCount();
    DWORD dw_time_delta  = dw_time_end - dw_time_start;
    SearchTree(DirTreeRoot->pChildItem);
    cout << "目录树建立时间:" << dw_time_delta / 1000.0 << "秒." << endl;
    cout << "文件和目录的总数为:" << sum << endl;
    system( "pause" );
}
相关文章
|
11月前
|
存储 C语言
c语言——二叉树
二叉树的概念在这里就不进行过多的赘述,那么主要说一下我认为重要的部分,第一点就是二叉树里面部分概念的理解:就比如说,你对于如何构建二叉树,掌握的十分深刻,但刷题的时候对于一些题目所给的概念不清楚,导致看不明白题目,这课不好,二叉树的概念如下图所示,其实都很简单,主要是当给他的名字时,你明不明白。还有对于满二叉树与完全二叉树。
299 0
|
存储 搜索推荐 算法
【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)
本文介绍了树和二叉树的基本概念及结构,重点讲解了堆这一重要的数据结构。堆是一种特殊的完全二叉树,常用于实现优先队列和高效的排序算法(如堆排序)。文章详细描述了堆的性质、存储方式及其实现方法,包括插入、删除和取堆顶数据等操作的具体实现。通过这些内容,读者可以全面了解堆的原理和应用。
803 16
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
518 5
|
存储 算法 C语言
C语言中常见的字符串处理技巧,包括字符串的定义、初始化、输入输出、长度计算、比较、查找与替换、拼接、截取、转换、遍历及注意事项
本文深入探讨了C语言中常见的字符串处理技巧,包括字符串的定义、初始化、输入输出、长度计算、比较、查找与替换、拼接、截取、转换、遍历及注意事项,并通过案例分析展示了实际应用,旨在帮助读者提高编程效率和代码质量。
869 4
|
C语言
【数据结构】二叉树(c语言)(附源码)
本文介绍了如何使用链式结构实现二叉树的基本功能,包括前序、中序、后序和层序遍历,统计节点个数和树的高度,查找节点,判断是否为完全二叉树,以及销毁二叉树。通过手动创建一棵二叉树,详细讲解了每个功能的实现方法和代码示例,帮助读者深入理解递归和数据结构的应用。
1773 9
|
C语言
数据结构基础详解(C语言):图的基本概念_无向图_有向图_子图_生成树_生成森林_完全图
本文介绍了图的基本概念,包括图的定义、无向图与有向图、简单图与多重图等,并解释了顶点度、路径、连通性等相关术语。此外还讨论了子图、生成树、带权图及几种特殊形态的图,如完全图和树等。通过这些概念,读者可以更好地理解图论的基础知识。
1516 8
|
存储 算法 C语言
数据结构基础详解(C语言): 二叉树的遍历_线索二叉树_树的存储结构_树与森林详解
本文从二叉树遍历入手,详细介绍了先序、中序和后序遍历方法,并探讨了如何构建二叉树及线索二叉树的概念。接着,文章讲解了树和森林的存储结构,特别是如何将树与森林转换为二叉树形式,以便利用二叉树的遍历方法。最后,讨论了树和森林的遍历算法,包括先根、后根和层次遍历。通过这些内容,读者可以全面了解二叉树及其相关概念。
519 6
|
存储 算法 C语言
C语言手撕实战代码_二叉树_构造二叉树_层序遍历二叉树_二叉树深度的超详细代码实现
这段代码和文本介绍了一系列二叉树相关的问题及其解决方案。其中包括根据前序和中序序列构建二叉树、通过层次遍历序列和中序序列创建二叉树、计算二叉树节点数量、叶子节点数量、度为1的节点数量、二叉树高度、特定节点子树深度、判断两棵树是否相似、将叶子节点链接成双向链表、计算算术表达式的值、判断是否为完全二叉树以及求二叉树的最大宽度等。每道题目均提供了详细的算法思路及相应的C/C++代码实现,帮助读者理解和掌握二叉树的基本操作与应用。
503 2
|
存储 C语言
数据结构基础详解(C语言): 树与二叉树的应用_哈夫曼树与哈夫曼曼编码_并查集_二叉排序树_平衡二叉树
本文详细介绍了树与二叉树的应用,涵盖哈夫曼树与哈夫曼编码、并查集以及二叉排序树等内容。首先讲解了哈夫曼树的构造方法及其在数据压缩中的应用;接着介绍了并查集的基本概念、存储结构及优化方法;随后探讨了二叉排序树的定义、查找、插入和删除操作;最后阐述了平衡二叉树的概念及其在保证树平衡状态下的插入和删除操作。通过本文,读者可以全面了解树与二叉树在实际问题中的应用技巧和优化策略。
609 2
|
存储 算法 C语言
【C语言程序设计——函数】素数判定(头歌实践教学平台习题)【合集】
本内容介绍了编写一个判断素数的子函数的任务,涵盖循环控制与跳转语句、算术运算符(%)、以及素数的概念。任务要求在主函数中输入整数并输出是否为素数的信息。相关知识包括 `for` 和 `while` 循环、`break` 和 `continue` 语句、取余运算符 `%` 的使用及素数定义、分布规律和应用场景。编程要求根据提示补充代码,测试说明提供了输入输出示例,最后给出通关代码和测试结果。 任务核心:编写判断素数的子函数并在主函数中调用,涉及循环结构和条件判断。
853 23