BÀI TOÁN LIỆT KÊ - Tài Liệu Text - 123doc

Tải bản đầy đủ (.pdf) (27 trang)
  1. Trang chủ
  2. >>
  3. Khoa Học Tự Nhiên
  4. >>
  5. Toán học
BÀI TOÁN LIỆT KÊ

Bạn đang xem bản rút gọn của tài liệu. Xem và tải ngay bản đầy đủ của tài liệu tại đây (435.39 KB, 27 trang )

Chương 3: Bài toán liệt kê CHƯƠNG III: BÀI TOÁN LIỆT KÊ Đối với một bài toán, khi chưa tìm được giải thuật tốt để giải thì liệt kê là biện pháp cuối cùng để thực hiện với sự hỗ trợ của máy tính. Có thể nói, liệt kê là phương pháp phổ dụng nhất để giải quyết một bài toán trên máy tính. Trái lại, bài toán tồn tại chỉ cần chỉ ra được bài toán có nghiệm hay không có nghiệm và thường là những bài toán khó. Nhiều bài toán tồn tại đã được phát biểu trong nhiều thập kỉ nhưng vẫn chưa được giải quyết.Giải quyết được chúng sẽ thúc đẩy sự phát triển của nhiều ngành toán học. Nội dung chính của chương này tập chung giải quyết những vấn đề cơ bản sau: 9 Giới thiệu bài toán liệt kê. 9 Giải quyết bài toán liệt kê bằng phương pháp sinh. 9 Giải quyết bài toán liệt kê bằng phương pháp quay lui dựa trên giải thuật đệ qui. Bạn đọc có thể tìm thấy cách giải nhiều bài toán liệt kê và bài toán tồn tại hay trong các tài liệu [1] và [2] trong tài liệu tham khảo. 3.1. GIỚI THIỆU BÀI TOÁN Bài toán đưa ra danh sách tất cả các cấu hình tổ hợp có thể có được gọi là bài toán liệt kê tổ hợp. Khác với bài toán đếm là tìm kiếm một công thức cho lời giải, bài toán liệt kê lại cần xác định một thuật toán để theo đó có thể xây dựng được lần lượt tất cả các cấu hình cần quan tâm. Một thuật toán liệt kê phải đảm bảo hai nguyên tắc:  Không được lặp lại một cấu hình  Không được bỏ xót một cấu hình Ví dụ 1. Cho tập hợp các số a1, a2,.., an và số M. Hãy tìm tất cả các tập con k phần tử của dãy số {an} sao cho tổng số các phần tử trong tập con đó đúng bằng M. Giải: Như chúng ta đã biết, số các tập con k phần tử của tập gồm n phần tử là C(n,k). Như vậy chúng ta cần phải duyệt trong số C(n,k) tập k phần tử để lấy ra những tập có tổng các phần tử đúng bằng M. Vì không thể xác định được có bao nhiêu tập k phần tử từ tập n phần tử có tổng các phần tử đúng bằng M nên chúng ta chỉ còn cách liệt kê các cấu hình thoả mãn điều kiện đã cho. Ví dụ 2. Một thương nhân đi bán hàng tại tám thành phố. Chị ta có thể bắt đầu hành trình của mình tại một thành phố nào đó nhưng phải qua 7 thành phố kia theo bất kỳ thứ tự nào mà chị ta muốn. Hãy chỉ ra lộ trình ngắn nhất mà chị ta có thể đi. Giải: Vì thành phố xuất phát đã được xác định. Do vậy thương nhân có thể chọn tuỳ ý 7 thành phố còn lại để hành trình. Như vậy, tất cả số hành trình của thương nhân có thể đi qua là 7! 49Chương 3: Bài toán liệt kê = 5040 cách. Tuy nhiên trong 5040 cách chúng ta phải duyệt toàn bộ để chỉ ra một hành trình là ngẵn nhất. Có thể nói phương pháp liệt kê là biện pháp cuối cùng nhưng cũng là biện pháp phổ dụng nhất để giải quyết các bài toán tổ hợp. Khó khăn chính của phương pháp này là sự bùng nổ tổ hợp. Để xây dựng chừng 1 tỷ cấu hình (con số này không phải là lớn đối với các bài toán tổ hợp như số mất thứ tự Dn, số phân bố Un, số hình vuông la tinh ln), ta giả sử cần 1 giây để liệt kê một cấu hình thì chúng ta cũng cần 31 năm mới giải quyết xong. Tuy nhiên với sự phát triển nhanh chóng của máy tính, bằng phương pháp liệt kê, nhiều bài toán khó của lý thuyết tổ hợp đã được giải quyết và góp phần thúc đẩy sự phát triển của nhiều ngành toán học. 3.2. ĐỆ QUI 3.2.1. Định nghĩa bằng đệ qui Trong thực tế, chúng ta gặp rất nhiều đối tượng mà khó có thể định nghĩa nó một cách tường minh, nhưng lại dễ dàng định nghĩa đối tượng qua chính nó. Kỹ thuật định nghĩa đối tượng qua chính nó được gọi là kỹ thuật đệ qui (recursion). Đệ qui được sử dụng rộng rãi trong khoa học máy tính và lý thuyết tính toán. Các giải thuật đệ qui đều được xây dựng thông qua hai bước: bước phân tích và bước thay thế ngược lại. Ví dụ 1. Để tính tổng S(n) = 1 + 2 +...+ n, chúng ta có thể thực hiện thông qua hai bước như sau: Bước phân tích:  Để tính toán được S(n) trước tiên ta phải tính toán trước S(n-1) sau đó tính S(n) = S(n-1) +n.  Để tính toán được S(n-1), ta phải tính toán trước S(n-2) sau đó tính S(n-1) = S(n-2) + n-1.  ......................................................  Để tính toán được S(2), ta phải tính toán trước S(1) sau đó tính S(2) = S(1) + 2.  Và cuối cùng S(1) chúng ta có ngay kết quả là 1. Bước thay thế ngược lại: Xuất phát từ S(1) thay thế ngược lại chúng ta xác định S(n):  S(1) = 1  S(2) = S(1) + 2  S(3) = S(2) + 3  ............  S(n) = S(n - 1) + n 50 Chương 3: Bài toán liệt kê Ví dụ 2. Định nghĩa hàm bằng đệ qui: Hàm f(n) = n! Dễ thấy f(0) = 1. Vì (n+1) ! = 1. 2.3... n(n+1) = n! (n+1), nên ta có: f(n+1) = ( n+1). f(n) với mọi n nguyên dương. Ví dụ 3. Tập hợp định nghĩa bằng đệ qui: Định nghĩa đệ qui tập các xâu: Giả sử Σ* là tập các xâu trên bộ chữ cái Σ. Khi đó Σ* được định nghĩa bằng đệ qui như sau:  λ ∈ Σ*, trong đó λ là xâu rỗng  wx ∈ Σ* nếu w ∈ Σ* và x ∈ Σ* 3.2.2. Giải thuật đệ qui Một thuật toán được gọi là đệ qui nếu nó giải bài toán bằng cách rút gọn bài toán ban đầu thành bài toán tương tự như vậy sau một số hữu hạn lần thực hiện. Trong mỗi lần thực hiện, dữ liệu đầu vào tiệm cận tới tập dữ liệu dừng. Ví dụ: để giải quyết bài toán tìm ước số chung lớn nhất của hai số nguyên dương a và b với b> a, ta có thể rút gọn về bài toán tìm ước số chung lớn nhất của (b mod a) và a vì USCLN(b mod a, a) = USCLN(a,b). Dãy các rút gọn liên tiếp có thể đạt được cho tới khi đạt điều kiện dừng USCLN(0, a) = USCLN(a, b) = a. Dưới đây là ví dụ về một số thuật toán đệ qui thông dụng. Thuật toán 1: Tính an bằng giải thuật đệ qui, với mọi số thực a và số tự nhiên n. double power( float a, int n ){ if ( n ==0) return(1); return(a *power(a,n-1)); } Thuật toán 2: Thuật toán đệ qui tính ước số chung lớn nhất của hai số nguyên dương a và b. int USCLN( int a, int b){ if (a == 0) return(b); return(USCLN( b % a, a)); } Thuật toán 3: Thuật toán đệ qui tính n! long factorial( int n){ 51Chương 3: Bài toán liệt kê if (n ==1) return(1); return(n * factorial(n-1)); } Thuật toán 4: Thuật toán đệ qui tính số fibonacci thứ n int fibonacci( int n) { if (n==0) return(0); else if (n ==1) return(1); return(fibonacci(n-1) + fibonacci(n-2)); } 3.3. PHƯƠNG PHÁP SINH Phương pháp sinh có thể áp dụng để giải các bài toán liệt kê tổ hợp đặt ra nếu như hai điều kiện sau được thực hiện: i. Có thể xác định được một thứ tự trên tập các cấu hình tổ hợp cần liệt kê. Từ đó có thể xác định được cấu hình tổ hợp đầu tiên và cuối cùng trong thứ tự đã được xác định. ii. Xây dựng được thuật toán từ cấu hình chưa phải là cuối cùng đang có để đưa ra cấu hình kế tiếp sau nó. Ta gọi thuật toán trong điều kiện (ii) là thuật toán sinh kế tiếp. Rõ ràng thuật toán này chỉ thực hiện được khi có một cấu hình được xác định theo điều kiện (i). Giả sử một bài toán đều thoả mãn các điều kiện trên, khi đó phương pháp sinh kế tiếp có thể được mô tả bằng thủ tục như sau: void Generate(void){ <Xây dựng cấu hình ban đầu>; stop =false while (not stop) { <Đưa ra cấu hình đang có>; Sinh_Kế_Tiếp; } } Trong đó Sinh_Kế_Tiếp là thủ tục sinh cấu hình kế tiếp từ cấu hình ban đầu. Nếu cấu hình là cấu hình cuối cùng, thủ tục này cần gán giá trị True cho stop, ngược lại thủ tục này sẽ xây dựng cấu hình kế tiếp của cấu hình đang có trong thứ tự đã xác định. Dưới đây là một số ví dụ điển hình mô tả thuật toán sinh kế tiếp. 52 Chương 3: Bài toán liệt kê Ví dụ 1. Liệt kê tất cả các dãy nhị phân độ dài n. Giải: Viết dãy nhị phân dưới dạng b1b2..bn, trong đó bi∈{0, 1 }. Xem mỗi dãy nhị phân b=b1b2..bn là biểu diễn nhị phân của một số nguyên p(b). Khi đó thứ tự hiển nhiên nhất có thể xác định trên tập các dãy nhị phân là thứ tự từ điển được xác định như sau: Ta nói dãy nhị phân b = b1b2..bn đi trước dãy nhị phân b’ = b’1b’2..b’n theo thứ tự từ điển và kí hiệu b<b’nếu p(b) <p(b’). Ví dụ với n=4, các xâu nhị phân độ dài 4 được liệt kê theo thứ tự từ điển là: b p(b) b p(b) 0000 0001 0010 0011 0100 0101 0110 0111 0 1 2 3 4 5 6 7 1000 1001 1010 1011 1100 1101 1110 1111 8 9 10 11 12 13 14 15 Như vậy, dãy đầu tiên là 0000 dãy cuối cùng là 1111. Nhận xét rằng, nếu xâu nhị phân chứa toàn bít 1 thì quá trình liệt kê kết thúc, trái lại dãy kế tiếp sẽ nhận được bằng cách cộng thêm 1 (theo modul 2 có nhớ) vào dãy hiện tại. Từ đó ta nhận được qui tắc sinh kế tiếp như sau:  Tìm i đầu tiên từ phải xang trái (i=n, n-1,..,1) thoả mãn bi =0.  Gán lại bi =1 và bj=0 với tất cả j>i. Dãy thu được là dãy cần tìm. Ví dụ ta có xâu nhị phân độ dài 10: 1100111011. Ta có i = 8, ta đặt b8 =1, b9,b10 =0 ta được xâu nhị phân kế tiếp: 1100111100. Thuật toán sinh kế tiếp được mô tả trong thủ tục sau: void Next_Bit_String( int *B, int n ){ i = n; while (bi ==1 ) { bi = 0 i = i-1; } bi = 1; } 53Chương 3: Bài toán liệt kê Dưới đây là chương trình liệt kê các xâu nhị phân có độ dài n. #include <stdio.h> #include <alloc.h> #include <stdlib.h> #include <conio.h> #define MAX 100 #define TRUE 1 #define FALSE 0 int Stop, count; void Init(int *B, int n){ int i; for(i=1; i<=n ;i++) B[i]=0; count =0; } void Result(int *B, int n){ int i;count++; printf("\n Xau nhi phan thu %d:",count); for(i=1; i<=n;i++) printf("%3d", B[i]); } void Next_Bits_String(int *B, int n){ int i = n; while(i>0 && B[i]){ B[i]=0; i--; } if(i==0 ) Stop=TRUE; else B[i]=1; } void Generate(int *B, int n){ 54 Chương 3: Bài toán liệt kê int i; Stop = FALSE; while (!Stop) { Result(B,n); Next_Bits_String(B,n); } } void main(void){ int i, *B, n;clrscr(); printf("\n Nhap n=");scanf("%d",&n); B =(int *) malloc(n*sizeof(int)); Init(B,n);Generate(B,n);free(B);getch(); } Ví dụ 2. Liệt kê tập con m phần tử của tập n phần tử. Cho X = { 1, 2,.., n }. Hãy liệt kê tất cả các tập con k phần tử của X (k≤ n). Giải: Mỗi tập con của tập hợp X có thể biểu diễn bằng bộ có thứ tự gồm k thành phần a =(a1a2..ak) thoả mãn 1 ≤ a1 ≤ a2 ≤..≤ ak ≤ n. Trên tập các tập con k phần tử của X có thể xác định nhiều thứ tự khác nhau. Thứ tự dễ nhìn thấy nhất là thứ tự từ điển được định nghĩa như sau: Ta nói tập con a = a1a2... ak đi trước tập con a’ = a1’a2’...ak’ trong thứ tự từ điển và ký hiệu là a<a’, nếu tìm được chỉ số j ( 1 ≤ j ≤ k ) sao cho: a1 = a1’, a2 = a2’,..., aj-1 = a’j-1, aj < a’j. Chẳng hạn X = { 1, 2, 3, 4, 5 }, k = 3. Các tập con 3 phần tử của X được liệt kê theo thứ tự từ điển như sau: 1 2 3 1 2 4 1 2 5 1 3 4 1 3 5 1 4 5 2 3 4 2 3 5 55Chương 3: Bài toán liệt kê 2 4 5 3 4 5 Như vậy, tập con đầu tiên trong thứ tự từ điển là (1, 2,.., k) và tập con cuối cùng là (n-k+1, n-k+2,.., n). Giả sử a = (a1, a2,.., ak) là tập con hiện tại và chưa phải là cuối cùng, khi đó có thể chứng minh được rằng tập con kế tiếp trong thứ tự từ điển có thể được xây dựng bằng cách thực hiện các qui tắc biến đổi sau đối với tập con đang có.  Tìm từ bên phải dãy a1, a2,.., ak phần tử ai≠n – k + i  Thay ai bởi ai +1,  Thay aj bởi ai + j – i, với j:= i+1, i + 2,..., k Chẳng hạn với n = 6, k =4. Giả sử ta đang có tập con (1, 2, 5, 6), cần xây dựng tập con kế tiếp nó trong thứ tự từ điển. Duyệt từ bên phải ta nhận được i =2, thay a2 bởi a2 + 1 = 2 + 1 =3. Duyệt j từ i + 1 = 3 cho đến k, ta thay thế a3 = a2 + 3 – 2 = 3 + 3 - 2 = 4, a4 = a2 + 4 - 2 = 3 + 4 – 2 = 5 ta nhận được tập con kế tiếp là ( 1, 3, 4, 5). Với qui tắc sinh như trên, chúng ta có thể mô tả bằng thuật toán sau: Thuật toán liệt kê tập con kế tiếp m phần tử của tập n phần tử: void Next_Combination( int *A, int m){ i = m; while ( ai == m-n+i) i = i -1; ai = ai + 1; for ( j = i+1; j <=m; j++) aj = ai + j - i; } Văn bản chương trình liệt kê tập các tập con m phần tử của tập n phần tử được thể hiện như sau: #include <stdio.h> #include <conio.h> #define TRUE 1 #define FALSE 0 #define MAX 100 int n, k, count, C[MAX], Stop; void Init(void){ int i; printf("\n Nhap n="); scanf("%d", &n); 56 Chương 3: Bài toán liệt kê printf("\n Nhap k="); scanf("%d", &k); for(i=1; i<=k; i++) C[i]=i; } void Result(void){ int i;count++; printf("\n Tap con thu %d:", count); for(i=1; i<=k; i++) printf("%3d", C[i]); } void Next_Combination(void){ int i,j; i = k; while(i>0 && C[i]==n-k+i) i--; if(i>0) { C[i]= C[i]+1; for(j=i+1; j<=k; j++) C[j]=C[i]+j-i; } else Stop = TRUE; } void Combination(void){ Stop=FALSE; while (!Stop){ Result(); Next_Combination(); } } void main(void){ clrscr(); Init();Combination();getch(); } 57Chương 3: Bài toán liệt kê Ví dụ 3. Liệt kê các hoán vị của tập n phần tử. Cho X = { 1, 2,.., n }. Hãy liệt kê các hoán vị từ n phần tử của X. Giải: Mỗi hoán vị từ n phần tử của X có thể biểu diễn bởi bộ có thứ tự n thành phần: a = (a1, a2,.., an) thoả mãn ai∈ X, i = 1, 2,.., n, ap≠ aq, p≠ q. Trên tập các hoán vị từ n phần tử của X có thể xác định nhiều thứ tự khác nhau. Tuy nhiên, thứ tự dễ thấy nhất là thứ tự từ điển được định nghĩa như sau: Ta nói hoán vị a = a1a2... an đi trước hoán vị a’ = a1’a2’...an’ trong thứ tự từ điển và ký hiệu là a<a’, nếu tìm được chỉ số k ( 1 ≤ k ≤ n ) sao cho: a1 = a1’, a2 = a2’,..., ak-1 = a’k-1, ak < a’k. Chẳng hạn X = { 1, 2, 3, 4}. Các hoán vị các phần tử của X được liệt kê theo thứ tự từ điển như sau: 1 2 3 4 3 1 2 4 1 2 4 3 3 1 4 2 1 3 2 4 3 2 1 4 1 3 4 2 3 2 4 1 1 4 2 3 3 4 1 2 1 4 3 2 3 4 2 1 2 1 3 4 4 1 2 3 2 1 4 3 4 1 3 2 2 3 1 4 4 2 1 3 2 3 4 1 4 2 3 1 2 4 1 3 4 3 1 2 2 4 3 1 4 3 2 1 Như vậy, hoán vị đầu tiên trong thứ tự từ điển là (1, 2, …, n) và hoán vị cuối cùng là (n, n-1,..., 1). Giả sử a = a1a2... an là một hoán vị chưa phải là cuối cùng. Khi đó ta có thể chứng minh được rằng, hoán vị kế tiếp trong thứ tự từ điển có thể xây dựng bằng cách thực hiện các qui tắc biến đổi sau đối với hoán vị hiện tại:  Tìm từ phải qua trái hoán vị có chỉ số j đầu tiên thoả mãn aj <aj+1(hay j là chỉ số lớn nhất để aj <aj+1);  Tìm ak là số nhỏ nhất còn lớn hơn aj trong các số ở bên phải aj;  Đổi chỗ aj với ak  Lật ngược đoạn từ aj+1 đến an. Chẳng hạn ta đang có hoán vị (3, 6, 2, 5, 4, 1), cần xây dựng hoán vị kế tiếp theo thứ tự từ điển. Ta duyệt từ j = n-1 sang bên trái để tìm j đầu tiên thoả mãn aj < aj+1 ta nhận được j = 3 58

Tài liệu liên quan

  • nghiên cứu số hóa đô thị thực nghiệm dùng trong các bài toán thiết kế tàu thủy nghiên cứu số hóa đô thị thực nghiệm dùng trong các bài toán thiết kế tàu thủy
    • 6
    • 607
    • 7
  • tính thông chỉnh của bài toán moment, kết luận tính thông chỉnh của bài toán moment, kết luận
    • 1
    • 245
    • 0
  • Bài toán chia kẹo Bài toán chia kẹo
    • 2
    • 1
    • 7
  • BÀI TOÁN LIỆT KÊ BÀI TOÁN LIỆT KÊ
    • 27
    • 3
    • 61
  • Nghiên cứu giải thuật di truyền ứng dụng vào giải một số bài toán thống kê Nghiên cứu giải thuật di truyền ứng dụng vào giải một số bài toán thống kê
    • 26
    • 575
    • 0
  • Bài giảng các chuyên đề: Bài toán liệt kê, Cấu trúc dữ liệu và giải thuật, Quy hoạch động, lý thuyết đồ thị. Bài giảng các chuyên đề: Bài toán liệt kê, Cấu trúc dữ liệu và giải thuật, Quy hoạch động, lý thuyết đồ thị.
    • 258
    • 1
    • 4
  • Phân lớp và tránh xung đột trong bài toán lập kế hoặch với thông tin không đầy đủ. pot Phân lớp và tránh xung đột trong bài toán lập kế hoặch với thông tin không đầy đủ. pot
    • 8
    • 420
    • 0
  • Thuật giải di truyền song song cho bài toán thiết kế mạch tổ hợp trên nhóm máy tính mạng. doc Thuật giải di truyền song song cho bài toán thiết kế mạch tổ hợp trên nhóm máy tính mạng. doc
    • 12
    • 396
    • 0
  • Bài toán thiết kế hệ thống khóa có mã hóa dựa trên các nguyên lý kỹ thuật số logic cơ bản (Sinh viên Nguyễn Việt Hùng) Bài toán thiết kế hệ thống khóa có mã hóa dựa trên các nguyên lý kỹ thuật số logic cơ bản (Sinh viên Nguyễn Việt Hùng)
    • 30
    • 785
    • 6
  • BÀI TOÁN THIẾT KẾ CHÂN VỊT doc BÀI TOÁN THIẾT KẾ CHÂN VỊT doc
    • 9
    • 1
    • 12

Tài liệu bạn tìm kiếm đã sẵn sàng tải về

(435.39 KB - 27 trang) - BÀI TOÁN LIỆT KÊ Tải bản đầy đủ ngay ×

Từ khóa » Thuật Toán Liệt Kê