利用哈夫曼树实现文件压缩

实现原理:

1.按照字符分析要压缩的文件得出结果(有哪些字符,每个字符出现的次数)。
2.根据字符出现的次数构建哈夫曼树(得出字符的哈夫曼编码)。
3.根据字符的哈夫曼编码进行转换、压缩,然后创建压缩文件。
4.读取压缩文件,读出哈夫曼编码和字符的对照表。解压缩。

数据结构的设计:
1.保存字符次数和字符的数据结构
struct _symbol{
char character;//字符
unsigned int number;//字符出现的次数
char huffecode[20];//编码
}

2.用一个结构体保存所有字符的信息
struct _filestate{
char symbol_count;//字符种类
struct _symbol symbol_array[128];//字符信息
};
3.哈夫曼树节点
struct node{
struct _symbol symbol;
struct node *left;
struct node *right;
};

4.哈夫曼编码
struct code{
char character;
char codebit[100];
};

源代码实现:
#include "header.h"
#include 

//单个字符的结构体
struct  _symbol{
	char character;//字符
	unsigned int number;//字符的次数
    	char huffcode[20]; //编码
};
//总结构体
struct _filestate{
    char symbol_count;//多少种字符
    struct _symbol symbol_array[128]; //结构体数组保存各种字符
};
//文件字符总结构体
struct _filestate filestate;

//树节点
struct _node{
    struct _node *parent;
    struct _symbol symbol;
    struct _node *left;
    struct _node *right;
    int idx; //在filestate中的索引
};

typedef struct _node node;


//创建节点
node *creatnode(struct _symbol symbol,int idx)
{
    
    node *temp=malloc(sizeof(node));
    temp->symbol=symbol;
    temp->left=NULL;
    temp->right=NULL;
    temp->idx=idx;
    temp->parent=NULL;
    return temp;
}


//构建一棵哈弗曼树
node *createhaffman(struct _filestate *pfilestate)
{
    //创建一维数组,里面保存指针
    node **array=malloc(pfilestate->symbol_count*sizeof(node *));
    node *temp=NULL;
    int i;
    for(i=0;isymbol_count;i++)
    {
        array[i]=creatnode(pfilestate->symbol_array[i],i);
    }

    int j;
    int minidx,secminidx;
    for(i=0;isymbol_count-1;i++)
    {
        j=0;
        //找出第一个非空节点
        while(array[j]==NULL)
        {
            j++;
        }
        minidx=j;

        //拿第一个非空节点和后面的对比
        for(j=0;jsymbol_count;j++)
        {
            if(array[j]&&array[j]->symbol.numbersymbol.number)
            {
                minidx=j;
            }

        }

        j=0;
        while(array[j]==NULL||j==minidx)
        {
            j++;
        }
        secminidx=j;
        for(j=0;jsymbol_count;j++)
        {
            if(array[j]&&j!=minidx&&array[j]->symbol.numbersymbol.number)
            {
              secminidx=j;  
            }
        }

        struct _symbol symbol={0,array[minidx]->symbol.number+array[secminidx]->symbol.number};
        temp=creatnode(symbol,-1);        
        temp->left=array[minidx];
        temp->right=array[secminidx];
        temp->parent=NULL;

        array[minidx]->parent=temp;
        array[secminidx]->parent=temp;
           
        array[minidx]=temp;
        array[secminidx]=NULL;
    }

    free(array);
    array=NULL;

    return temp;
}

//判断是否叶子节点
int isleaf(node *cur)
{
    if(cur->left==NULL&&cur->right==NULL)
    return 1;
    else
    {
        return 0;
    }
}

//取得哈夫曼编码
void huffmancode(node *leaf,char code[20])
{
    char a[20]={0};
    node *temp=leaf;
    int i=0;
    while(temp->parent)
    {
        if(temp->parent->left==temp)
        {
            a[i]=2;
        }
        else
        {
            a[i]=1;
        }
        temp=temp->parent;
        i++;
    }
    int length=i;
    char tmp=0;
    for(i=0;iidx;
            huffmancode(root,pfilestate->symbol_array[idx].huffcode);
        }
        else
        {
            fillhuffmancode(root->left,pfilestate);
            fillhuffmancode(root->right,pfilestate);
        }

    }


}


//根据频率创建树且获取编码
void createandcoding(struct _filestate *pfilestate)
{
    node *root=createhaffman(pfilestate);
    fillhuffmancode(root,pfilestate);
    //haffmancode(root,0,&dictionary);
    
     int i,j;
    for(i=0;isymbol_count;i++)
    {
        printf("%c ",pfilestate->symbol_array[i].character);
        for(j=0;j<20;j++)
        printf("%d",pfilestate->symbol_array[i].huffcode[j]);
        puts("\n");
    }
}






//创建结构体
void calccount(struct _filestate *pfilestate,char ch)
{
    int i;
    int flagexist=0;
    for(i=0;isymbol_count;i++)
    {
        if(ch==pfilestate->symbol_array[i].character)
        {
            pfilestate->symbol_array[i].number++;
            flagexist=1;
            break;
        }
    }
    if(!flagexist)
    {
        pfilestate->symbol_array[pfilestate->symbol_count].character=ch;
        pfilestate->symbol_count++;
    }

}

//查字典,获得编码
void lookupdict(char character,struct _filestate *pfilestate,char bitcode[])
{
    int i=0;
    for(i=0;isymbol_count;i++)
    {
        if(character==pfilestate->symbol_array[i].character)
        {
            strcpy(bitcode,pfilestate->symbol_array[i].huffcode);
            break;
        }
    }
}


//写文件头
void writeheader(int fd,struct _filestate *pfilestate)
{
    write(fd,pfilestate,sizeof(struct _filestate));
}

//读文件头
void readheader(int fd,struct _filestate *pfilestate)
{
    read(fd,pfilestate,sizeof(struct _filestate));
}

//写编码
void writecode(int fd_r,int fd_w,struct _filestate *pfilestate)
{
    int flgend=0;
    //临时变量,用来保存取出来哈夫曼编码
    char bitcode[20];
    //临时变量
    char temp;
    char buffer;
    //写字节的索引
    int buf_idx=0;
    //读到文件尾
    int bitcode_idx=0;
    //从源文件读一个字符
    read(fd_r,&temp,1);
    //从字典里查出字符的哈夫曼编码
    lookupdict(temp,pfilestate,bitcode);
   
   //一直读字符,直到文件的末尾
    while(!flgend)
    {
    //用来写的字符,需要根据哈夫曼编码构建
    buffer=0;
    //现在指向buffer的第0位
    buf_idx=0;
    while(buf_idx<8)
    {
        if(bitcode[bitcode_idx]==2)
        {
            //把对应的位清零
            buffer&=~(1<right;
            }
            else
            {
                tmpnode=tmpnode->left;
            }
            if(isleaf(tmpnode))
            {
                write(fd_w,&(tmpnode->symbol.character),1);
                tmpnode=root;
            }
            bit_idx++;
        }


    }
    close(fd_w);
}


//解压
void decompress(char *filename)
{
    
    int fd=open(filename,O_RDWR);
    if(fd<0)
    {
        perror("open");
        return;
    }
    memset(&filestate,0,sizeof(filestate));
    //读文件头
    readheader(fd,&filestate);


    //创建哈弗曼树
    node *root=NULL;
    root=createhaffman(&filestate);

    writedecompressfile(fd,filename,root);

    close(fd);


}

int main(int argc,char const *argv[])
{
    char *filename=(char *)argv[2];
    if(argc<3)
    {
        puts("用法:tar -y[-j] filename");
        exit(EXIT_FAILURE);
    }
    if(strcmp(argv[1],"-y")==0)
    {
        compress(filename);
    }
    else if(strcmp(argv[1],"-j")==0)
    {
        decompress(filename);
    }
    else
    {
        puts("参数错误!");
        exit(EXIT_FAILURE);
    }

}
这样的压缩可以做到压缩50%的效果。

 
   









你可能感兴趣的:(数据结构)