[Из песочницы] Моделируем вселенную: небесная механика наглядно

qrsqrtig-au84ql57mqzfsvfz4c.jpeg

Давайте представим, что нам нужно запустить футбольный мяч на орбиту Земли. Никакие ракеты не нужны! Хватит горы, высотой 100 километров и недюжинной силы. Но насколько сильно нужно пнуть мяч, чтобы он никогда больше не вернулся на Землю? Как отправить мяч в путешествие к звёздам, имея только грубую силу и знание небесной механики?

Сегодня в программе:

  • Бесконечные возможности одной формулы
  • Как взять энергию у Юпитера
  • Откуда у планет берутся кольца
  • Как математика помогла открыть Нептун


Благо, мы живём в век компьютерных технологий. Нам не нужно забираться на высокую гору и пинать мяч со всей силы, всё можно смоделировать! Давайте приступим.

Одна формула


Та самая, известная с уроков физики и астрономии:

ndzgmjxjc0wp_f9mesuk0msgmeo.png

Показывает, насколько сильно будут взаимодействовать тела, в зависимости от их масс, расстояния между ними и гравитационной постоянной G.

Я написал программу, в которой можно расставлять шарики, взаимодействующие друг с другом силами гравитации, при этом у каждого шарика есть своя масса, скорость и координаты. Для наглядности шарики оставляют за собой след.

Давайте поставим большой и массивный голубой шар (Землю) и маленький красный мячик недалеко от него. Запускаем симуляцию:

bdc9g74bhavrklo1fn9nimxf1ha.png

Он упал!

Для выхода на орбиту нужна скорость, чтобы шарик падал и все время промахивался мимо Земли. Но КАКАЯ скорость? И снова школьные знания приходят на помощь:

Минимальная скорость, необходимая для выхода на орбиту Земли называется первой комической скоростью.

Для Земли она равна 7.91 км/с. А для симуляции её можно легко вычислить:

stttsvj38vrmajpgoau213lohts.jpeg

Разгоняем мячик и смотрим результат:

kwwosxhyppp5qmnuhr0c5b-1pnw.png

Полёт нормальный!

Шарик описывает окружность с Землёй в центре. Что будет, если придать ему чуть больше скорости? Сейчас проверим:

odinumvzny6ywgxhdsq16idxigo.png

Теперь форма орбиты эллиптическая, можно выделить 2 очень важные точки — апогей и перигей.

Апогей — это точка, в которой мячик максимально удалён от Земли.

Перигей — наоборот, самая близкая к Земле точка.

При увеличении начальной скорости перигей не меняется, а вот апогей становится всё дальше, и в конце концов имеет бесконечное расстояние до Земли. Тут мы вплотную приблизились к понятию второй космической скорости. Это скорость, которую надо придать шарику, чтобы он преодолел гравитацию Земли и улетел бороздить просторы вселенной. Для земли она равна 11.2 км/с.

Интересный фокус: если мы умножим первую космическую скорость на √2, то получим вторую космическую.

Умножили. Запустили. Получили:

n2oap-wffotxeboxo2ezcu7acva.png

Он улетел безвозвратно! Кстати, теперь он имеет параболическую орбиту. А если запустить шарик ещё сильнее, получим гиперболу. Интересно получается, везде нас преследует математика.

При этом формула остаётся всё той же. Окружность превращается в эллипс, эллипс в параболу, а парабола в гиперболу из-за вытягивания орбиты (увеличения эксцентриситета).

Как взять энергию у Юпитера?


Давайте расширим нашу модель, добавим Солнце, заставим Землю крутиться вокруг него.

8svcvt5ubgpzguozlheom9o28uw.png

Представим, что мячу нужно придать такую скорость, чтобы он улетел за пределы Солнечной системы — третью космическую скорость. В реальном мире она равна 16.7 км/с. К сожалению, эта скорость слишком большая, боюсь, нам не хватит сил…

Постойте! А что, если забрать немного скорости у какого-нибудь массивного тела, например, Юпитера. Мы можем подлететь к чему-то очень массивному и совершить гравитационный манёвр. При пролёте мимо Юпитера силы гравитации взаимно притягивают мячик и газовый гигант, но масса мячика настолько мала, что почти никак не влияет на движение Юпитера, а сам Юпитер разгоняет пролетающее мимо тело до высоких скоростей.

Меньше слов — больше дела:

au021ho0j9l8arq8dn8qb4y9fj8.png

Момент гравитационного манёвра — шарик подлетел к Юпитеру.

vlzcsrmmjiydwlzbsteqebnjgck.png

Ура! Мы получили скорость, достаточную для выхода из Солнечной системы, при этом ничего не потратили. Правда, Юпитер стал двигаться чуть медленнее, но мы этого точно не заметим.

Все космические аппараты, запущенные человеком за пределы солнечной системы («Вояджеры» 1 и 2, «Пионеры» 10 и 11, «Новые горизонты») использовали именно такой способ для ускорения.

Увеличиваем масштаб!


Я добавил трение частиц, чтобы, сталкиваясь, они передавали часть энергии друг другу. Также я ввёл силу нормальной реакции, теперь частицы уважают своё личное пространство, отталкивая от себя других.

Поставим случайную генерацию шариков и зададим им случайное направление и скорость. Пусть их будет, допустим, 100 штук.

wb9anvwqfq8qfd_fvrvmycfrvvw.png

Полный хаос, каждая частица движется куда хочет, но всё же силы гравитации берут своё, и начинают образовываться скопления шариков:

msbmpbcbw6_rt83vav4sbxqnuey.png

И через некоторое время получается большое тело, состоящее из 99 шариков и один-единственный шарик, обращающийся вокруг него:

eijlnrddky0hdpzvviqlgwa7ibo.png

При другом запуске получилось следующее:

et-yfufpsujieyl8t8s3leh7m9y.png

Два массивных тела, обращающихся вокруг общего центра масс. Если представить, что эти два объекта — звёзды, то мы получили двойную звезду. Интересно, что примерно половина звёзд в нашей галактике — двойные. Если бы у нашего Солнца была звезда — компаньон, то в небе мы могли бы наблюдать такую картину:

lcjgneyjd4py4kgkzeoe_oskpns.jpeg

Откуда у планет берутся кольца?


Основная причина появления колец — это разрушение спутников, подлетевших слишком близко к планете, а точнее, пересёкших предел Роша. В таком случае приливные силы, вызываемые гравитацией планеты, становятся больше сил, удерживающих спутник целым, и он разрывается на много частей, оставляя после себя кольцо, которое опоясывает планету. Давайте смоделируем эту ситуацию:

gol9db01lgvjj-puclravymtvki.png

Спутник чуть дальше предела Роша, он вращается вокруг планеты по стабильной круговой орбите. Но что будет, если сгенерировать его чуть-чуть ближе к планете?

fw7zql9igjiz5qm11gap36ksetw.png

Спутник разлетелся на множество маленьких частей, которые образовали кольца вокруг планеты. Так же и в реальном мире. Тритон (спутник Нептуна) постепенно приближается к планете, и через 2 миллиарда лет будет разорван, а у Нептуна появятся кольца больше, чем у Сатурна.

Как открыли Нептун и при чём здесь математика?


Раз уж зашла речь о Нептуне, давайте поговорим о его открытии. «Планета, открытая на кончике пера» имеет массу, а значит, действует на объекты вокруг. Астрономы 19 века заметили изменения в орбите Урана, его орбита отличалась от расчётной, видимо, что-то влияло на него. Орбита Урана имела возмущения:

jrhszio9dizwno87wq-uuxjc2xw.png

Это утрированная модель показывает, как неизвестное тело за Ураном влияло на его орбиту. Астрономам оставалось только вычислить положение тайной планеты и посмотреть в телескоп. Действительно, планета Нептун оказалась именно там, где её и предсказывали!

ued7vyhfbawgo-0nqa8swbmyeva.png

Заключение


Конечно, эта симуляция не обобщает все законы и явления, происходящие в космосе, например, здесь не учитывается теория относительности Эйнштейна, так как скорость частиц далека от скорости света. Но есть ещё много интересных вещей, которые можно реализовать в этой симуляции. Попробуйте сами! Понадобится только Python3 и библиотека Pygame.

Исходный код
#Траектория частицы
Track = True

#длина траектории
Track_time = 5

#Гравитационная постоянная
G = 5

#Сила магнитного взаимодействия(одинаковые цвета -
#притягиваются, разные - отталкиваются)
MagnConst = 0

#Количество частиц
count = 100

#Начальная скорость частиц
kv = 6

#Случайная генерация частиц
RANDOM = True

#Радиус случайных частиц
r = 3

#Ширина и высота окна
WIN_WIDTH, WIN_HEIGHT = 900, 650


'''всё, что дальше, лучше не менять'''

#Закон для гравитационного взаимодействия
zg = 2

#Закон для магнитного взаимодействия
zm = 2

#Коэф. трения, чем он больше - тем меньше трение
k = 40

#Отталкивание частиц
antiG = 0.1

max_speed = 3

ResDist = 1

#Притяжение вниз
EarthG = 0

#Отражение частиц от стенок
Mirror = True

import pygame
from math import hypot, ceil, sqrt
from random import randint, random


def custom_pos():
    '''Здесь вы можете написать своё расположение планет'''
    '''не забудьте установить RANDOM = FALSE'''
    B.append(Ball(200, 300, YELLOW, r = 10, mass = 200, vx = 0.151))    #x, y, col, r, vx, vy, mass
    B.append(Ball(200, 50, GREEN, r = 6, mass = 10, vx = -(200 * G / 250)**0.5))
    

class Ball:
    def __init__(self, x, y, col, r = 4, vx = 0, vy = 0, mass = 4):
        self.x = x
        self.y = y
        self.r = r
        self.col = col
        self.vx = vx
        self.vy = vy
        self.mass = mass
        
    def move(self, Walls, WIN_WIDTH, WIN_HEIGHT, ofs_x, ofs_y):
        if Walls:
            x = self.x - ofs_x
            y = self.y - ofs_y
            if x <= 0 and self.vx < 0:
                if Mirror:
                    self.vx = -self.vx
                else:
                    self.x += WIN_WIDTH
                self.vx, self.vy = self.v_norm(self.vx, self.vy)
            if x >= WIN_WIDTH and self.vx > 0:
                if Mirror:
                    self.vx = -self.vx
                else:
                    self.x -= WIN_WIDTH
                self.vx, self.vy = self.v_norm(self.vx, self.vy)
            if y <= 0 and self.vy < 0:
                if Mirror:
                    self.vy = -self.vy
                else:
                    self.y += WIN_HEIGHT
                self.vx, self.vy = self.v_norm(self.vx, self.vy)
            if y >= WIN_HEIGHT and self.vy > 0:
                if Mirror:
                    self.vy = -self.vy
                else:
                    self.y -= WIN_HEIGHT
                self.vx, self.vy = self.v_norm(self.vx, self.vy)
            
        self.x += self.vx
        self.y += self.vy

        
    def force(self, ind, selfind):
        ox = B[ind].x
        oy = B[ind].y
        m = B[ind].mass
        if m < 0.01 and self.mass < 0.01:
            return 0
        r = B[ind].r
        vx = B[ind].vx
        vy = B[ind].vy
        dist = hypot(self.x - ox, self.y - oy)
        min_dist = (self.r + B[ind].r) * ResDist
        f = 0
        m_relative = self.mass / B[ind].mass
        if dist <= min_dist:
            newVx = (vx * m + self.vx * self.mass) / (m + self.mass)
            newVy = (vy * m + self.vy * self.mass) / (m + self.mass)
            self.vx = (newVx + k * self.vx) / (k + 1)
            B[ind].vx = (newVx + k * B[ind].vx) / (k + 1)
            self.vy = (newVy + k * self.vy) / (k + 1)
            B[ind].vy = (newVy + k * B[ind].vy) / (k + 1)
            f -= antiG * min(abs(min_dist - dist), min(m, self.mass) * 3)
        else:
            f += min(self.mass * B[ind].mass * G  / (dist ** zg), G / 10)
            mf = MagnConst * self.mass / (dist ** zm)
            if B[ind].col == B[selfind].col:
                mf = - mf
            f += mf
        fx = f * ((ox - self.x) / dist)
        fy = f * ((oy - self.y) / dist)
        ax = fx / self.mass
        ay = fy / self.mass
        self.vx += ax
        self.vy += ay + EarthG
        B[ind].vx -= ax * m_relative
        B[ind].vy -= ay * m_relative - EarthG

    @staticmethod
    def v_norm(vx, vy):
        v = hypot(vx, vy)
        if v > max_speed:
            vx = max_speed * (vx / v)
            vy = max_speed * (vy / v)
        return vx, vy


class Point:
    def __init__(self, x, y, col, r = 0, max_age = Track_time):
        self.age = 0
        self.x = x
        self.y = y
        self.col = col
        self.r = r
        self.max_age = max_age
    def vis(self, ofs_x, ofs_y):
        pygame.draw.circle(sc, self.col, (round(self.x - ofs_x),
                                          round(self.y - ofs_y)), self.r, 0)
        self.age += 1
        if self.age > self.max_age:
            T.remove(self)
        
def rand(count, WIN_WIDTH, WIN_HEIGHT):
    global kv
    B = []
    for i in range(count):
        m = r ** 2
        x = randint(0, WIN_WIDTH) + random()
        y = randint(0, WIN_HEIGHT) + random()
        vx = kv * randint(-100, 100) / 100
        vy = kv * randint(-100, 100) / 100
        col = Colors[randint(0, len(Colors) - 1)]
        B.append(Ball(x, y, col, r = r, vx = vx, vy = vy, mass = m))
    return B

def createBall(col, x, y, r = r, m = r):
    m = r
    B.append(Ball(x, y, col))

def get_offset(B):
    sum_x, sum_y = 0, 0
    m = 0
    for i in range(len(B)):
        sum_x += B[i].x * B[i].mass
        sum_y += B[i].y * B[i].mass
        m += B[i].mass
    if len(B) == 0:
        return 0, 0
    return sum_x / m, sum_y / m

def visBalls(B):
    for i in range(len(B)):
        pygame.draw.circle(sc, B[i].col, (round(B[i].x - ofs_x),
                                          round(B[i].y - ofs_y)), B[i].r, 0)
        T.append(Point(B[i].x, B[i].y, B[i].col))
        
FPS = 60
darkblue = (0, 2, 25)
ORANGE = (255, 200, 150)
RED = (255, 150, 150)
GREEN = (150, 255, 150)
BLUE = (150, 150, 255)
YELLOW = (255, 255, 0)
Colors = [RED, BLUE]#, GREEN]#, ORANGE]                       
pygame.init() 
clock = pygame.time.Clock()
sc = pygame.display.set_mode((WIN_WIDTH, WIN_HEIGHT))
sc.fill(darkblue)

maxF = 0.3
minv = 0.01
Walls = True
Collisions = True
Same = True
Check = False
tt = []

B = []
if RANDOM:
    B = rand(count, WIN_WIDTH, WIN_HEIGHT)
else:
    custom_pos()
    
Pause = False
delay = 0

if Track:
    T = []
for z in range(100000):
    sc = pygame.display.set_mode((WIN_WIDTH, WIN_HEIGHT))
    sc.fill(darkblue)
    ofs_x, ofs_y = get_offset(B)
    ofs_x -= WIN_WIDTH // 2
    ofs_y -= WIN_HEIGHT // 2
    for i in pygame.event.get():
        if i.type == pygame.QUIT:
            pygame.quit()
            quit()
        if i.type == pygame.KEYDOWN:
            if i.key == pygame.K_SPACE:
                Pause = not Pause
            elif i.key == pygame.K_w:
                WIN_HEIGHT += 10
                WIN_WIDTH += 10
            elif i.key == pygame.K_s:
                WIN_HEIGHT -= 10
                WIN_WIDTH -= 10
                
    pressed = pygame.mouse.get_pressed()
    pos = pygame.mouse.get_pos()
    x = pos[0]
    y = pos[1]
    if pressed[0] and delay < 0:
        delay = 20
        createBall(RED, x + ofs_x, y + ofs_y)
    if pressed[2] and delay < 0:
        delay = 20
        createBall(BLUE, x + ofs_x, y + ofs_y )
    delay -= 1
    
    if not Pause:
        for i in range(len(B)):
            for j in range(i + 1, len(B)):
                B[i].force(j, i)
        for i in range(len(B)):
            B[i].move(Walls, WIN_WIDTH, WIN_HEIGHT, ofs_x, ofs_y)
    for i in range(len(T)):
        try:
            T[i].vis(ofs_x, ofs_y)
        except IndexError:
            pass
    visBalls(B)
    
    pygame.display.update()
    clock.tick(FPS)


Надеюсь, эта статья оказалась познавательной для вас. Спасибо за внимание!

© Habrahabr.ru