矩阵化为最简型矩阵JAVA语言实现――线性代数

createh53周前 (12-18)技术教程20

前言

线性代数在大学中的许多专业都是一门非常重要的课程,而在线性代数的课程中,对于矩阵的化简是每个学习线性代数的大学生都必须掌握的一项技能,矩阵的化简能帮助我们解决求矩阵的秩和求线性方程组解的个数的等问题。但对于矩阵的化简,许多时候都让我们非常的头疼,如果计算量过大,就无法保证计算的正确性。因此,对于学计算机专业的我来说,我想能否通过代码的形式解决这一类问题呢。

一、输入矩阵的信息

首先,要对矩阵进行处理,就需要知道矩阵的行数和列数。因此,第一步需要输入矩阵的行数和列数。

这里定义了两个变量作为输入,通过do——while语句进行限制,当输入的值为负数或0时,提示重新输入。

具体的代码如下:

int A,B;
System.out.println("矩阵化简为最简行列式");
System.out.println("请输入矩阵的行数:");
do {
    A=input.nextInt();
    if(A<=0)
    {
        System.out.println("输入的矩阵的行数错误!");
        System.out.println("请重新输入:");
    }
}while (A<=0);
System.out.println("请输入矩阵的列数:");
do {
    B=input.nextInt();
    if(B<=0)
    {
        System.out.println("输入的矩阵的列数错误!");
        System.out.println("请重新输入:");
    }
}while (B<=0);

然后通过两个for循环将矩阵中的所有的数进行输入。

具体的代码如下:

int [][]C=new int[A][B];
for(int i=0;i<A;i++)
{
    System.out.printf("请输入第%d行的所有数,每一个数之间用空格(或回车)隔开\n",i+1);
    for(int j=0;j<B;j++)
    {
        C[i][j]=input.nextInt();
    }
}

二、矩阵的初始处理


第一步:

需要将矩阵的同行同列的数通过与其他行进行交换使的同行同列的第一个数不为0,如果矩阵的某一列的数全为0,则不需要对这一行进行处理。

首先定义了一个变量D,使得D等于A(A为上面定义的矩阵的行数),再定义F=A-D,之后通过do——while语句,每次让D减1,那么F就可以从0一直加到(A-1)。

具体的代码如下:

F=A-D;
for(int i=F+1;i<A;i++)
{
    if(C[F][F]!=0)
    {
        break;
    }
    else
    {
        if(C[i][F]!=0)
        {
            for(int j=0;j<B;j++)
            {
                E=C[F][j];
                C[F][j]=C[i][j];
                C[i][j]=E;
            }
            break;
        }
    }
}

这一过程是在do——while语句里面进行的。

第二步:

当把矩阵同行同列的数全都换成不为0的数之后,就需要将该行与其他的行进行加法或减法运算,将其他行在该列下的所有数全部都化为0。

如下图所示的第一步操作:

第一个for循环是对每一行进行处理,然后在判断同行同列的数是否为0,目的是为了防止出现某一列数全为0的情况。之后在判断与他进行运算的其他行的该列下的数是否为0,如果为0则不进行运算。如果不为0则在判断这个数是大于0还是小于0,对于这两种情况有不同的处理。主要步骤是求两个数的最小公倍数,再将这个最小公倍数分别除以这两个数,得到两个新的倍数,之后通过这两个倍数与原来的两个数相乘在进行加法或减法运算,通过这样的操作,对于矩阵的初始处理就完成了。

具体的代码如下:

for(int i=0;i<A;i++)
{
    if(i!=F&&C[i][F]!=0)
    {
        if(C[F][F]!=0)
        {
            if(C[i][F]<0)
            {
                for(G=C[F][F];G%(-C[i][F])!=0;)
                    G+=C[F][F];
                G1=G/C[F][F];
                G2=G/(-C[i][F]);
                for(int j=0;j<B;j++)
                {
                    C[i][j]=C[i][j]*G2+C[F][j]*G1;
                }
            }
            else
            {
                for(G=C[F][F];G%C[i][F]!=0;)
                    G+=C[F][F];
                G1=G/C[F][F];
                G2=G/C[i][F];
                for(int j=0;j<B;j++)
                {
                    C[i][j]=C[i][j]*G2-C[F][j]*G1;
                }
            }
        }
    }
}

第一个for循环是对每一行进行处理,再判断同行同列的数是否为0,目的是为了防止出现某一列数全为0的情况。之后再判断与他进行运算的其他行的该列下的数是否为0,如果为0则不进行运算。如果不为0则在判断这个数是大于0还是小于0,对于这两种情况有不同的处理。主要步骤是求两个数的最小公倍数,再将这个最小公倍数分别除以这两个数,得到两个新的倍数,之后通过这两个倍数与原来的两个数相乘在进行加法或减法运算,通过这样的操作,对矩阵的初始处理就完成了。

三.矩阵的进一步处理

通过上面的处理已经大致的完成了矩阵的化简操作,但在后面的测试中发现了一个问题,就是通过上面的运算结果,如果第一个非0的数不在同行同列,那么就无法与其他行进行加法或减法运算了。因此,就需要对这些行进行处理。

第一步:

在对这些行进行处理之前,先需要对所有行进行排序,排序的规则是每一行中第一个非0的数越靠前,则该行的排序越靠前。因此,需要创建了一个数组来记录每一行中第一个非0的数的位置。

具体的代码如下:

int []H=new int[A];
for(int i=0;i<A;i++)
{
    for(int j=0;j<B;j++)
    {
        if(C[i][j]!=0)
        {
            H[i]=B-j;
            break;
        }
    }
}

记录好之后,再进行排序,其中不仅要交换两行的所有数,还要交换上面定义的数组里的数。

具体的代码如下:

for(int i=0;i<A;i++)
{
    for(int j=i+1;j<A;j++)
    {
        if(H[i]<H[j])
        {
            for(int k=0;k<B;k++)
            {
                E=C[i][k];
                C[i][k]=C[j][k];
                C[j][k]=E;
            }
            E=H[i];
            H[i]=H[j];
            H[j]=E;
        }
    }
}

之后再通过与矩阵的初始处理第二步一样的操作,就能够对其做进一步的处理了。

具体的代码如下:

for(int i=0;i<A;i++)
{
    if(R[i]==0)
    {
        for(int j=0;j<B;j++)
        {
            if(C[i][j]!=0)
            {
                if(C[i][j]<0)
                {
                    for(int Q=0;Q<i;Q++)
                    {
                        if(C[Q][j]!=0)
                        {
                            for(G=C[Q][j];G%(-C[i][j])!=0;)
                                G+=C[Q][j];
                            G1=G/C[Q][j];
                            G2=G/(-C[i][j]);
                            for(int k=0;k<B;k++)
                            {
                                C[Q][k]=C[Q][k]*G1+C[i][k]*G2;
                            }
                        }
                    }
                }
                else
                {
                    for(int Q=0;Q<i;Q++)
                    {
                        if(C[Q][j]!=0)
                        {
                            for(G=C[Q][j];G%C[i][j]!=0;)
                                G+=C[Q][j];
                            G1=G/C[Q][j];
                            G2=G/C[i][j];
                            for(int k=0;k<B;k++)
                            {
                                C[Q][k]=C[Q][k]*G1-C[i][k]*G2;
                            }
                        }
                    }
                }
                break;
            }
        }
    }
}

四.显示最终结果

矩阵化为最简型矩阵我们都知道,第一个不为0的数一定是1。

首先,可以通过一个for循环将为0的数全部输出,对于不为0的数再做进一步的处理,因此,需要再循环里面定义一个变量U来记录第一个数是否为0,如果不为0则通过break退出循环,再对后面不为0的数进行处理。对于第一个不为0的数,需要判断其大于0还是小于0,并通过除以它本身使它变为1,而后面的数都要除以前面第一个不为0的数。如果后面的数除以第一个不为0的数除不尽,就需要用分数表示,且分子和分母都需要同时除以他们的最大公约数,最后就得到了最简型矩阵了。

具体的代码如下:

for(int i=0;i<A;i++)
    {
        E=0;U=0;
        for(int j=0;j<B;j++)
        {
            if(C[i][j]!=0)
            {
                E=C[i][j];
                U=1;
                break;
            }
        }
        if(U==1)
        {
            for(int j=0;j<B;j++)
            {
                if(j==0)
                {
                    if(C[i][j]%E!=0)
                    {
                        if(C[i][j]%E<0)
                        {
                            J1=AAA(Math.abs(C[i][j]),Math.abs(E));
                            System.out.print("-"+Math.abs(C[i][j])/J1+"/"+Math.abs(E)/J1);
                        }
                        else
                        {
                            J1=AAA(Math.abs(C[i][j]),Math.abs(E));
                            System.out.print(Math.abs(C[i][j])/J1+"/"+Math.abs(E)/J1);
                        }
                    }
                    else
                    {
                        System.out.print(C[i][j]/E);
                    }
                }
                else
                {
                    if(C[i][j]%E!=0)
                    {
                        if(C[i][j]%E<0)
                        {
                            J1=AAA(Math.abs(C[i][j]),Math.abs(E));
                            System.out.print(" "+"-"+Math.abs(C[i][j])/J1+"/"+Math.abs(E)/J1);
                        }
                        else
                        {
                            J1=AAA(Math.abs(C[i][j]),Math.abs(E));
                            System.out.print(" "+Math.abs(C[i][j])/J1+"/"+Math.abs(E)/J1);
                        }
                    }
                    else
                    {
                        System.out.print(" "+C[i][j]/E);
                    }
                }
            }
            System.out.println();
        }
        else
        {
            for(int j=0;j<B;j++)
            {
                if(j==0)
                {
                    System.out.print("0");
                }
                else
                {
                    System.out.print(" 0");
                }
            }
            System.out.println();
        }
    }
}
public static int AAA(int J1,int J2){
    while (J1!=J2)
    {
        if (J1>J2)
            J1=J1-J2;
        else
            J2=J2-J1;
    }
    return J1;
}

五.代码使用过程

六.整体代码

package zuopin;
import java.util.Scanner;
public class str1{
    public static void main(String[] args) {
        Scanner input=new Scanner(System.in);
        int A,B;
        System.out.println("矩阵化简为最简行列式");
        System.out.println("请输入矩阵的行数:");
        do {
            A=input.nextInt();
            if(A<=0)
            {
                System.out.println("输入的矩阵的行数错误!");
                System.out.println("请重新输入:");
            }
        }while (A<=0);
        System.out.println("请输入矩阵的列数:");
        do {
            B=input.nextInt();
            if(B<=0)
            {
                System.out.println("输入的矩阵的列数错误!");
                System.out.println("请重新输入:");
            }
        }while (B<=0);
        int [][]C=new int[A][B];
        for(int i=0;i<A;i++)
        {
            System.out.printf("请输入第%d行的所有数,每一个数之间用空格(或回车)隔开\n",i+1);
            for(int j=0;j<B;j++)
            {
                C[i][j]=input.nextInt();
            }
        }
        int D=A,E,F,G,G1,G2;
        do{
            F=A-D;
            for(int i=F+1;i<A;i++)
            {
                if(C[F][F]!=0)
                {
                    break;
                }
                else
                {
                    if(C[i][F]!=0)
                    {
                        for(int j=0;j<B;j++)
                        {
                            E=C[F][j];
                            C[F][j]=C[i][j];
                            C[i][j]=E;
                        }
                        break;
                    }
                }
            }
            for(int i=0;i<A;i++)
            {
                if(i!=F&&C[i][F]!=0)
                {
                    if(C[F][F]!=0)
                    {
                        if(C[i][F]<0)
                        {
                            for(G=C[F][F];G%(-C[i][F])!=0;)
                                G+=C[F][F];
                            G1=G/C[F][F];
                            G2=G/(-C[i][F]);
                            for(int j=0;j<B;j++)
                            {
                                C[i][j]=C[i][j]*G2+C[F][j]*G1;
                            }
                        }
                        else
                        {
                            for(G=C[F][F];G%C[i][F]!=0;)
                                G+=C[F][F];
                            G1=G/C[F][F];
                            G2=G/C[i][F];
                            for(int j=0;j<B;j++)
                            {
                                C[i][j]=C[i][j]*G2-C[F][j]*G1;
                            }
                        }
                    }
                }
            }
            D--;
        }while (D>0);
        int []H=new int[A];
        for(int i=0;i<A;i++)
        {
            for(int j=0;j<B;j++)
            {
                if(C[i][j]!=0)
                {
                    H[i]=B-j;
                    break;
                }
            }
        }
        for(int i=0;i<A;i++)
        {
            for(int j=i+1;j<A;j++)
            {
                if(H[i]<H[j])
                {
                    for(int k=0;k<B;k++)
                    {
                        E=C[i][k];
                        C[i][k]=C[j][k];
                        C[j][k]=E;
                    }
                    E=H[i];
                    H[i]=H[j];
                    H[j]=E;
                }
            }
        }
        int []R=new int[A];
        for(int i=0;i<A;i++)
        {
            for(int j=0;j<=i;j++)
            {
                if(C[i][j]!=0)
                {
                    R[i]=1;
                    break;
                }
            }
        }
        for(int i=0;i<A;i++)
        {
            if(R[i]==0)
            {
                for(int j=0;j<B;j++)
                {
                    if(C[i][j]!=0)
                    {
                        if(C[i][j]<0)
                        {
                            for(int Q=0;Q<i;Q++)
                            {
                                if(C[Q][j]!=0)
                                {
                                    for(G=C[Q][j];G%(-C[i][j])!=0;)
                                        G+=C[Q][j];
                                    G1=G/C[Q][j];
                                    G2=G/(-C[i][j]);
                                    for(int k=0;k<B;k++)
                                    {
                                        C[Q][k]=C[Q][k]*G1+C[i][k]*G2;
                                    }
                                }
                            }
                        }
                        else
                        {
                            for(int Q=0;Q<i;Q++)
                            {
                                if(C[Q][j]!=0)
                                {
                                    for(G=C[Q][j];G%C[i][j]!=0;)
                                        G+=C[Q][j];
                                    G1=G/C[Q][j];
                                    G2=G/C[i][j];
                                    for(int k=0;k<B;k++)
                                    {
                                        C[Q][k]=C[Q][k]*G1-C[i][k]*G2;
                                    }
                                }
                            }
                        }
                        break;
                    }
                }
            }
        }
        int U,J1;
        for(int i=0;i<A;i++)
        {
            for(int j=0;j<B;j++)
            {
                if(C[i][j]!=0)
                {
                    if(C[i][j]<0)
                    {
                        for(int k=0;k<B;k++)
                        {
                            C[i][k]=-C[i][k];
                        }
                    }
                    break;
                }
            }
        }
        System.out.println("化简结果为:");
        for(int i=0;i<A;i++)
        {
            E=0;U=0;
            for(int j=0;j<B;j++)
            {
                if(C[i][j]!=0)
                {
                    E=C[i][j];
                    U=1;
                    break;
                }
            }
            if(U==1)
            {
                for(int j=0;j<B;j++)
                {
                    if(j==0)
                    {
                        if(C[i][j]%E!=0)
                        {
                            if(C[i][j]%E<0)
                            {
                                J1=AAA(Math.abs(C[i][j]),Math.abs(E));
                                System.out.print("-"+Math.abs(C[i][j])/J1+"/"+Math.abs(E)/J1);
                            }
                            else
                            {
                                J1=AAA(Math.abs(C[i][j]),Math.abs(E));
                                System.out.print(Math.abs(C[i][j])/J1+"/"+Math.abs(E)/J1);
                            }
                        }
                        else
                        {
                            System.out.print(C[i][j]/E);
                        }
                    }
                    else
                    {
                        if(C[i][j]%E!=0)
                        {
                            if(C[i][j]%E<0)
                            {
                                J1=AAA(Math.abs(C[i][j]),Math.abs(E));
                                System.out.print(" "+"-"+Math.abs(C[i][j])/J1+"/"+Math.abs(E)/J1);
                            }
                            else
                            {
                                J1=AAA(Math.abs(C[i][j]),Math.abs(E));
                                System.out.print(" "+Math.abs(C[i][j])/J1+"/"+Math.abs(E)/J1);
                            }
                        }
                        else
                        {
                            System.out.print(" "+C[i][j]/E);
                        }
                    }
                }
                System.out.println();
            }
            else
            {
                for(int j=0;j<B;j++)
                {
                    if(j==0)
                    {
                        System.out.print("0");
                    }
                    else
                    {
                        System.out.print(" 0");
                    }
                }
                System.out.println();
            }
        }
    }
    public static int AAA(int J1,int J2){
        while (J1!=J2)
        {
            if (J1>J2)
                J1=J1-J2;
            else
                J2=J2-J1;
        }
        return J1;
    }
}

相关文章

Java泛型T、E、K、V、N、?和Object区别和含义

通常我们在看一些源码时,发现全是T、?,晕乎乎的:sob:。于是,把泛型掌握好十分重要!什么是泛型Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型安全检测机...

2 分钟快速搞懂,Java 泛型中的通配符 T,E,K,V

Java 泛型中的通配符 T , E , K , V , ? 是什么?经常有同学会分不清楚。本文我们一起来了解下。送:《泛型最全知识导图》、《大厂泛型面试真题26道》,到本篇结尾处获得~1 什么是泛型...

java高级用法之:JNA类型映射应该注意的问题

简介JNA提供JAVA类型和native类型的映射关系,但是这一种映射关系只是一个大概的映射,我们在实际的应用中还有很多需要注意的事项,本文将会为大家详细讲解在使用类型映射中可能会出现的问题。一起来看...

PHP 8.0正式发布:支持JIT编译器,性能提升高达3倍

美国时间11月26日,PHP团队宣布PHP 8.0正式GA。PHP 8.0是PHP语言的最新主要版本,带来了许多新特性和优化,包括命名参数(named arguments)、联合类型(union ty...

升级IDEA后Lombok不能用了,如何解决?

今天到工作室比较晚,在电脑前吃着早饭,看到提示IDEA提示升级,寻思已经有好久没有升过级了。一样等着,就升级下吧。升级完毕重启之后,突然发现好多错误,原来的应用也没法启动了。仔细一看报错信息,是由于L...

SpringBoot 2.5.5整合轻量级的分布式日志标记追踪神器TLog

TLog能解决什么痛点随着微服务盛行,很多公司都把系统按照业务边界拆成了很多微服务,在排错查日志的时候。因为业务链路贯穿着很多微服务节点,导致定位某个请求的日志以及上下游业务的日志会变得有些困难。这时...