Nang Luong Vu Tru

 avatar
quoc14
c_cpp
5 months ago
5.9 kB
4
Indexable
PrimDijktra
Level 4
Năng Lượng Vũ Trụ
Năng Lượng Vũ Trụ

Vào năm 2900, khoa học kỹ thuật phát triển, con người có thể sinh sống trên nhiều hành tinh khác nhau. 

Do nhu cầu sử dụng năng lượng tăng cao, năng lượng trên mỗi hành tinh không còn đủ để đáp ứng, vì vậy, một bước đột phá khoa học năng lượng, 
con người có thể khai thác nguồn năng lượng lớn từ một ngôi sao trung tâm.

Năng lượng khai thác được từ ngôi sao trung tâm này sẽ được truyền tải đến tất cả các hành tinh, và các hành tinh cũng có thể truyền tải qua các hành tinh khác.

Khoảng cách giữa chúng là K, khoảng cách càng xa, lượng hao phí năng lượng càng lớn.

Ví dụ:  hành tinh số 1 cách hành tinh số 2 là K = 1000, thì số năng lượng hao hụt cũng sẽ là 1000.

Vì thế, cần tối ưu hoá khoảng cách nhất có thể tới tất cả các hành tinh. (như hình dưới)





Để tiếp tục giảm mức hao phí này, người ta tạo ra một máy truyền tải năng lượng, tại mỗi hành tinh, người ta đặt một máy truyền tải năng lượng này, 
có thể làm giảm đi phần lớn số hao phí, và nó tuân theo công thức của máy truyền tải:

K = A.H^3 + B.H^2 + C.

Trong đó:
-	K là khoảng cách giữa 2 hành tinh
-	A, B, C là hằng số của máy truyền tải
-	H là hao phí. (Luôn dương)

Ví dụ: Khao phí = 100, A = 3, B = 2, C = 1
Áp dụng công thức: K = A.H^3 + B.H^2 + C.
<=> 100  = 3.H^3 + 2.H^2 + 1.
=> H hao phí giữa 2 hành tinh là: 3

[Bài toán]
Hãy tính tổng hao phí nhỏ nhất khi truyền tải năng lượng tới toàn bộ hành tinh
Tất cả quá trình tính toán sẽ được làm tròn tới 10^-6  sau dấu thập phân.
Kết quả in ra sẽ lấy 10^-3 sau dấu thập phân

[Giới Hạn]
-	Số lượng hành tinh N <= 50
-	Khoảng cách K là số nguyên và đảm bảo rằng tổng của K < 2^64
-	A,B,C <= 1000 và là số nguyên

[Input]
-	Dòng đầu tiên T biểu thị số trường hợp thử nghiệm
-	Với mỗi trường hợp thử nghiệm:
   •	Dòng đầu tiên gồm giá trị N là số lượng hành tinh (N0 là ngôi sao chủ)
   •	Dòng tiếp theo là 3 số A B C
   •	N dòng tiếp theo là ma trận kề biểu bị khoảng cách của các hành tinh (Lưu ý, hành tinh không kết nối với chính nó)

[Output]
-	In “#T” cho phép thử thứ T, tiếp theo là một khoảng cách và in câu trả lời (T là số thứ tự của phép thử bắt đầu từ 1).

[Input]
50
3
3 2 1
0 100 100
100 0 100
100 100 0
4
10 1 4
0 452 995 613 
452 0 302 173 
995 302 0 309 
613 173 309 0 
4
9 6 2
0 351 912 720 
351 0 251 107 
912 251 0 941 
720 107 941 0 
7
1 9 9
0 604 571 668 261 511 546 
604 0 435 979 223 356 602 
571 435 0 842 482 759 109 
668 979 842 0 972 306 438 
261 223 482 972 0 988 468 
511 356 759 306 988 0 601 
546 602 109 438 468 601 0 
7
6 6 4
0 766 956 661 962 422 999 
766 0 203 784 105 890 887 
956 203 0 157 409 329 460 
661 784 157 0 928 733 707 
962 105 409 928 0 315 817 
422 890 329 733 315 0 467 
999 887 460 707 817 467 0

[Output]
#1 6.000
#2 9.119
#3 8.060
#4 26.371
#5 18.983


#1 6.000
#2 9.119
#3 8.060
#4 26.371
#5 18.983
#6 29.117
#7 18.328
#8 30.336
#9 22.253
#10 33.033
#11 11123563.825
#12 30373854.103
#13 7567499.915
#14 11797034.099
#15 8463058.600
#16 2439733.779
#17 16178946.582
#18 9403868.822
#19 6891839.275
#20 7824343.076
#21 8766828.461
#22 7565958.813
#23 6602238.269
#24 11277332.021
#25 4449955.050
#26 7952083.534
#27 3242380.583
#28 4534051.931
#29 2589291.416
#30 3826996.791
#31 3513981.974
#32 6257813.395
#33 7217559.068
#34 8888707.143
#35 5783796.235
#36 10532357.478
#37 5135140.983
#38 10406849.351
#39 4593651.588
#40 5006037.684
#41 15247286.743
#42 11474689.224
#43 8663165.338
#44 10914022.383
#45 7609414.126
#46 12268572.649
#47 14053342.152
#48 13705856.786
#49 10789210.075
#50 18318118.359
Time: 1.03800 s.

#include <iostream>
#include <time.h>

using namespace std;

int oo = 2000000000;

int T, n, a, b, c;
int vs[51];
unsigned long long mp[50][50], minn, maxx;
double result;

double Abs(double a) {return (a > 0 ? a : -a);}


double calcEnergy(double k){
	double l = 1.0, r = 100000000.0, m, calc;
	while(l < r){
		m = (l + r) / 2.0;
		calc = m*m*m*a*1.0 + m*m*b*1.0 + c*1.0;
		if(Abs(calc - k) < k*0.00000000001){
			return m;
		} else if(calc > k) {
			r = m*1.0;
		} else{
			l = m*1.0;
		}
	}
}

void prim(){
	int selected = 1;
	vs[0]++;
	while(selected < n){
		double minV = oo * 1.0;
		int minI = -1;
		for(int i = 0; i < n; i++){
			if(vs[i] == 1){
				for(int j = 0; j < n; j++){
					if(vs[j] == 0){
						double h = calcEnergy(1.0*mp[i][j]);
						if(h < minV){
							minV = h;
							minI = j;
						}
					}
				}
			}
		}
		vs[minI]++;
		selected++;
		result += minV;
	}
}

int main(){
	// read input
	freopen("input.txt", "r", stdin);

	//calc time
	clock_t tStart, tStop;
	tStart = clock();

	cin >> T;
	
	for(int tc = 1; tc <= T; tc++){
		// input and initial
		minn = -1, maxx = -1;
		cin >> n;
		cin >> a >> b >> c;
		for(int i = 0; i < n; i++){
			for(int j = 0; j < n; j++){
				cin >> mp[i][j];
			}
			vs[i] = 0;
		}
		result = 0.0;

		// solve problem
		prim();

		// output
		cout.setf(ios::fixed);
		cout.precision(3);
		cout << "#" << tc << " " << result << endl;
	}

	//calc time
	tStop = clock();
	cout.setf(ios::fixed);
	cout.precision(5);
	cout << "Time: " << double(tStop - tStart) / CLOCKS_PER_SEC << " s." << endl;

	return 0;
}
Leave a Comment