Professional Documents
Culture Documents
Bài tập 3.1. Dãy Lucas được định nghĩa bởi Ln = Ln-1+ Ln-2 với L0 = 2, L1 = 1.
Hãy viết hàm tính số Lucas thứ n.
Source code:
/*
Bài tập 3.1. Dãy Lucas được định nghĩa bởi Ln = Ln-1+ Ln-2 với L0 = 2, L1 = 1.
Hãy viết hàm tính số Lucas thứ n.
*/
int lucas(int n) {
if(n<=1) return 2-n; //Tinh L0 va L1
return lucas(n-1) + lucas(n-2); //De quy de tinh Ln voi n>=2
}
Bài tập 3.2: Trên bàn cờ vua kích thước n*n có một quân mã đang ở ô (1, 1).
Hãy đưa ra một dãy các di chuyển của mã sao cho mỗi ô trên bàn cờ đều được đi
qua đúng 1 lần (ô (1, 1) được xem là đã đi qua).
Source code:
/*
Bài tập 3.2: Trên bàn cờ vua kích thước n*n có một quân mã đang ở ô (1, 1).
Hãy đưa ra một dãy các di chuyển của mã sao cho mỗi ô trên bàn cờ đều được đi
qua đúng 1 lần (ô (1, 1) được xem là đã đi qua).
*/
#include <iostream>
using namespace std;
int n;
int X[100], Y[100]; //# Lưu tọa độ các bước di chuyển của quân mã
int mark[100][100]; //# Đánh dấu vị trí các ô mà quân mã đã di chuyển qua
//# Mảng hx, hy mô tả 8 vị trí quân mã có thể di chuyển kể từ vị trí hiện tại
const int hx[] = {1, 1, 2, 2, -1, -1, -2, -2};
const int hy[] = {2, -2, 1, -1, 2, -2, 1, -1};
int main(){
cin >> n;
mark[1][1] = 1;
X[1] = Y[1] = 1;
TRY(2);
return 0;
}
Bài tập 3.3. Một người xuất phát tại thành phố 1, muốn đi thăm tất cả các thành
phố khác, mỗi thành phố đúng 1 lần và quay về 1. Chi phí để đi từ thành phố i sang
thành phố j là cij. Hãy tìm tổng chi phí nhỏ nhất có thể.
Source code:
/*
Bài tập 3.3. Một người xuất phát tại thành phố 1, muốn đi thăm tất cả các thành
phố khác, mỗi thành phố đúng 1 lần và quay về 1. Chi phí để đi từ thành phố i sang
thành phố j là cij. Hãy tìm tổng chi phí nhỏ nhất có thể.
*/
#include <bits/stdc++.h>
using namespace std;
#define MAX 100
}
}
}
int main() {
input();
x[1] = 1;
TRY(2);
cout << best;
return 0;
}
Bài tập 3.4. Cho dãy a có n phần tử. Một dãy con của a là dãy thu được bằng cách
xóa đi một số phần tử của a và giữ nguyên thứ tự các phần tử còn lại (có thể không
xóa phần tử nào). Hãy tìm dãy con tăng dài nhất của a.
Source code:
/*
Bài tập 3.4. Cho dãy a có n phần tử. Một dãy con của a là dãy thu được bằng cách
xóa đi một số phần tử của a và giữ nguyên thứ tự các phần tử còn lại (có thể không
xóa phần tử nào). Hãy tìm dãy con tăng dài nhất của a.
*/
#include <bits/stdc++.h>
using namespace std;
int a[1000], n;
int mem[1000]; //# mảng ghi nhớ lời giải các bài toán con đã được giải
void init(){
memset(mem, -1, sizeof(mem));
}
mem[0] = 1;
for(int i=1; i<n; i++){
mem[i] = 1;
for(int j=0; j<i; j++){
if(a[j] < a[i]){ //nếu a[j] < a[i]
if(mem[i] < mem[j] + 1){ //nếu mem[i] < mem[j] + 1
mem[i] = mem[j] + 1; //thì ta sẽ cộng 1 vào giá trị mem[j]
}
}
}
}
return mem[i]; //trả về mem[i]
}
int main(){
init();
cin >> n;
for(int i = 0; i < n; i++) cin >> a[i];
int res = 1, pos = 0;
for(int i = 1; i < n; i++){
if (res < lis(i)){
res = lis(i);
pos = i;
} //hàm for để xác định dãy tăng, đặt lại vị trí để đếm số lượng phần tử
}
cout << res << endl;
trace(pos);
return 0;
int main() {
int m;
cin >> m;
for (int n = 1; n <= m; ++n){
for (int k = 0; k <= n; ++k)
printf("%d ", binom(n, k)); // in ra n dòng hệ số tổ hợp
printf("\n");
}
for (int n = 1; n <= m; ++n){
for (int k = 0; k <= n; ++k)
printf("%d ", binom2(n, k));// in ra n dòng hệ số tổ hợp bằng cách 2
printf("\n");
}
return 0;
}
Bài tập 3.6. Tìm ước chung lớn nhất của hai số nguyên a, b cho trước.
Source code:
/*
Bài tập 3.6. Tìm ước chung lớn nhất của hai số nguyên a, b cho trước.
*/
#include <iostream>
int main() {
int a, b;
cin >> a >> b;
cout << gcd(a, b) << endl << gcd2(a, b); //in ra giá trị gcd
return 0;
}
Bài tập 3.7. Sử dụng phương pháp khử đệ quy bằng stack, hãy liệt kê các xâu nhị
phân độ dài n không có k bit 1 nào liên tiếp.
Source code:
/*
Bài tập 3.7. Sử dụng phương pháp khử đệ quy bằng stack, hãy liệt kê các xâu nhị
phân độ dài n không có k bit 1 nào liên tiếp.
*/
#include <bits/stdc++.h>
using namespace std;
struct state{
int i, j, old_L;
//# constructor
state(int _i = 0, int _j = 0, int _L = 0):
int main() {
int n, k;
cin >> n >> k;
int x[n+1];
stack<state> s;
//# number of consecutive suffix 1
int L = 0;
s.push(state(1, 0));
while (!s.empty()){
state &top = s.top();
//# if a new binary sequence is found
if (top.i > n){
for (int i = 1; i <= n; ++i)
cout << x[i] << " \n"[i == n];
s.pop();
continue;
}
if(L+1<k || top.j==0){
x[top.i] = top.j;//gán giá trị của top.i cho phần tử top.j
top.old_L = L;//lưu giá trị
if(top.j){ //nếu top.j khác 0
L = L + 1;//tăng giá trị của 0
} else {L = 0;}//nếu không, L = 0
s.push(state(top.i+1,0));
//thêm một trạng thái mới vào ngăn xếp, top.i tăng 1, top.j về 0
}
struct state{
int i, j, s;
state(int _i = 0, int _j = 0, int _s=0): i(_i), j(_j), s(_s){}
};//khai báo cấu trúc state để xử lý các đối tượng
int main() {
int n, M;
cin >> n >> M; //nhập các giá trị đầu vào của các cân
int m[n+1];//khai báo mảng m có n + 1 giá trị
for (int i = 1; i <= n; ++i) cin >> m[i];
int x[n+1];
stack<state> s;
//# sum of selected weights
int sum = 0;
s.push(state(1, -1, -m[1])); //đưa vào ngăn xếp state để bắt đầu duyệt
s.push(state(1, 0, 0)); //đưa vào ngăn xếp một state khác để tiếp tục duyệt
s.push(state(1, 1, m[1]));//đưa vào ngăn xếp một state khác để tiếp tục duyệt
while (!s.empty()){ //lặp đến khi ngăn xếp trống
state top = s.top();//lấy phần tử đầu tiên của ngăn xếp gắn cho top
if (top.i >= n){ //kiểm tra độ lớn của top.i
if (top.s == M){
for (int i = 1; i <= n; ++i){
if (x[i] == -1) cout << '-' << m[i];//in ra kết quả của bài toán
if (x[i] == 1) cout << '+' << m[i];//in ra kết quả của bài toán
}
cout << "=" << M;
exit(0);
}
s.pop();
continue; //nếu vòng lặp đúng, loại bỏ phần tử ở trên đỉnh, và tiếp tục
}
return 0;
}
void inputData(){
cin >> n >> k1 >> k2;//nhập dữ liệu k1, k2 là số ngày nghỉ cho phép
}
return true;
}
}
void solution(){
/*
khai báo int so1 = 0;
chạy for cho đến khi so1 khác 0 và nhỏ hơn k1
*/
if(a==n) solution();
else TRY(a+1);
so1 = pre;
}
}
}
int main(){
inputData();
TRY(1);
//cout << cnt % du;
return 0;
}
void input(){
void solution(){
if(checkHamming(x,S) == H){ //nếu khoảng cách Hamming bằng H
for(int i=0; i<N; i++)
cout << x[i];//liệt kê tất cả các xâu thoả mãn, theo thứ tự từ điển
cout << endl;
}
}
int main(){
int T;
cin >> T;
while(T > 0){
input();
TRY(0);//Bắt đầu TRY phần tử đầu tiên với vòng while T>0
T--;
}
}
/*
Bài tập 3.11. Lịch trình chụp ảnh
Superior là một hòn đảo tuyệt đẹp với n địa điểm chụp ảnh và các đường một chiều
nối các điểm chụp ảnh với nhau. Đoàn khách tham quan có r người với sở thích
chụp ảnh khác nhau. Theo đó, mỗi người sẽ đưa ra danh sách các địa điểm mà họ
muốn chụp. Bạn cần giúp mỗi người trong đoàn lập lịch di chuyển sao cho đi qua
các điểm họ yêu cầu đúng một lần, không đi qua điểm nào khác, bắt đầu tại điểm
đầu tiên và kết thúc tại điểm cuối cùng trong danh sáchmà họ đưa ra, và có tổng
khoảng cách đi lại là nhỏ nhất.
*/
#include<bits/stdc++.h>
using namespace std;
const int MAX = 10000;
int n, r;
int price[MAX][MAX];
int x[MAX];
bool visited[MAX];
vector<int> vt;
int min_price;
int sum_price;
int start, destination, numberOfPoint;
//Khai báo các biến cần thiết để đánh dấu giá tiền, tổng tiền, các địa điểm đã thăm,
điểm bắt đầu
void input(){
cin >> n >> r; //Nhập số lượng địa điểm và số lượng người
for(int i=0; i<n; i++)
for(int j=0; j<n; j++){
cin >> price[i][j]; //nhập dữ liệu về giá giữa hai địa điểm i và j
}
}
void solution(){
if(price[x[numberOfPoint-2]][destination] == 0) return;
min_price = min(min_price, sum_price + price[x[numberOfPoint-2]]
[destination]);
//kiểm tra và lưu trữ giá tiền nhỏ nhất dựa trên tổng số điểm và địa điểm
}
int main(){
string str;
input(); getline(cin,str);//input lịch di chuyển
while(r > 0){
min_price = INT_MAX;
sum_price = 0;
getline(cin, str);
while (!str.empty()){
stringstream convert(str.substr(0, str.find(" ")));
int tmp = 0;
convert >> tmp;
vt.push_back(tmp - 1);
} else {
str.erase(0, str.find(" ") + 1); //Cập nhật lịch di chuyển
}
}
//Khởi tạo các input cần thiết
start = vt[0]; // điểm bắt đầu
destination = vt[vt.size()-1]; // điểm đích
numberOfPoint = vt.size(); // số điểm cần đi qua
x[0] = start; x[numberOfPoint-1] = destination;
for(int i=0; i<n; i++)
visited[i] = false; //chạy thử đi qua các vị trí
TRY(1);
int index = 0;
for(int j=0; j<vt[x[a-1]].size(); j++){ //kiểm tra xem có cạnh nối giữa 2 đỉnh hay
không
if(i == vt[x[a-1]][j]) index++; //nếu có, index cộng 1
}
if(index == 0) return false; //không thì trả về false
return true;
}
void solution(){
res++; //khi thoả mãn, cộng 1 vào res, in ra solution
}
void TRY(int a){ //Thử tất cả các khả năng chọn đỉnh, chọn cạnh
for(int i=0; i<n; i++){
if(check(a, i)){
visited[i] = true;//nếu đã thăm đỉnh i
x[a] = i; //gán i vào x[a]
int main(){
input();
TRY(0); //thử bắt đầu từ đỉnh xuất phát
cout << res / 2; //in ra số lượng đường đi cần tìm
}