数据结构上机作业(三)

本文介绍了使用Java实现串的模式匹配算法,包括Brute-Force和KMP两种方法,并详细讲解了如何在三元组表压缩存储的稀疏矩阵基础上进行转置操作,包括普通转置和快速转置,重点讨论了矩阵中每一行非零元素的个数计算和转置后元素的存储位置计算。
摘要由CSDN通过智能技术生成

一.上机内容

1、实现串的模式匹配算法。
2、实现稀疏矩阵的三元组表压缩存储方法
3 *、三元组表压缩存储方法基础上实现稀疏矩阵的转置,主要涉及矩阵中每一行非零元素的个数的求解方法和转置矩阵的每一行中的第一个元素在其三元组表中的存储位置的计算。

二.代码实现

1、实现串的模式匹配算法。
1.Brute-Force模式匹配算法
/**
     * 返回模式串t在主串从start开始的第一次匹配位置,匹配失败返回-1
     * @param t
     * @param start
     * @return
     */
    public int indexof_BF(IString t,int start){
        // 当主串比模式串长的时候才开始比较
        if (this!=null && t!= null && t.length() > 0 && this.length > t.length()) {
            // i表示主串中某个子串的序号
            int slen, tlen, i = start, j = 0;
            // 主串的长度
            slen = this.length();
            // 匹配串的长度
            tlen = t.length();
            // i和j都在比较范围内
            while ((i < slen) && (j < tlen)) {
                // j为模式串当前字符的下标
                if (this.charAt(i) == t.charAt(j)) {
                    i++;
                    j++;
                } else {
                    // 继续比较主串中的下一个子串
                    // i-j+1就是在本次比较前,往后走一个字符
                    i = i - j + 1;
                    // 模式串下标退回0
                    j = 0;
                }
            }
            // 如果匹配成功,返回子串序号
            // 运行到这里,不一定是j超出了范围,还有可能是i超出了范围,i超出范围但是j没有,说明匹配失败
            if (j >= t.length()) {
                return i - tlen;
            } else {
                return -1;
            }
        }
        return -1;
    }
2.KMP模式匹配算法
/**
     * next[j]函数
     * @param T
     * @return      返回模式串的next[j]数组
     */
    public int[] getNext(IString T){
        // next[]数组
        int[] next = new int[T.length()];
        //主串指针
        int j = 1;
        // 模式串指针
        int k = 0;
        next[0] = -1;
        next[1] = 0;
        while (j < T.length() -1){
            // 匹配
            if (T.charAt(j) == T.charAt(k)){
                next[j + 1] = k + 1;
                j++;
                k++;
            }else if (k == 0){
                next[j + 1] = 0;
                j++;
            }else {
                k = next[k];
            }
        }
        return next;
    }

    /**
     * 计算模式串T的nextVal[]
     * @param T
     * @return
     */
    private int[] getNextVal(IString T){
        // nextVal数组
        int[] nextVal = new int[T.length()];
        int j = 0;
        int k = -1;
        nextVal[0] = -1;
        while(j < T.length() - 1){
            if (k == -1 || T.charAt(j) == T.charAt(k)){
                j++;
                k++;
                if (T.charAt(j) != T.charAt(k)){
                    nextVal[j] = k;
                }else {
                    nextVal[j] = nextVal[k];
                }
            }else {
                k = nextVal[k];
            }
        }
        return nextVal;
    }

    /**
     * 模式匹配KMP算法
     * @param T
     * @param start
     * @return
     */
    public int index_KMP(IString T,int start){
        int[] next = getNext(T);
        int i = start;
        int j = 0;
        // 对两串从左到右逐个比较字符
        while (i < this.length && j <T.length()){
            // 若对应字符匹配
            if (j == -1 || this.charAt(i) == T.charAt(j)){      // j==-1 表示 S[i]!=T[0],那么i和j都要后移一位,这样j就变成第一位了
                i++;
                j++;
            }else {
                // 模式串后移
                j = next[j];
            }
        }
        if (j < T.length()){
            // 匹配失败
            return -1;
        }else {
            // 匹配成功
            return i - T.length();
        }
    }
2、实现稀疏矩阵的三元组表压缩存储方法
/**
     * 初始化三元组顺序表
     * @param mat   一个普通的二元数组
     */
    public SparseMatrix(int mat[][]){
        int i,j,k = 0,count =0;
        // 行数
        rows = mat.length;
        // 列数
        cols = mat[0].length;
        // 统计非零元素个数
        for(i = 0;i <mat.length;i++){
            for ( j = 0; j < mat[i].length; j++) {
                if (mat[i][j] != 0){
                    count++;
                }
            }
        }
        //非零元素的个数
        nums = count;
        // 申请三元节点空间
        data = new TripleNode[nums];
        for (i = 0;i<mat.length;i++){
            for (j = 0; j < mat[i].length;j++){
                if (mat[i][j] != 0){
                    // 建立三元组
                    data[k] = new TripleNode(i,j,mat[i][j]);
                    // 计数器
                    k++;
                }
            }
        }

    }
3 *、三元组表压缩存储方法基础上实现稀疏矩阵的转置,主要涉及矩阵中每一行非零元素的个数的求解方法和转置矩阵的每一行中的第一个元素在其三元组表中的存储位置的计算。
1.转置
/**
     * 矩阵转置
     * @return
     */
    public SparseMatrix transpose(){
        // 创建转置矩阵对象
        SparseMatrix tm = new SparseMatrix(nums);
        // 行数改为列数,列数改行数,非零个数不变
        tm.rows = cols;
        tm.cols = rows;
        tm.nums = nums;
        int q = 0;
        for (int col = 0 ; col < cols;col++){
            for (int p = 0; p < nums;p++){
                // 外层循环是按列的大小,所以这样是按列从小到大来寻找非零元素
                // 遍历三元组这个一维数组
                if (data[p].column == col){
                    // 转置
                    tm.data[q].row = data[p].column;
                    tm.data[q].column = data[p].row;
                    tm.data[q].value = data[p].value;
                    // 计数器
                    q++;
                }
            }
        }
        return tm;
    }
2.快速转置
/**
     * 矩阵快速转置算法
     * @return
     */
    public SparseMatrix fastTranspose(){
        // 创建矩阵对象
        SparseMatrix tm = new SparseMatrix(nums);
        // 行数变为列数
        tm.cols = rows;
        // 列数变为行数
        tm.rows = cols;
        // 非零元素个数不变
        tm.nums = nums;
        int i,j = 0,k = 0;
        int[] num,cpot;
        if (nums > 0){
            num = new int[cols];
            cpot = new int[cols];
            // 每列非零元素个数数组num初始化
            for (i = 0; i < cols; i++) {
                num[i] = 0;
            }
            // 计算每列非零元素个数
            for (i = 0; i <nums;i++){
                j = data[i].column;
                num[j]++;
            }
            cpot[0] = 0;
            for (i = 1; i < cols;i++){
                cpot[i] = cpot[i-1] + num[i+1];
            }
            // 执行转置操作
            for (i = 0; i < nums;i++){
                // 扫描整个三元组顺序表
                j = data[i].column;
                // 该元素在tm中的位置
                k = cpot[j];
                // 转置
                tm.data[k].row = data[i].column;
                tm.data[k].column = data[i].row;
                tm.data[k].value = data[i].value;
                // 该列下一个非零元的存放位置
                cpot[j]++;
            }
        }
        return tm;
    }
实验1: 1)熟悉Vc 6.0环境 2)用两种算法实现1-1/x+1/x*x-1/x*x*x+1/x*x*x*x…., 注(algo1-1,algo1-2) 实验2:线性表 1) 顺序表的合并:实现书中P26中算法2.7,La=1 2 3 4 5, Lb=2 4 6 8 10。要求得到合并后的Lc=1 2 3 4 5 6 8 10 注(algo2-4 ) 2) 一元多项式加法:Pa=Pa+Pb,并销毁一元多项式Pb 3) 一元多项式乘法:Pa=Pa*Pb,并销毁一元多项式Pb 实验3:栈和队列 1) 利用栈求解迷宫问题;行列各为10(包括外墙),迷宫内墙单元数为18。 2) 利用栈和递归求解Hanoi塔问题,圆盘数3。 3) 利用非循环顺序队列采用广度搜索法求解迷宫问题(一条路径);行列各为5(包括外墙),迷宫内墙单元数为2。 实验4: 1) 模式匹配改进算法:KMP算法实现书中4.6,4.7,4.8算法。 实验5: 数组和广义表 1) 求稀疏矩阵的加,乘和转置矩阵。 2) 求广义表的深度。 实验6:树和二叉树 1) 求赫夫曼编码。(w存放n个字符的权值(均>0),构造赫夫曼树HT,并求出n个字符的赫夫曼编码HC) 实验7:图 1)实现教科书中图7.33的程序,并能求出任意两点的最短路径。 实验8:动态存储管理 1) 边界标识法,程序实现教科书中算法8.1 2) 伙伴系统,程序实现教科书中算法8.2 实验9:查找 1) 哈希表的查找及其分析:以书中图9-25为例。 实验10:内部排序 1) 以书中10-4的数据,程序实现直接插入排序,折半排序和2路插入排序。 2) 以书中图10.6的数据,程序实现快速排序。 3) 以书中图10.9的数据,程序实现树形选择排序。 4) 以书中图10.13的数据,程序实现归并排序。 5) 以书中图10.14的数据,程序实现链式基数排序。 实验11: 外部排序 1) 以书中图11.4的数据,程序实现多路平衡归并排序。 2)以书中图11.5的数据,程序实现置换-选择排序。 实验12:文件 1)以书中图12.4的数据,程序实现顺序文件。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值