Я пытаюсь реализовать this solution для расчета положения солнца в Swift3. Затем я переношу это в другую функцию, которая просто циклически проходит через день с полуночи, каждые 10 минут до 23:50.Позиция Солнца в Swift
Я действительно не понимаю R, и есть некоторые детали ответа, который я не полностью понимаю, в частности, что-то вроде функции if/clamp с квадратными скобками. Я сделал все возможное, сравнивая с версией Python, когда я запутался. В противном случае единственные различия связаны с использованием NSDate
, что упростило часть кода вверху.
Некоторые из значений, которые я возвращаю, кажутся правильными, и я могу увидеть основание кривой, когда я рисую результаты. Однако результат одного вызова, например, 7AM, а затем следующего, 7:10, сильно отличается.
Я сильно подозреваю, что я сделал что-то не так с зажимами, и что незначительные изменения в входах получают mod/trunced по-разному и качают выход. Но я не могу это заметить. Может ли кто-нибудь, кто понимает этот алгоритм, помочь?
Вот пример вывода я получаю:
2017-06-21 00:10:00 +0000 -16.0713262209521 31.7135341633943
2017-06-21 00:20:00 +0000 61.9971433936385 129.193513530349
2017-06-21 00:30:00 +0000 22.5263575559266 78.5445189561018
2017-06-21 00:40:00 +0000 29.5973897349096 275.081637736092
2017-06-21 00:50:00 +0000 41.9552795956374 262.989819486864
Как вы можете видеть, это качается дико между итерациями. Земля так не поворачивается! Мой код следующим образом, эта версия просто отправляет результаты в журнал:
class func julianDayFromDate(_ date: Date) -> Double {
let ti = date.timeIntervalSince1970
return ((ti/86400.0) + 2440587)
}
class func sunPath(lat: Double, lon: Double, size: CGSize) -> UIImage {
var utzCal = Calendar(identifier: .gregorian)
utzCal.timeZone = TimeZone(secondsFromGMT: 0)!
let year = utzCal.component(.year, from: Date())
let june = DateComponents(calendar: utzCal, year: year, month: 6, day: 21).date!
// now we loop for every 10 minutes (2 degrees) and plot those points
for time in stride(from:0, to:(24 * 60), by: 10) {
let calcdate = june.addingTimeInterval(Double(time) * 60.0)
let (alt, az) = sun(date: calcdate, lat: lat, lon: lon)
print(calcdate, alt, az)
}
class func sun(date: Date, lat: Double, lon: Double) -> (altitude: Double, azimuth: Double) {
// these come in handy
let twopi = Double.pi * 2
let deg2rad = Double.pi/180.0
// latitude to radians
let lat_radians = lat * deg2rad
// the Astronomer's Almanac method used here is based on Epoch 2000, so we need to
// convert the date into that format. We start by calculating "n", the number of
// days since 1 January 2000
let n = julianDayFromDate(date) - 2451545.0
// it continues by calculating the position in ecliptic coordinates,
// starting with the mean longitude of the sun in degrees, corrected for aberation
var meanlong_degrees = 280.460 + (0.9856474 * n)
meanlong_degrees = meanlong_degrees.truncatingRemainder(dividingBy: 360.0)
// and the mean anomaly in degrees
var meananomaly_degrees = 357.528 + (0.9856003 * n)
meananomaly_degrees = meananomaly_degrees.truncatingRemainder(dividingBy: 360.0)
let meananomaly_radians = meananomaly_degrees * deg2rad
// and finally, the eliptic longitude in degrees
var elipticlong_degrees = meanlong_degrees + (1.915 * sin(meananomaly_radians)) + (0.020 * sin(2 * meananomaly_radians))
elipticlong_degrees = elipticlong_degrees.truncatingRemainder(dividingBy: 360.0)
let elipticlong_radians = elipticlong_degrees * deg2rad
// now we want to convert that to equatorial coordinates
let obliquity_degrees = 23.439 - (0.0000004 * n)
let obliquity_radians = obliquity_degrees * deg2rad
// right ascention in radians
let num = cos(obliquity_radians) * sin(elipticlong_radians)
let den = cos(elipticlong_radians)
var ra_radians = atan(num/den)
ra_radians = ra_radians.truncatingRemainder(dividingBy: Double.pi)
if den < 0 {
ra_radians = ra_radians + Double.pi
} else if num < 0 {
ra_radians = ra_radians + twopi
}
// declination is simpler...
let dec_radians = asin(sin(obliquity_radians) * sin(elipticlong_radians))
// and from there, to local coordinates
// start with the UTZ sidereal time
let cal = Calendar.current
let h = Double(cal.component(.hour, from: date))
let m = Double(cal.component(.minute, from: date))
let f: Double
if h == 0 && m == 0 {
f = 0.0
} else if h == 0 {
f = 60.0/m
} else if h == 0 {
f = 24.0/h
} else {
f = (24.0/h) + (60.0/m)
}
var utz_sidereal_time = 6.697375 + 0.0657098242 * n + f
utz_sidereal_time = utz_sidereal_time.truncatingRemainder(dividingBy: 24.0)
// then convert that to local sidereal time
var localtime = utz_sidereal_time + lon/15.0
localtime = localtime.truncatingRemainder(dividingBy: 24.0)
var localtime_radians = localtime * 15.0 * deg2rad
localtime_radians = localtime.truncatingRemainder(dividingBy: Double.pi)
// hour angle in radians
var hourangle_radians = localtime_radians - ra_radians
hourangle_radians = hourangle_radians.truncatingRemainder(dividingBy: twopi)
// get elevation in degrees
let elevation_radians = (asin(sin(dec_radians) * sin(lat_radians) + cos(dec_radians) * cos(lat_radians) * cos(hourangle_radians)))
let elevation_degrees = elevation_radians/deg2rad
// and azimuth
let azimuth_radians = asin(-cos(dec_radians) * sin(hourangle_radians)/cos(elevation_radians))
// now clamp the output
let azimuth_degrees: Double
if (sin(dec_radians) - sin(elevation_radians) * sin(lat_radians) < 0) {
azimuth_degrees = (Double.pi - azimuth_radians)/deg2rad
} else if (sin(azimuth_radians) < 0) {
azimuth_degrees = (azimuth_radians + twopi)/deg2rad
} else {
azimuth_degrees = azimuth_radians/deg2rad
}
return (elevation_degrees, azimuth_degrees)
}
Hi Maury, я думаю, что смогу вам помочь, хотя я мог бы сделать это только в прямом R, это вам поможет? –
Это не помешает! Как я уже сказал, основной проблемой, с которой я столкнулся, является понимание синтаксиса [], и я не уверен, что я правильно его преобразовал. Фактически, если бы вы могли использовать R-версию и пройти через линии и отправить мне свои результаты, это, вероятно, будет особенно полезно! –
Я могу это сделать, я не уверен с вашим быстрым кодом, там указаны даты и координаты, которые вы должны использовать. Если вы скажете мне, что я считаю, что могу сделать это довольно быстро –