13af6ab5fSopenharmony_ci/* 23af6ab5fSopenharmony_ci * Copyright (c) 2022-2024 Huawei Device Co., Ltd. 33af6ab5fSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License"); 43af6ab5fSopenharmony_ci * you may not use this file except in compliance with the License. 53af6ab5fSopenharmony_ci * You may obtain a copy of the License at 63af6ab5fSopenharmony_ci * 73af6ab5fSopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0 83af6ab5fSopenharmony_ci * 93af6ab5fSopenharmony_ci * Unless required by applicable law or agreed to in writing, software 103af6ab5fSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS, 113af6ab5fSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 123af6ab5fSopenharmony_ci * See the License for the specific language governing permissions and 133af6ab5fSopenharmony_ci * limitations under the License. 143af6ab5fSopenharmony_ci */ 153af6ab5fSopenharmony_ci 163af6ab5fSopenharmony_ciimport { PI } from "std/math"; 173af6ab5fSopenharmony_ci 183af6ab5fSopenharmony_ci// readonly expected : double = -1.3524862408537381; 193af6ab5fSopenharmony_ciclass Body { 203af6ab5fSopenharmony_ci readonly SOLAR_MASS : double = 4 * PI * PI; 213af6ab5fSopenharmony_ci x : double ; 223af6ab5fSopenharmony_ci y : double ; 233af6ab5fSopenharmony_ci z : double ; 243af6ab5fSopenharmony_ci vx : double ; 253af6ab5fSopenharmony_ci vy : double ; 263af6ab5fSopenharmony_ci vz : double ; 273af6ab5fSopenharmony_ci mass : double ; 283af6ab5fSopenharmony_ci public constructor(x : double, y : double, z : double, vx : double, vy : double, vz : double, mass : double) { 293af6ab5fSopenharmony_ci this.x = x; 303af6ab5fSopenharmony_ci this.y = y; 313af6ab5fSopenharmony_ci this.z = z; 323af6ab5fSopenharmony_ci this.vx = vx; 333af6ab5fSopenharmony_ci this.vy = vy; 343af6ab5fSopenharmony_ci this.vz = vz; 353af6ab5fSopenharmony_ci this.mass = mass; 363af6ab5fSopenharmony_ci } 373af6ab5fSopenharmony_ci 383af6ab5fSopenharmony_ci public offsetMomentum(px : double, py : double, pz : double): Body { 393af6ab5fSopenharmony_ci this.vx = -px / this.SOLAR_MASS; 403af6ab5fSopenharmony_ci this.vy = -py / this.SOLAR_MASS; 413af6ab5fSopenharmony_ci this.vz = -pz / this.SOLAR_MASS; 423af6ab5fSopenharmony_ci return this; 433af6ab5fSopenharmony_ci } 443af6ab5fSopenharmony_ci} 453af6ab5fSopenharmony_ci 463af6ab5fSopenharmony_ciclass NBodySystem { 473af6ab5fSopenharmony_ci bodies : Body[] ; 483af6ab5fSopenharmony_ci public constructor(bodies : Body[]) { 493af6ab5fSopenharmony_ci this.bodies = bodies; 503af6ab5fSopenharmony_ci let px : double = 0.0; 513af6ab5fSopenharmony_ci let py : double = 0.0; 523af6ab5fSopenharmony_ci let pz : double = 0.0; 533af6ab5fSopenharmony_ci let size : int = this.bodies.length; 543af6ab5fSopenharmony_ci for (let i : int = 0; i < size; i++) { 553af6ab5fSopenharmony_ci let b : Body = this.bodies[i]; 563af6ab5fSopenharmony_ci let m : double = b.mass; 573af6ab5fSopenharmony_ci px += b.vx * m; 583af6ab5fSopenharmony_ci py += b.vy * m; 593af6ab5fSopenharmony_ci pz += b.vz * m; 603af6ab5fSopenharmony_ci } 613af6ab5fSopenharmony_ci this.bodies[0].offsetMomentum(px, py, pz); 623af6ab5fSopenharmony_ci } 633af6ab5fSopenharmony_ci 643af6ab5fSopenharmony_ci public advance(dt : double): void { 653af6ab5fSopenharmony_ci let dx : double ; 663af6ab5fSopenharmony_ci let dy : double ; 673af6ab5fSopenharmony_ci let dz : double ; 683af6ab5fSopenharmony_ci let distance : double ; 693af6ab5fSopenharmony_ci let mag : double ; 703af6ab5fSopenharmony_ci let size : int = this.bodies.length; 713af6ab5fSopenharmony_ci for (let i : int = 0; i < size; i++) { 723af6ab5fSopenharmony_ci let bodyi : Body = this.bodies[i]; 733af6ab5fSopenharmony_ci for (let j : int = i + 1; j < size; j++) { 743af6ab5fSopenharmony_ci let bodyj : Body = this.bodies[j]; 753af6ab5fSopenharmony_ci dx = bodyi.x - bodyj.x; 763af6ab5fSopenharmony_ci dy = bodyi.y - bodyj.y; 773af6ab5fSopenharmony_ci dz = bodyi.z - bodyj.z; 783af6ab5fSopenharmony_ci distance = sqrt(dx * dx + dy * dy + dz * dz); 793af6ab5fSopenharmony_ci mag = dt / (distance * distance * distance); 803af6ab5fSopenharmony_ci bodyi.vx -= dx * bodyj.mass * mag; 813af6ab5fSopenharmony_ci bodyi.vy -= dy * bodyj.mass * mag; 823af6ab5fSopenharmony_ci bodyi.vz -= dz * bodyj.mass * mag; 833af6ab5fSopenharmony_ci bodyj.vx += dx * bodyi.mass * mag; 843af6ab5fSopenharmony_ci bodyj.vy += dy * bodyi.mass * mag; 853af6ab5fSopenharmony_ci bodyj.vz += dz * bodyi.mass * mag; 863af6ab5fSopenharmony_ci } 873af6ab5fSopenharmony_ci } 883af6ab5fSopenharmony_ci for (let i : int = 0; i < size; i++) { 893af6ab5fSopenharmony_ci let body : Body = this.bodies[i]; 903af6ab5fSopenharmony_ci body.x += dt * body.vx; 913af6ab5fSopenharmony_ci body.y += dt * body.vy; 923af6ab5fSopenharmony_ci body.z += dt * body.vz; 933af6ab5fSopenharmony_ci } 943af6ab5fSopenharmony_ci } 953af6ab5fSopenharmony_ci public energy(): double { 963af6ab5fSopenharmony_ci let dx : double ; 973af6ab5fSopenharmony_ci let dy : double ; 983af6ab5fSopenharmony_ci let dz : double ; 993af6ab5fSopenharmony_ci let distance : double ; 1003af6ab5fSopenharmony_ci let e : double = 0.0; 1013af6ab5fSopenharmony_ci let size : int = this.bodies.length; 1023af6ab5fSopenharmony_ci for (let i : int = 0; i < size; i++) { 1033af6ab5fSopenharmony_ci let bodyi : Body = this.bodies[i]; 1043af6ab5fSopenharmony_ci e += 0.5 * bodyi.mass * (bodyi.vx * bodyi.vx + bodyi.vy * bodyi.vy + bodyi.vz * bodyi.vz); 1053af6ab5fSopenharmony_ci for (let j : int = i + 1; j < size; j++) { 1063af6ab5fSopenharmony_ci let bodyj : Body = this.bodies[j]; 1073af6ab5fSopenharmony_ci dx = bodyi.x - bodyj.x; 1083af6ab5fSopenharmony_ci dy = bodyi.y - bodyj.y; 1093af6ab5fSopenharmony_ci dz = bodyi.z - bodyj.z; 1103af6ab5fSopenharmony_ci distance = sqrt(dx * dx + dy * dy + dz * dz); 1113af6ab5fSopenharmony_ci e -= (bodyi.mass * bodyj.mass) / distance; 1123af6ab5fSopenharmony_ci } 1133af6ab5fSopenharmony_ci } 1143af6ab5fSopenharmony_ci return e; 1153af6ab5fSopenharmony_ci } 1163af6ab5fSopenharmony_ci } 1173af6ab5fSopenharmony_ci 1183af6ab5fSopenharmony_ciexport class AccessNBody { 1193af6ab5fSopenharmony_ci static readonly SOLAR_MASS : double = 4 * PI * PI; 1203af6ab5fSopenharmony_ci static readonly DAYS_PER_YEAR : double = 365.24; 1213af6ab5fSopenharmony_ci n1 : int = 3; 1223af6ab5fSopenharmony_ci n2 : int = 24; 1233af6ab5fSopenharmony_ci 1243af6ab5fSopenharmony_ci static jupiter(): Body { 1253af6ab5fSopenharmony_ci return new Body(4.84143144246472090e+00, -1.16032004402742839e+00, -1.03622044471123109e-01, 1.66007664274403694e-03 * AccessNBody.DAYS_PER_YEAR, 7.69901118419740425e-03 * AccessNBody.DAYS_PER_YEAR, -6.90460016972063023e-05 * AccessNBody.DAYS_PER_YEAR, 9.54791938424326609e-04 * AccessNBody.SOLAR_MASS); 1263af6ab5fSopenharmony_ci } 1273af6ab5fSopenharmony_ci static saturn(): Body { 1283af6ab5fSopenharmony_ci return new Body(8.34336671824457987e+00, 4.12479856412430479e+00, -4.03523417114321381e-01, -2.76742510726862411e-03 * AccessNBody.DAYS_PER_YEAR, 4.99852801234917238e-03 * AccessNBody.DAYS_PER_YEAR, 2.30417297573763929e-05 * AccessNBody.DAYS_PER_YEAR, 2.85885980666130812e-04 * AccessNBody.SOLAR_MASS); 1293af6ab5fSopenharmony_ci } 1303af6ab5fSopenharmony_ci static uranus(): Body { 1313af6ab5fSopenharmony_ci return new Body(1.28943695621391310e+01, -1.51111514016986312e+01, -2.23307578892655734e-01, 2.96460137564761618e-03 * AccessNBody.DAYS_PER_YEAR, 2.37847173959480950e-03 * AccessNBody.DAYS_PER_YEAR, -2.96589568540237556e-05 * AccessNBody.DAYS_PER_YEAR, 4.36624404335156298e-05 * AccessNBody.SOLAR_MASS); 1323af6ab5fSopenharmony_ci } 1333af6ab5fSopenharmony_ci static neptune(): Body { 1343af6ab5fSopenharmony_ci return new Body(1.53796971148509165e+01, -2.59193146099879641e+01, 1.79258772950371181e-01, 2.68067772490389322e-03 * AccessNBody.DAYS_PER_YEAR, 1.62824170038242295e-03 * AccessNBody.DAYS_PER_YEAR, -9.51592254519715870e-05 * AccessNBody.DAYS_PER_YEAR, 5.15138902046611451e-05 * AccessNBody.SOLAR_MASS); 1353af6ab5fSopenharmony_ci } 1363af6ab5fSopenharmony_ci static sun(): Body { 1373af6ab5fSopenharmony_ci return new Body(0.0, 0.0, 0.0, 0.0, 0.0, 0.0, AccessNBody.SOLAR_MASS); 1383af6ab5fSopenharmony_ci } 1393af6ab5fSopenharmony_ci readonly expected : double = -1.3524862408537381; 1403af6ab5fSopenharmony_ci public run(): void { 1413af6ab5fSopenharmony_ci let ret : double = 0; 1423af6ab5fSopenharmony_ci for (let n : int = this.n1; n <= this.n2; n *= 2) { 1433af6ab5fSopenharmony_ci let bodies : NBodySystem = new NBodySystem([AccessNBody.sun(), AccessNBody.jupiter(), AccessNBody.saturn(), AccessNBody.uranus(), AccessNBody.neptune()]); 1443af6ab5fSopenharmony_ci let max : int = n * 100; 1453af6ab5fSopenharmony_ci ret += bodies.energy(); 1463af6ab5fSopenharmony_ci for (let i : int = 0; i < max; i++) { 1473af6ab5fSopenharmony_ci bodies.advance(0.01); 1483af6ab5fSopenharmony_ci } 1493af6ab5fSopenharmony_ci ret += bodies.energy(); 1503af6ab5fSopenharmony_ci } 1513af6ab5fSopenharmony_ci 1523af6ab5fSopenharmony_ci assert ret == this.expected: "Incorrect result"; 1533af6ab5fSopenharmony_ci } 1543af6ab5fSopenharmony_ci} 1553af6ab5fSopenharmony_ci 1563af6ab5fSopenharmony_cifunction main(): void { 1573af6ab5fSopenharmony_ci let a = new AccessNBody; 1583af6ab5fSopenharmony_ci a.run(); 1593af6ab5fSopenharmony_ci} 160