Nang Luong Vu Tru
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