SPOJ.COM - Thuật toán bài SAMER08F - Feynman

Posted on November 5th, 2016

Đề bài:

Richard Phillips Feynman được biết đến như là một nhà vật lý và một người được nhận giải Nobel về vật lý. Ông làm việc trong ngành lý thuyết vật lý và tiên phong trong việc tính toán trường của lượng tử. Ông đã đến thăm Nam Mĩ được khoảng 10 tháng, giảng dạy và tận hưởng cuộc sống ở vùng chí tuyến này. Ông cũng được biết đến cho cuốn sách của mình "Surely You're Joking, Mr. Feynman!" và "What Do You Care What Other People Think?", bao gồm những cuộc phiêu lưu của ông ở vùng dưới xích đạo này.

Đam mê trong cuộc sống của ông là giải quyết và tạo ra những câu đố. Gần đây, một nông dân ở vùng Nam Mĩ, người là một ông chủ cho đến một nhà Vật lý năm 1949, đã tìm thấy một vài tờ giấy và những cuốn sổ ghi chú, cái được tin rằng là thuộc về Feynman. Ở giữa cuốn sổ, phần về vật lý Mezon và điện tử học, có một tờ giấy, ở đó Feynman đã ghi ra một câu đố đơn giản. Đó là: có bao nhiêu hình vuông khác nhau ở một cái lưới kích thước NxN.

Ở tờ giấy đó, có một hình vẽ, được chép lại như dưới đây, cho thấy với N = 2 thì đáp án là 5 hình vuông khác nhau.

spoj-com-thuat-toan-bai-samer08f-feynman-pic-thuattoan-phamvanlam-com

Đầu vào

Bao gồm nhiều test case. Mỗi test case là 1 dòng, chứa số N, biểu diễn số hình vuông ở mỗi cạnh, (1 ≤ N ≤ 100). Kết thúc của đầu vào là số 0.

Đầu ra

Với mỗi test case, in ra một dòng chứa số lượng hình vuông khác nhau tương ứng với mỗi test case.

Ví dụ

Đầu vào:

2
1
8
0

Đầu ra:

5
1
204

Bạn có thể tham khảo link gốc đề bài và submit code tại đây:  http://www.spoj.com/problems/SAMER08F/

Phân tích

  • Có thể sẽ có nhiều cách giải bài toán này. Ở đây mình đơn giản sử dụng thuật toán tham lam Greedy để giải quyết bài toán.

  • Yêu cầu của bài toán là tìm ra số hình vuông khác nhau ở trên lưới NxN. Dễ thấy rằng, kết quả sẽ là tổng số hình vuông có kích thước cạnh 1, cạnh 2, cạnh 3,... cạnh N.

  • Để đếm được hết tất cả các hình vuông mà không bị lặp, mình sẽ duyệt tất cả các ô trong lưới theo chiều từ trái sang phải, từ trên xuống dưới. Tại mỗi ô mình sẽ đếm số hình vuông có điểm bắt đầu là tại ô đó.

  • Trong ví dụ trên với N = 2:

    • Trước tiên mình xét điểm (0,0): khi đó mình có 2 hình vuông bắt đầu tại điểm này là: hình 1 và hình 5
    • Tiếp theo là điểm (0, 1): chỉ có 1 hình là hình 2
    • Xét điểm (1, 0): có 1 hình là hình 3
    • Xét điểm (0, 1): có 1 hình là hình 4.

    Như vậy, tổng là 5 hình.

  • Qua phân tích, mình suy ra một công thức đó là: tại điểm (row, col), số hình vuông có điểm bắt đầu tại đó là: N - Max(row, col). Bạn có thể thử lại nếu như còn thấy nghi ngờ về công thức này.

Lời giải

Bạn nên tự mình nghĩ ra thuật toán của bài toán trước khi tham khảo code của mình nhé. Hãy phát huy tối đa khả năng sáng tạo của bản thân. Hơn nữa code mình viết ra cũng chưa thật sự tối ưu. Nên rất mong nhận được sự chia sẻ của bạn.

Code C/C++

#include <iostream>
using namespace std;

int main()
{
	ios::sync_with_stdio(false);
	freopen("input.txt","r",stdin);

	// Kích thước lưới
	int N = 0;
	while(true)
	{
		cin >> N;
		if(N == 0) break;

		// Đếm số lượng hình vuông khác nhau.
		int num = 0;

		// Duyệt các điểm và đếm số hình vuông
		// có chứa ô đầu tiên nằm tại vị trí (row, col)
		for(int row = 0; row < N; row++ )
		{
			for(int col = 0; col < N; col++)
			{
				int max = (row > col) ? row : col;
				num += N - max;
			}
		}
		cout << num << endl;
	}
	return 0;
}

★ Nếu bạn thấy bài viết này hay thì hãy theo dõi mình trên Facebook để nhận được thông báo khi có bài viết mới nhất nhé:

Xin chào và hẹn gặp lại, thân ái!