文件目录管理与显示
给出目录和文件信息,编程实现将其排列成一棵有一定缩进的树。
要求:
(1)设计文件和目录信息树的存储结构。
(2)从文件或键盘输入目录和文件信息,输入格式采用绝对路径法,即:
\A
\A\AA1
\A\AA1\aa1.doc

创建时要检查同一路径下不能有同名的目录或文件名。
(3)设计文件和目录信息树的输出格式(以凹入表的形式显示)。
(4)查找指定目录和文件。
(5)添加新目录或新文件。
(6)删除指定目录或文件,子目录能够被删除的前提是其为空,既不包含任何子目录和文件;根目录不能删除。
(7)扩充目录或文件信息,如创建时间、读写权限、文件长度或子目录包含的子目录和文件数等。
(8)对同一层次下的子目录或文件按创建时间有序输出。
(9)通配符的使用。如用“?”代表任意一个字符,用“*”表示任意多个任意字符。
扩展内容:
实现相对路径表示法。

#include <stdio.h>  
#include <malloc.h>  
#include <string>  
#include <iostream>  
#include <sstream>  

using namespace std;  

#include "Sibtree.h"  

void add(SibTree *t1)  
{  
    printf("输入目录文件信息以输入0结束:\n");  
    SibTreeNode *r, *r1, *r2;  
    char ifm[1000] = { '\0' }, name1[50], name2[50];  
    char *str;  
    int i, j, n;  
    while (1)  
    {  
        cin >> ifm;  
        str = ifm;  
        if (*str == '0')break;  
        r = t1->root;  
        if (r == NULL)  
        {  
            for (i = 0, str += 1; *str != '\\'&&*str != '\0'; str++, i++)  
            {  
                name1[i] = *str;  
            }  
            name1[i] = '\0';  
            if (*str == '\0')insertRoot(t1, name1);  
            else  
                printf("%s的路径不存在,不能添加\n", name1);  
        }  
        else  
        {  
            for (i = 0, str += 1; *str != '\\'&&*str != '\0'; str++, i++)  
            {  
                name1[i] = *str;  
            }  
            name1[i] = '\0';  
            if (search(t1, name1) == 0 && *str == '\0')  
            {  
                insertnextSibing(t1, r, name1);  
            }  
            if (search(t1, name1) == 0 && *str != '\0')printf("%s的路径不存在,不能添加\n", name1);  
            r1 = LevelOrderTraverse(t1->root, name1);  
            while (r1 != NULL)  
            {  
                if (*str == '\0')break;  
                for (i = 0, str += 1; *str != '\\'&&*str != '\0'; str++, i++)  
                {  
                    name2[i] = *str;  
                }  
                name2[i] = '\0';  
                if (*str == '\0')  
                {  
                    if (insertChild(t1, r1, name2)){ break; }  
                }  
                else  
                {  
                    n = children(r1);  
                    for (j = 1; j <= n; j++)  
                    {  
                        if (strcmp(child(r1, j)->data, name2) == 0)break;  
                    }  
                    if (j <= n)r2 = child(r1, j);  
                    else r2 = NULL;  
                    if (r2 == NULL)printf("%s的路径不存在,不能添加\n", name2);  
                }  
                n = children(r1);  
                for (j = 1; j <= n; j++)  
                {  
                    if (strcmp(child(r1, j)->data, name2) == 0)break;  
                }  
                if (j <= n)r1 = child(r1, j);  
                else r1 = NULL;  
            }  
        }  
    }  
}  
void check(SibTree *t)  
{  
    SibTreeNode *r;  
    printf("输入要查找的文件名:");  
    char name[50];  
    cin >> name;  
    r = mohu1LevelOrderTraverse(t->root, name);  
    if (r == NULL)
        //printf("没有此文件\n");
        printf("\n");
    else  
    {  
        printf("此文件找到,创建时间为:");  
        printf("%d/%d/%d %d:%d:%d", r->year, r->month, r->day, r->hour, r->minute, r->second);  
        printf("\t子目录个数为:");  
        printf("%d\n", children(r));  
    }  
} 

void Delete(SibTree *t)  
{  
    SibTreeNode *r, *p, *kid;  
    printf("输入要删除的文件名:");  
    char name[50];  
    cin >> name;  
    r = LevelOrderTraverse(t->root, name);  
    if (r == NULL)printf("没有此文件!\n");  
    else  
    {  
        if (r->firstChild != NULL)printf("有子文件不能删除!\n");  
        else  
        {  
            p = r->parent;  
            kid = p->firstChild;  
            if (strcmp(kid->data, r->data) == 0)p->firstChild = kid->nextSibling;  
            else  
            {  
                while (strcmp(kid->nextSibling->data, r->data))kid = kid->nextSibling;  
                kid->nextSibling = kid->nextSibling->nextSibling;  
            }  
            printf("已删除!\n");  
        }  
    }  
}  
void out(SibTree *t)  
{  
    printf("输出文件目录的信息:\n");  
    printf("root:\n");  
    printf("%s", toString(t).c_str());  
}  
void repair(SibTree *t)//修改扩充  
{  
    SibTreeNode *r, *p, *kid;  
    printf("输入想要扩充或修改的文件或目录的名称\n");  
    char name[100];  
    cin >> name;  
    r = LevelOrderTraverse(t->root, name);  
    if (r == NULL)  
        printf("没有此文件!\n");  
    else  
    {  
        printf("   查找成功\n\n   1 修改or 2扩充?\n\n\n");  
        int k;  
        cin >> k;  
        if (k == 1)  
        {  
            printf("输入目录或文件的新名称 \n");  
            char Name[100];  
            cin >> Name;  
            strcpy(r->data, Name);  
            printf(" 修改成功!\n\n");  
            printf(" 修改时间为:");  
            printf("%d年/%d月/%d%d时:%d分:%d秒\n\n\n", sys.wYear, sys.wMonth, sys.wDay, sys.wHour, sys.wMinute, sys.wSecond);  
            r->year = sys.wYear;  
            r->month = sys.wMonth;  
            r->day = sys.wDay;  
            r->hour = sys.wHour;  
            r->minute = sys.wMinute;  
            r->second = sys.wSecond;  
        }  
        else  
        {  
            printf("当前目录的信息为:\n 限制子目录的个数 %d 个 限制子文件的个数 %d 个\n", r->sonmulu, r->sonfile);  
            printf("请输入修改后的限制子目录的个数和限制子文件的个数:\n");  
            int a, b;  
            cin >> a >> b;  
            r->sonmulu = a;  
            r->sonfile = b;  
            printf("修改成功!\n\n\n");  
        }  
    }  
}  

void mohucheck(SibTree *t)  
{  
    SibTreeNode *r;  
    printf("输入要查找的文件或目录名:");  
    char name[50];  
    cin >> name;  
    r = mohuLevelOrderTraverse(t->root, name);  
    if (r == NULL) //printf("没有此文件\n");  
        printf("\n");  
    else  
    {  
        printf("此文件或目录找到,创建时间为:");  
        printf("%d/%d/%d %d:%d:%d", r->year, r->month, r->day, r->hour, r->minute, r->second);  
        printf("\t子目录个数为:");  
        printf("%d\n", children(r));  
    }  
}  

void main()  
{  
    SibTree t1;  
    int k;  
    TreeInitiate(&t1);  
    while (1)  
    {  
        printf("\n");  
        printf("\t                      文件管理                        \t\n");  
        printf("\t********************************************************\t\n");  
        printf("\t  *************对照号码输入你要进行的操作*************\n");  
        printf("\t                     1 ——>添加目录或文件      \n");  
        printf("\t                     2 ——>查找目录或文件      \n");  
        printf("\t                     3 ——>删除目录或文件      \n");  
        printf("\t                     4 ——>输出文件信息        \n");  
        printf("\t                     5 ——>扩充目录或文件信息   \n");  
        printf("\t                     6 ——>模糊查找            \n");  
        printf("\t                     7 ——>退出                \n");  
        printf("输入你想进行的操作号码:");  
        scanf("%d", &k);  
        if (k == 7)break;  
        switch (k)  
        {  
        case 1: add(&t1); break;  
        case 2: check(&t1); break;  
        case 3: Delete(&t1); break;  
        case 4: out(&t1); break;  
        case 5: repair(&t1); break;  
        case 6: mohucheck(&t1); break;  
        default:  
            printf("输入号码错误!\n");  
        }  
    }  
}  
#include <windows.h>  
#include <queue>
SYSTEMTIME sys;  
int ok;  
typedef struct SibTreeNode  
{  
    char data[50];  
    int valid;  
    int year;  
    int month;  
    int day;  
    int hour;  
    int minute;  
    int second;  
    int sonmulu;  
    int sonfile;  
    struct SibTreeNode * parent;  
    struct SibTreeNode *firstChild;  
    struct SibTreeNode *nextSibling;  
}SibTreeNode;  

typedef struct  
{  
    SibTreeNode *root;  
    int size;  
}SibTree;  

typedef SibTreeNode *QueueDataType;  
#include "SeqCQueue.h"  

int IsTreeEmpty(SibTree *t)  
{  
    return t->size == 0;  
}  

int isValidNode(SibTreeNode *p)  
{  
    if (p)  
        return p->valid;  
    else return 0;  
}  
//创建一个空树  
void  TreeInitiate(SibTree *t)  
{  
    t->root = NULL;  
    t->size = 0;  
}  

//创建结点  
SibTreeNode * newSibTreeNode(char x[50])  
{  
    GetLocalTime(&sys);  
    SibTreeNode *p = (SibTreeNode *)malloc(sizeof(SibTreeNode));  
    strcpy(p->data, x);  
    p->valid = true;  
    p->year = sys.wYear;  
    p->month = sys.wMonth;  
    p->day = sys.wDay;  
    p->hour = sys.wHour;  
    p->minute = sys.wMinute;  
    p->second = sys.wSecond;  
    p->sonmulu = 5;    //扩充的信息,限制子目录的个数  
    p->sonfile = 2;     //限制子文件的个数  
    p->parent = NULL;  
    p->firstChild = NULL;  
    p->nextSibling = NULL;  
    return  p;  
}  

//创建包含一个结点的树  
void onenodeSibTree(SibTree *t, char x[50])  
{  
    t->root = newSibTreeNode(x);  
    strcpy(t->root->data, x);  
    t->size = 1;  
}  

//输出当前结点的父亲节点  
SibTreeNode *root(SibTreeNode *p)  
{  
    if (p == NULL)  
    {  
        p = (SibTreeNode*)malloc(sizeof(SibTreeNode));  
        return p;  
    }  
    else  
    {  
        return p;  
    }  
}  
//插入根节点  
void insertRoot(SibTree *t, char x[50])  
{  
    SibTreeNode * newRoot = newSibTreeNode(x);  
    newRoot->firstChild = t->root;  
    if (t->root != NULL)  
    {  
        t->root->parent = newRoot;  
    }  
    t->root = newRoot;  
    t->size++;  
}  

// 输出树的大小  
int size(SibTree *t)  
{  
    return t->size;  
}  


// 构造一个不合法的结点  
SibTreeNode * invalidSibTreeNode()  
{  
    SibTreeNode *p = (SibTreeNode *)malloc(sizeof(SibTreeNode));  
    p->valid = false;  
    return  p;  
}  

//当前结点的孩子的个数  
int children(SibTreeNode *p)  
{  
    int count = 0;  
    SibTreeNode *countNode;  
    if (isValidNode(p))  
    {  
        countNode = p->firstChild;  
        while (countNode != NULL)  
        {  
            count++;  
            countNode = countNode->nextSibling;  
        }  
        return count;  
    }  
    else  
    {  
        return 0;  
    }  
}  

//返回当前结点的cth孩子结点  
SibTreeNode *child(SibTreeNode *p, int c)  
{  
    if (isValidNode(p))  
    {  
        if (c < 1)  
        {  
            return invalidSibTreeNode();  
        }  
        SibTreeNode *kid = p->firstChild;  
        while ((kid != NULL) && (c > 1))  
        {  
            kid = kid->nextSibling;  
            c--;  
        }  
        if (kid == NULL)  
        {  
            return invalidSibTreeNode();  
        }  
        else  
        {  
            return kid;  
        }  
    }  
    else  
        throw "the node is not a valid node!";  

}  

//返回当前结点的兄弟结点  
SibTreeNode *nextSibling(SibTreeNode *p)  
{  
    if (isValidNode(p))  
    {  
        if (p->nextSibling == NULL)  
        {  
            return invalidSibTreeNode();  
        }  
        else  
        {  
            return p->nextSibling;  
        }  
    }  
    else  
    {  
        throw "the node is not a valid node!";  
        return invalidSibTreeNode();  
    }  
}  

//插入孩子结点  
int insertChild(SibTree *t, SibTreeNode *p, char x[50])  
{  
    SibTreeNode *n;  
    int i, flog = 0;  
    for (i = 0; x[i] != '\0'; i++)  
    {  
        if (x[i] == '.'){ flog = 1; break; }  
    }  

    if (isValidNode(p))  
    {  
        n = p->firstChild;  
        if (n == NULL)  
        {  
            p->firstChild = newSibTreeNode(x);  
            p->firstChild->parent = p;  
        }  
        else  
        {  
            while (n != NULL)  
            {  
                if (strcmp(n->data, x) == 0){ printf("输入的文件有重名,请更改名字!"); break; }  
                if (n->nextSibling == NULL)break;  
                n = n->nextSibling;  
            }  
            if (strcmp(n->data, x) != 0)  
            {  
                if (flog == 1 && children(n->parent)<n->parent->sonfile)  
                {  
                    n->nextSibling = newSibTreeNode(x); n->nextSibling->parent = p;  
                }  
                else  
                    if (flog == 1)printf("文件数超出,不能继续存储\n");  
                    else  
                        if (flog == 0 && children(n->parent)<n->parent->sonmulu)  { n->nextSibling = newSibTreeNode(x); n->nextSibling->parent = p; }  
                        else if (flog == 0)printf("子目录超出,不能继续存储\n");  

            }  
        }  
        t->size++;  
        return 1;  
    }  
    else return 0;  
}  
//为当前结点插入兄弟结点  
int insertnextSibing(SibTree *t, SibTreeNode *p, char x[50])  
{  
    SibTreeNode *n;  
    if (isValidNode(p))  
    {  
        n = p->nextSibling;  
        if (n == NULL)p->nextSibling = newSibTreeNode(x);  
        else  
        {  
            while (n->nextSibling != NULL)n = n->nextSibling;  
            n->nextSibling = newSibTreeNode(x);  
            n->nextSibling->parent = p->parent;  
        }  
        t->size++;  
        return 1;  
    }  
    else return 0;  
}  

int mohucheck(char a[], char b[])  
{  
    int len = strlen(b);  
    int len1 = strlen(a);  
    for (int i = 0; i < len; i++)  
    {  
        if (b[i] == '*')  
        {  
            int j = len - 1;  
            int k = len1 - 1;  

            while (j != i)  
            {  
                if (b[j] != a[k])  
                    return 0;  
                --j;   
                --k;  
            }  
            return 1;  
        }  
        else if (b[i] != '?')  
        {  
            if (a[i] != b[i])  
                return 0;  
        }  
    }  
    return 1;  
}  

//模糊查找  
SibTreeNode *mohuLevelOrderTraverse(SibTreeNode *t, char a[50])  
{  
    int ok = 0;  
    SeqCQueue Q;  
    SibTreeNode *x;  
    if (t == NULL)return NULL;  
    QueueInitiate(&Q);  
    QueueAppend(&Q, t);  
    x = t->nextSibling;  
    while (x != NULL)  
    {  
        QueueAppend(&Q, x);  
        x = x->nextSibling;  
    }  
    while (QueueNotEmpty(Q))  
    {  
        QueueDelete(&Q, &x);  
        if (mohucheck(x->data, a) == 1)  
        {  
            ok = 1;
            printf("查找到符合条件的有");//%s\n",x->data);
            
            SibTreeNode *xx;
            xx = x;



            char sss[100][100];
            char gan = '\\';
            int len = 0;
            while (xx!=root(t) )//&& xx->parent != NULL)
            {
                strcpy(sss[len++],xx->data);
                //printf("\%s   ",xx->data);
                xx=xx->parent;
            }
            if (strcmp(sss[len-1],"\\")!=0)
                printf("%ca",gan);
            for(int i=len-1;i>=0;i--)
            {
                printf("%c",gan);
                printf("%s",sss[i]);
            }
            printf("\n");
           ///
        }  
        x = x->firstChild;  
        while (x != NULL)  
        {  
            QueueAppend(&Q, x);  
            x = x->nextSibling;  
        }  
    }  
    if (!ok) printf("不存在此文件或目录\n");  
    return x;  
}  

SibTreeNode *mohu1LevelOrderTraverse(SibTreeNode *t, char a[50])  
{  
    int ok = 0;  
    SeqCQueue Q;  
    SibTreeNode *x;  
    if (t == NULL)return NULL;  
    QueueInitiate(&Q);  
    QueueAppend(&Q, t);  
    x = t->nextSibling;  
    while (x != NULL)  
    {  
        QueueAppend(&Q, x);  
        x = x->nextSibling;  
    }  
    while (QueueNotEmpty(Q))  
    {  
        QueueDelete(&Q, &x);  
        if (strcmp(x->data, a) == 0)  
        {  
            ok = 1;
            printf("查找到符合条件的有");//%s\n",x->data);
            
            SibTreeNode *xx;
            xx = x;



            char sss[100][100];
            char gan = '\\';
            int len = 0;
            while (xx!=root(t) )//&& xx->parent != NULL)
            {
                strcpy(sss[len++],xx->data);
                //printf("\%s   ",xx->data);
                xx=xx->parent;
            }
            if (strcmp(sss[len-1],"\\")!=0)
                printf("%ca",gan);
            for(int i=len-1;i>=0;i--)
            {
                printf("%c",gan);
                printf("%s",sss[i]);
            }
            printf("\n");
            ///
        }  
        x = x->firstChild;  
        while (x != NULL)  
        {  
            QueueAppend(&Q, x);  
            x = x->nextSibling;  
        }  
    }  
    if (!ok) printf("不存在此文件或目录\n");  
    return x;  
}  

//层序遍历  
SibTreeNode *LevelOrderTraverse(SibTreeNode *t, char a[50])  
{  
    SeqCQueue Q;  
    SibTreeNode *x;  
    if (t == NULL)return NULL;  
    QueueInitiate(&Q);  
    QueueAppend(&Q, t);  
    x = t->nextSibling;  
    while (x != NULL)  
    {  
        QueueAppend(&Q, x);  
        x = x->nextSibling;  
    }  
    while (QueueNotEmpty(Q))  
    {  
        QueueDelete(&Q, &x);  
        if (strcmp(x->data, a) == 0)break;  
        x = x->firstChild;  
        while (x != NULL)  
        {  
            QueueAppend(&Q, x);  
            x = x->nextSibling;  
        }  
    }  
    return x;  
}  

//查找是否存在  
int search(SibTree *t, char x[50])  
{  
    if (LevelOrderTraverse(t->root, x) == NULL)return 0;  
    else return 1;  
}  

string preorderString(SibTreeNode *currentNode, int depth)  
{  
    string s = "    ";  
    stringstream out;  
    if (currentNode == NULL)  
    {  
        return "";  
    }  

    for (int i = 0; i < depth; i++)  
    {  
        s = s + "   ";  
    }  
    out << currentNode->data;  
    s += out.str();  
    s += "\n";  
    s += preorderString(currentNode->firstChild, depth + 1);  
    s += preorderString(currentNode->nextSibling, depth);  
    return s;  
}  

string toString(SibTree *t)  
{  
    return preorderString(t->root, 0);  
}  
typedef  struct  
{  
    QueueDataType  queue[50];  
    int front;  
    int rear;  
    int count; /* 队列的当前表长 */  
} SeqCQueue;  

//初始化  
void QueueInitiate(SeqCQueue *Q)  
{  
    Q->front = 0;  
    Q->rear = 0;  
    Q->count = 0;  
}  

//判断队列非空  
int QueueNotEmpty(SeqCQueue Q)  
{  
    if (Q.count != 0) return 1;  
    else return 0;  
}  

//入队列  
int QueueAppend(SeqCQueue *Q, QueueDataType x)  
{  
    if (Q->count>0 && Q->rear == Q->front)  
    {  
        printf("队列已满无法插入!\n");  
        return 0;  
    }  
    else  
    {  
        Q->queue[Q->rear] = x;  
        Q->rear = (Q->rear + 1) % 50;  
        Q->count++;  
        return 1;  
    }  
}  
//出队列  
int QueueDelete(SeqCQueue *Q, QueueDataType *x)  
{  
    if (Q->count == 0)  
    {  
        printf("队列已空无数据元素出队列!\n");  
        return 0;  
    }  
    else  
    {  
        *x = Q->queue[Q->front];  
        Q->front = (Q->front + 1) % 50;  
        Q->count--;  
        return 1;  
    }  
}  
//取对头数据元素  
int QueueGet(SeqCQueue *Q, QueueDataType *x)  
{  
    if (Q->count == 0)  
    {  
        printf("队列已空无数据元素出队列!\n");  
        return 0;  
    }  
    else  
    {  
        *x = Q->queue[Q->front];  
        return 1;  
    }  
}  

测试数据:

1  
\a  
\a\aad
\a\abcde
\a\abcde\fghij
\a\abcde\fghij\aaaaabbbbb.doc
\a\aad\aa
\a\aad\aa\aaa  
\a\aad\aa\aaa\aaaaaa.doc 
0  
4  
2  
aa  
3  
aaaaaa.doc
4  
5  
aaa  
1  
asd  
4  
5  
asd  
2  
12 3  
6  
a?d  
4  
1  
\a\aad\aa\asd\aaaaabbbbb.doc
0  
4  
6  
aa*bb.doc
6  
aa*cc.doc
4  
6  
a*