Hàm đa thức - Chương 4: Giải hệ phương trình đại số tuyến tính (Phần 2)

Phương pháp lặp Gauss-Seidel được cải tiến từ phương pháp lặp đơn . Nội dung cơ bản của phương pháp là ở chỗ khi tính nghiệm xấp xỉ thứ (k+1) của ẩn xi ta sở dụng các xấp xỉ thứ (k+1) đã tính của các ẩn x1,.,xi-1. Giả sử đã cho hệ : AX = B và ta có nghiệm :

Lấy xấp xỉ ban đầu tuỳ ý x1(o) , x2(o) ,., xn(o) và tất nhiên ta cố gắng lấy chúng tương ứng với x1, x2 ,., xn (càng gần càng tốt). Tiếp theo ta giả sử rằng đã biết xấp xỉ thứ k xi(k) của nghiệm .

 

doc17 trang | Chuyên mục: Giải Tích | Chia sẻ: tuando | Lượt xem: 684 | Lượt tải: 0download
Tóm tắt nội dung Hàm đa thức - Chương 4: Giải hệ phương trình đại số tuyến tính (Phần 2), để xem tài liệu hoàn chỉnh bạn click vào nút "TẢI VỀ" ở trên
 buoc tinh",k);
	printf("\n");
	printf("NGHIEM CUA HE");
	printf("\n");
	for (i=1;i<=n;i++)
	 printf("x[%d] = %10.5f\n",i,x1[i]);
	 }
	k=k+1;
	if (k>l)
	 {
	t=1;
	printf("Phep lap khong hoi tu sau %d buoc tinh",k-1);
	 }
	 }
	while (t==0);
	getch();
 }
§6. PHƯƠNG PHÁP LẶP GAUSS - SEIDEL
	Phương pháp lặp Gauss-Seidel được cải tiến từ phương pháp lặp đơn . Nội dung cơ bản của phương pháp là ở chỗ khi tính nghiệm xấp xỉ thứ (k+1) của ẩn xi ta sở dụng các xấp xỉ thứ (k+1) đã tính của các ẩn x1,...,xi-1. Giả sử đã cho hệ : AX = B và ta có nghiệm :	
Lấy xấp xỉ ban đầu tuỳ ý x1(o) , x2(o) ,...., xn(o) và tất nhiên ta cố gắng lấy chúng tương ứng với x1, x2 ,..., xn (càng gần càng tốt). Tiếp theo ta giả sử rằng đã biết xấp xỉ thứ k xi(k) của nghiệm . Theo Seidel ta sẽ tìm xấp xỉ thứ (k+1) của nghiệm theo các công thức sau :
	......
	......
	Thông thường phương pháp Gauss - Seidel hội tụ nhanh hơn phương pháp lặp đơn nhưng tính toán phức tạp hơn. Dể dễ hiểu phương pháp này chúng ta xét một ví dụ cụ thể:
	Cho hệ phương trình :
nghiệm đúng của hệ là (1 , 1, 1)
Ta đưa về dạng thuận tiện cho phép lặp :
Lấy x1(o) = 1.2 ; x2(o) = 0 ; x3(o) = 0;
Sử dụng phương pháp lặp Seidel ta có :
và cứ thế tiếp tục. Chương trình mô tả thuật toán Gauss - Seidel như sau:
Chương trình 4-8
#include 
#include 
#include 
#include 
#include 
#define max 6
void main()
 {
	float b[max],x[max];
	float a[max][max];
	int i,j,k,n,dem,t1;
	float t,s,d,w,epsi;
	char tl;
 clrscr();
	printf("Cho so an so n = ");
	scanf("%d",&n);
	printf("Cho cac phan tu cua ma tran a : \n");
	for (i=1;i<=n;i++)
	 for (j=1;j<=n;j++)
	{
	 printf("a[%d][%d] = ",i,j);
	 scanf("%f",&a[i][j]);
	}
	printf("\n");
	printf("Ma tran a ma ban da nhap\n");
	printf("\n");
	for (i=1;i<=n;i++)
	 {
	for (j=1;j<=n;j++)
	 printf("%10.5f",a[i][j]);
	printf("\n");
	 }
	printf("\n");
	t1=1;
	flushall();
	while (t1)
	 {
	printf("Co sua ma tran a khong(c/k)?");
	scanf("%c",&tl);
	if (toupper(tl)=='C')
	 {
	printf("Cho chi so hang can sua : ");
	scanf("%d",&i);
	printf("Cho chi so cot can sua : ");
	scanf("%d",&j);
	printf("a[%d][%d] = ",i,j);
	scanf("%f",&a[i][j]);
	flushall();
	 }
	if (toupper(tl)=='K')
	 t1=0;
	 }
	printf("Ma tran a\n");
	printf("\n");
	for (i=1;i<=n;i++)
	 {
	for (j=1;j<=n;j++)
	 printf("%15.5f",a[i][j]);
	printf("\n");
	 }
	printf("\n");
	printf("Cho cac phan tu cua ma tran b : \n");
	for (i=1;i<=n;i++)
	 {
	printf("b[%d] = ",i);
	scanf("%f",&b[i]);
	 }
	printf("\n");
	printf("Ma tran b");
	printf("\n");
	for (i=1;i<=n;i++)
	 printf("b[%d] = %10.5f\n",i,b[i]);
	printf("\n");
	t1=1;
	flushall();
	while (t1)
	 {
	printf("Co sua ma tran b khong(c/k)?");
	scanf("%c",&tl);
	if (toupper(tl)=='C')
	 {
	printf("Cho chi so hang can sua : ");
	scanf("%d",&i);
	printf("b[%d] = ",i);
	scanf("%f",&b[i]);
	flushall();
	 }
	if (toupper(tl)=='K')
	 t1=0;
	 }
	printf("\n");
	printf("Ma tran b");
	printf("\n");
	for (i=1;i<=n;i++)
	 printf("b[%d] = %10.5f\n",i,b[i]);
	printf("\n");
	printf("Cho so lan lap k : ");
	scanf("%d",&k);
	printf("\n");
	w=1;
	epsi=1e-8;
	for (i=1;i<=n;i++)
	 x[i]=0.0;
	dem = 0;
	do
	 {
	dem=dem+1;
	for (i=1;i<=n;i++)
	 {
	s=0.0;
	for (j=1;j<=n;j++)
	 s=s+a[i][j]*x[j];
	d=x[i];
	x[i]=(1-w)*d+w*(-s+a[i][i]*d+b[i])/a[i][i];
	t=fabs(d-x[i]);
	 }
 }
	while ((demepsi*fabs(x[n])));
	if (t<epsi*fabs(x[n]))
	 {
	printf("Nghiem sau %d lan lap la :\n",dem);
	for (i=1;i<=n;i++)
	 printf("x[%d] = %12.8f\n",i,x[i]);
	 }
	else
	 {
	printf("Khong dat do chinh xac sau %d lan lap\n",k);
	printf("Nghiem cua lan lap cuoi cung la : \n");
	for (i=1;i<=n;i++)
	 printf("x[%d] = %12.8f\n",i,x[i]);
	 }
	getch();
 }
§7. PHƯƠNG PHÁP CRAMER
 Một trường hợp riêng của hệ phương trình, trong đó số phương trình bằng số ẩn, nghĩa là hệ có dạng : 
trong đó A= (aij) là một ma trận không suy biến. Một hệ phương trình như vậy có tên là hệ thống Cramer
Định lí Crame: Hệ thống Crame có nghiệm duy nhất được cho bởi công thức 
trong đó A(i) là ma trận nhận được từ A bằng cách thay cột thứ i bởi cột B. 
Như vậy để giải hệ bằng phương pháp Cramer chúng ta lần lượt tính các định thức của ma trận và ma trận thay thế rồi tìm nghiệm theo công thức
Cramer. Chương trình sau mô tả thuật toán này:
Chương trình 4-9
// Cramer;
#include 
#include 
#include 
#define max 50
void main()
 {
	float r[max][max],a[max][max];
	float b[max],x[max];
	float delta[max];
	int i,j,k,l,t,n,ok1,ok2,t1;
	float c,d;
	char tl;
	clrscr();
	printf("Cho so an cua phuong trinh n = ");
	scanf("%d",&n);
	printf("Cho cac phan tu cua ma tran a : \n");
	for (i=1;i<=n;i++)
	 for (j=1;j<=n;j++)
	{
	 printf("a[%d][%d] = ",i,j);
	 scanf("%f",&a[i][j]);
	}
	printf("\n");
	printf("Ma tran a ma ban da nhap\n");
	printf("\n");
	for (i=1;i<=n;i++)
	 {
	for (j=1;j<=n;j++)
	 printf("%10.5f",a[i][j]);
	printf("\n");
	 }
	printf("\n");
	t1=1;
	flushall();
	while (t1)
	 {
	printf("Co sua ma tran a khong(c/k)?");
	scanf("%c",&tl);
	if (toupper(tl)=='C')
	 {
	printf("Cho chi so hang can sua : ");
	scanf("%d",&i);
	printf("Cho chi so cot can sua : ");
	scanf("%d",&j);
	printf("a[",i,",",j,"] = ");
	scanf("%f",&a[i][j]);
	flushall();
	 }
	if (toupper(tl)=='K')
	 t1=0;
	 }
	printf("Ma tran a \n");
	printf("\n");
	for (i=1;i<=n;i++)
	 {
	for (j=1;j<=n;j++)
	 printf("%10.5f",a[i][j]);
	printf("\n");
	 }
	printf("\n");
 printf("Cho cac phan tu cua ma tran b : \n");
 for (i=1;i<=n;i++)
	 {
	printf("b[%d] = ",i);
	scanf("%f",&b[i]);
	 }
	printf("\n");
	printf("Ma tran b ma ban da nhap\n");
	printf("\n");
	for (i=1;i<=n;i++)
	 printf("b[%d] = %10.5f\n",i,b[i]);
	printf("\n");
	t1=1;
	flushall();
	while (t1)
	 {
	printf("Co sua ma tran b khong(c/k)?");
	scanf("%c",&tl);
	if (toupper(tl)=='C')
	 {
	printf("Cho chi so hang can sua : ");
	scanf("%d",&i);
	printf("b[%d] = ",i);
	scanf("%f",&b[i]);
	flushall();
	 }
	if (toupper(tl)=='K')
	 t1=0;
	 }
	printf("\n");
	printf("Ma tran b\n");
	printf("\n");
	for (i=1;i<=n;i++)
	 printf("%10.5f",b[i]);
	printf("\n");
 //thay b vao tung cot cua a de tinh cac dinh thuc
	for (k=0;k<=n;k++)
	 {
	for (i=1;i<=n;i++)
	 for (j=1;j<=n;j++)//thay cot b vao a
	if (i==k)
	 r[j][i]=b[j];
	else
	 r[j][i]=a[j][i];
 //tinh dinh thuc
	 d=1.0;
	 i=1;
	 ok2=1;
	 while (ok2&&(i<=n))
	{
	 if (r[i][i]==0.0)
	{
	 ok1=1;
	 t=t+1;
	 while (ok1&&(t<=n))
	if (r[t][i]!=0)
	 {
	for (j=i;j<=n;j++)
	 {
	c=r[i][j];
	r[i][j]=r[t][j];
	r[k][j]=c;
	 }
	d=-d;
	ok1=0;
	 }
	else
	 t=t+1;
	if (k>n)
	 {
	printf("\n");
	printf("** MA TRAN SUY BIEN **");
	ok2=0;
	d=0.0;
	 }
	 }
	if (r[i][i]!=0)
	 {
	c=r[i][i];
	for (j=i+1;j<=n;j++)
	 r[i][j]=r[i][j]/c;
	for (t=i+1;t<=n;t++)
	 {
	c=r[t][i];
	for (j=i+1;j<=n;j++)
	 r[t][j]=r[t][j]-r[i][j]*c;
	 }
	 }
	i=i+1;
	 }
	if (ok2)
	 for (i=1;i<=n;i++)
	d=d*r[i][i];
	delta[k]=d;
	 }
	if (delta[0]==0.0)
	 printf("He da cho vo nghiem\n");
	else
	 {
	printf("\nNGHIEM CUA HE :\n");
	printf("\n");
	for (i=1;i<=n;i++)
	 {
	x[i]=delta[i]/delta[0];
	printf("x[%d] = %10.5f\n",i,x[i]);
	 }
	 }
	getch();
 }
§8. HỆ PHƯƠNG TRÌNH SỐ PHỨC
	Giả sử ta có một hệ phương trình dạng số phức dạng AX = B trong đó A = C + jD , B = E +jF và X = Y + jZ . Ta viết lại phương trình dưới dạng :
	(C + jD)(Y + jZ) = (E +jF)
Nhân biểu thức về trái và cân bằng phần thực với phần thực và phần ảo với phần ảo ta nhận được hệ mới :
Như vậy chúng ta nhận được một hệ gồm 2n phương trình số thực. Giải hệ này và kết hợp các phần thực và phần ảo ta nhận được nghiệm của hệ phương trình ban đầu. Chương trình giải hệ phương trình như vậy cho ở 
dưới đây: 
Chương trình 4-10
#include 
#include 
#include 
#include 
#include 
#define max 20
void main()
 {
	int i,j,k,l,n,m;
	float s,t,a[max][max],b[max][max],x[max];
	clrscr();
	printf("Cho so an so cua phuong trinh n = ");
	scanf("%d",&n);
	printf("Cho phan thuc cua cac he so,ke ca ve phai\n");
	for (i=1;i<=n;i++)
	 for (j=1;j<=n+1;j++)
	{
	 printf("a[%d][%d] = ",i,j);
	 scanf("%f",&a[i][j]);
	}
 printf("\n");
	printf("Cho phan ao cua cac he so,ke ca ve phai\n");
	for (i=1;i<=n;i++)
	 for (j=1;j<=n+1;j++)
	{
	 printf("b[%d][%d] = ",i,j);
	 scanf("%f",&b[i][j]);
	}
	for (i=1;i<=n;i++)
	 a[i][2*n+1]=a[i][n+1];
	for (i=n+1;i<=2*n;i++)
	 a[i][2*n+1]=b[i-n][n+1];
	for (i=n+1;i<=2*n;i++)
	 for (j=1;j<=n;j++)
	a[i][j]=b[i-n][j];
	for (i=1;i<=n;i++)
	 for (j=n+1;j<=2*n;j++)
	a[i][j]=-b[i][j-n];
	for (i=n+1;i<=2*n;i++)
	 for (j=n+1;j<=2*n;j++)
	a[i][j]=a[i-n][j-n];
	m=2*n;
	for (k=1;k<=m-1;k++)
	 {
	s=0.0;
	for (i=k;i<=m;i++)
	 {
	t=fabs(a[i][k]);
	if (s<=t)
	 {
	s=t;
	l=i;
	 }
	 }
	for (j=k;j<=m+1;j++)
	 {
	s=a[k][j];
	a[k][j]=a[l][j];
	a[l][j]=s;
	 }
	if (fabs(a[k][k]/a[1][1])<=1e-08)
	 {
	printf("Ma tran suy bien\n");
	getch();
	exit(1);
	 }
	for (i=k+1;i<=m;i++)
	 {
	s=a[i][k]/a[k][k];
	a[i][k]=0.0;
	for (j=k+1;j<=m+1;j++)
	 a[i][j]-=s*a[k][j];
	 }
	 }
	x[m]=a[m][m+1]/a[m][m];
	for (i=m-1;i>=1;i--)
	 {
	s=a[i][m+1];
	for (j=i+1;j<=m;j++)
	 s-=a[i][j]*x[j];
	x[i]=s/a[i][i];
	 }
	printf("\n");
	printf("Nghiem phuc cua he\n");
	for (i=1;i<=n;i++)
	 if (x[i+n]<0)
	printf("%10.5f-%10.5fj\n",x[i],fabs(x[i+n]));
	 else
	printf("%10.5f+%10.5fj\n",x[i],x[i+n]);
	getch();
 }
	Dùng chương trình này giải hệ phương trình :
Ta nhận được các nghiệm x = 2 + 3j ; y = 1 - 2j ; z = -1 + 4j và r = 1- j
	Ngoài các phương pháp nêu trên ta thấy rằng từ hệ phương trình AX = B ta có thể tìm nghiệm X của hệ bằng cách viết lại phương trình dưới dạng X = B/A =A-1B với A-1 là ma trận nghịch đảo của A. Do vậy trước hết ta cần tìm A-1 và sau đó tính tích A-1B.

File đính kèm:

  • docham_da_thuc_chuong_4_giai_he_phuong_trinh_dai_so_tuyen_tinh.doc