解n元线性方程组的C语言实现

#include<stdio.h>
int main ()
{
    int n,i,j,p,a,b,q;

    printf("Enter value n:\t");//输入阶数n// 
	scanf("%d",&n);
    double A[n][n],B[n-1][n-1],C[n][n],D[n][n];//A为原始数组,B为A的余子式,C为A的伴随矩阵(未转置),D为A的伴随矩阵A//
    double X[n],Q[n];//X    ,Q   //
	double det(double*p,int n);

    printf("Enter the array A:\n");//读入数组A// 
	for(i=0;i<n;i++)
	{
		for(j=0;j<n;j++)
			scanf("%lf",&A[i][j]);
	}
    printf("Enter the array A:\n");//读入数组Q// 
	for(i=0;i<n;i++)
	{
			scanf("%lf",&Q[i]);
	}
	
    printf("\n\nThe array A you enter:\n");//print original array/
	for(i=0;i<n;i++)
	{
		for(j=0;j<n;j++)
			printf("%10lf\t",A[i][j]);
		printf("\n\n");
	}
	 printf("\n\nThe array Q you enter:\n");//print original array/
	for(i=0;i<n;i++)
	{
			printf("\t%10lf\n",Q[i]);
		printf("\n\n");
	}
	
	
	
//(n-1)array to a new array//
    for(i=0;i<n;i++)
    {
	    for(j=0;j<n;j++)
	    {
		    for(p=0,a=0;p<n;p++)
		    {
			    if(p!=i)
			    { 
				    for(q=0,b=0;q<n;q++)
				    {
					    if(q!=j)
					    {
						    B[a][b]=A[p][q];

							if((i+j)%2==0)
								C[i][j]=det(B,n-1);//forget *(-1)//
							else
								C[i][j]=-det(B,n-1);

						    b++;
					    }
					
				    }
				    a++; 
			    }	
		    }	
	    }	
    }// end//


double result;
result=det(A,n);


if(result==0)
{
	printf("The array X is not exist.\n\n\n");
}

else                                        //error?//
{

    for(i=0;i<n;i++)
    {
	    for(j=0;j<n;j++)
		    D[j][i]=C[i][j]/result;
    }	

//计算A逆(D)*Q==X//
    for(i=0;i<n;i++)
    {
        X[i]=0;
        for(j=0;j<n;j++)
        {
            X[i]+=D[i][j]*Q[j];
        }
    }

    printf("\n\nThe array X :\n");
	for(i=0;i<n;i++)
	{
			printf("%10lf\t",X[i]);
	}
	
}
    return 0;
}



//det A//
double det(double*p,int n)
{

	int x,k,i,j;
	double t,h,det;
	double A[n][n],C[n][n];

for(i=0;i<n;i++)
	{
		for(j=0;j<n;j++)
            A[i][j]= *(p+i+n*j);
	}

	//核心代码// 
	for(j=0;j<n-1;j++)//第一次化归//
	{ 
		
		for(i=n-1,t=0;i>=j;i--)
		{
			if(A[i][j]!=0) 
			{
				t=A[i][j];
				x=i;
			}
		}
		if(t!=0)
		{
		for(h=0,i=n-1;i>=j+1;i--)
		{
			if(i!=x)
			{   h=A[i][j];
				for(k=0;k<n;k++)
				{
					A[i][k]-=h*A[x][k]/t;//error?// 
				}
			}
		}
		}
	}
	
	
	for(i=0;i<n;i++)//旋转数组// 
	{
		for(j=0;j<n;j++)
		{
			C[i][j]=A[n-j-1][n-i-1];
		}
	}
	for(j=0;j<n-1;j++)//第2次化归//
	{ 
		
		for(i=n-1,t=0;i>=j;i--)
		{
			if(C[i][j]!=0) 
			{
				t=C[i][j];
				x=i;
			}
		}
		if(t!=0)
		{
		for(i=n-1;i>=j+1;i--)
		{
			h=C[i][j];
			if(i!=x)
			{
				for(k=0;k<n;k++)
				{
					C[i][k]-=h*C[x][k]/t;//error// 
				}
			}
		}
		}
		
	}
	for(i=0;i<n;i++)//旋转回去// 
	{
		for(j=0;j<n;j++)
		{
			A[i][j]=C[n-j-1][n-i-1];
		}
	}
	for(det=1,i=0;i<n;i++)
	{
		det*=A[i][i];
	}
	return (det);
		//end det//
}


  • 2
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
线性方程组的通常方法是使用数值计算方法,其中最常用的方法是牛顿迭代法。下面是使用C语言实现牛顿迭代线性方程组的示例代码: ```c #include <stdio.h> #include <math.h> // 定义需要求的非线性方程组,这里以两个方程为例 double f1(double x, double y) { return x*x + y*y - 1; } double f2(double x, double y) { return x - y*y; } // 定义方程组的导数 double df1_dx(double x, double y) { return 2*x; } double df1_dy(double x, double y) { return 2*y; } double df2_dx(double x, double y) { return 1; } double df2_dy(double x, double y) { return -2*y; } // 牛顿迭代法求线性方程组 void newton(double x0, double y0, double eps) { double x = x0, y = y0, dx, dy; int iter = 0; do { iter++; double J[2][2] = {{df1_dx(x, y), df1_dy(x, y)}, {df2_dx(x, y), df2_dy(x, y)}}; double F[2] = {-f1(x, y), -f2(x, y)}; // 线性方程组 J * (dx, dy) = F dx = (F[0]*J[1][1] - F[1]*J[0][1]) / (J[0][0]*J[1][1] - J[1][0]*J[0][1]); dy = (F[1]*J[0][0] - F[0]*J[1][0]) / (J[0][0]*J[1][1] - J[1][0]*J[0][1]); x += dx; y += dy; } while (fabs(dx) > eps || fabs(dy) > eps); printf("Solution: (%f, %f)\n", x, y); printf("Iterations: %d\n", iter); } int main() { newton(0.5, 0.5, 1e-6); return 0; } ``` 这里定义了两个需要求的非线性方程以及它们的导数,然后使用牛顿迭代法求方程组。在迭代过程中,需要计算方程组的雅可比矩阵和残差向量,并且线性方程组来求迭代方向。最终迭代到一定的精度后输出和迭代次数。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值