360-> 0을 넘을 때 각도의 추세를 계산

Aug 17 2020

두 개체의 상대적 위치를 설명하는 각도를 측정하는 변수가 있는데 (즉, 0-359 범위 일 수 있음) 이것이 시간이 지남에 따라 어떻게 변했는지 수량화하고 싶습니다.

예를 들어, 여기에 두 항목의 상대적 위치가 매년 1 도씩 변경됩니다.

year <- seq(1981, 2020)
angle <- c(seq(341, 359), seq(0, 20))

그러나 여기서 기울기를 취하는 것은 2000 년에 발생하는 "크로스 오버"로 인해 의미가 없습니다. 여러 가지 샘플이 있으며 일부는이 문제가 있고 일부는 그렇지 않습니다. 어떤 샘플에이 문제가 발생 하는지 , 크로스 오버가 언제 발생 하는지 사전에 알 수 없으므로 일종의 오프셋 (예 : 지난 20 년에 360을 추가)을 적용 할 수 없습니다.

0 = 360이라는 사실을 고려하여 각도 추세를 계산하는 허용 된 방법이 있습니까?

답변

1 whuber Aug 17 2020 at 23:00

각도를 생각해 $y$ 언제든지 $t$각도의 작은 변화의 축적으로. 상징적으로, 언제$f(t)$ 시간에 따른 각도의 변화율 $t$$t_0$ 관찰의 시작입니다.

$$y(t) = y(t_0) + \int_{t_0}^t f(t)\,\mathrm{d}t.$$

당신의 문제는 $y(t)$ 모듈로 기록되었습니다 $360$ 학위-아마도 약간의 오류가 있음 $\epsilon(t).$ 즉, 값만 관찰했습니다.

$$y^{*}(t) = y(t) + \epsilon(t) \mod 360.$$

그러나 재구성 할 수 있습니다. $y(t) + \epsilon(t)$충분히 자주 관찰하는 경우. 연속해서$t \lt s,$ 주의

$$y^{*}(s) - y^{*}(t) = y(s) - y(t) + \epsilon(s) - \epsilon(t) \mod 360 = \int_t^s f(t)\,\mathrm{d}t + \delta$$

어디 $\delta$ 오류의 기여도와 동일 $\epsilon(s)-\epsilon(t)$ 플러스, 아마도$360$ 사이에 각도가 끊길 때마다 $y^{*}(t)$$y^{*}(s).$이제 총 오류의 크기를 제공$|\epsilon(s)-\epsilon(t)|$ 보다 작다 $180$각이 한 번 이상 주위에 가지 않았다 제공은 중단이 발생했는지 여부를 우리가 알아낼 수있는 경우$|\epsilon(s)-\epsilon(t)| \gt 180,$ 더하기 또는 빼기 $360$$\delta$ 간격에 배치 $-180$ ...에 $+180$ 도.

이러한 오류를 직접 관찰 할 수는 없지만 증가 할만큼 자주 샘플링하는 경우 $y(t_i) - y(t_{i-1})$이 조정을 관찰 된 차이에 적용하기 만하면됩니다. 그러므로,

할때는 언제나 $|y^{*}(s)-y^{*}(t)| \gt 180,$ 더하기 또는 빼기 $360$$\delta$ 간격에 배치 $-180$ ...에 $+180$ 도.

동등하게, 차이를 모듈로 계산 $180$ 그러나 범위에서 표현하십시오 $-180$ ...에 $+180$ (전통적인 것과 같이) 범위보다 $0$ ...에 $360.$

조정 된 값을 부르 자 $\delta^{*}(t,s),$ 그래서

$$y^{*}(s) - y^{*}(t) = \int_t^s f(t)\,\mathrm{d}t + \delta(t,s)^{*}.$$

이것은 평등 이지 평등 모듈로가 아닙니다.$360.$ 이제 모듈로 각도를 기록하는 효과를 제거 할 수 있습니다. $360$조정 된 차이를 합산하여 때때로 관찰이 이루어질 때$t_0 \lt t_1\lt \cdots \lt t_n,$ 우리는

$$\begin{aligned} y^{*}(t_i) &= y^{*}(t_0) + \left[y^{*}(t_1) - y^{*}(t_0)\right] + \cdots + \left[y^{*}(t_i) - y^{*}(t_{i-1})\right] \\ &=y(t_0) + \int_{t_0}^{t_i} f(t)\,\mathrm{d}t + \delta(t_0,t_1)^{*} + \delta(t_1,t_2)^{*} + \cdots + \delta(t_{i-1},t_i)^{*} \\ &= y(t_i) + \left[\epsilon(t_i) - \epsilon(t_0)\right]. \end{aligned}$$

계산 모듈로의 문제 $360$사라졌습니다 : 이제 원하는 절차를 사용하여 응답을 모델링 할 수 있습니다 .$y^{*}(t).$


다음은 상당히 어려운 데이터 세트의 그림입니다. 모델에 따라 데이터가 생성되었습니다.$y(t) = 30t \mod 360$1980 년부터 2020 년까지 매년 iid 표준 편차의 정규 분포 오차로 관찰$60$ 도 (다량).

추세는 원시 데이터에서 거의 식별 할 수 없지만 각도 조정 알고리즘은 시각적으로 정렬되었습니다. 예를 들어 최소 제곱 모델을 조정 된 데이터에 맞출 수 있습니다.

원시 데이터에 대한 확장 된 수직 스케일은 적합의 세부 사항과 그것과의 편차를 보여줍니다. 덧붙여서,이 예에서 기울기의 추정치는 다음과 같습니다.$28.0 \pm 0.74$ 학위, 실제 값과 크게 다르지 않습니다. $30$ 도 (이 비교에 대한 p- 값은 $1.1\%$).

오류의 표준 편차가 $\epsilon(t)$ 큼 (보다 큼 $180/2/\sqrt{2} \approx 64$대략적인 각도), 때로는 각도 조정이 올바르지 않습니다. 이것은 모델 잔차에 360도 정도의 값으로 갑작스러운 변화로 나타납니다. 따라서 모형 잔차에 대한 일상적인 분석을 통해 이러한 문제를 탐지 할 수 있으므로 더 나은 적합을 위해 수정 된 값을 수정할 수 있습니다. 자세한 내용은 모델과 피팅 절차에 따라 다릅니다.


R코드는 그림을 만들었습니다. "각도 조정"에서는 각도 조정을 효율적으로 계산할 수있는 방법을 보여줍니다.

#
# Specify the data-generation process.
#
year <- 1980:2020 # Dates to use
beta <- 30        # Annual rate of change
sigma <- 60       # Error S.D.
#
# Generate the data.
#
set.seed(17)
angle <- (year * beta + rnorm(length(year), 0, sigma)) %% 360
X <- data.frame(year, angle)
#
# Adjust the angles.
#
X$`total angle` <- with(X, { d <- (diff(angle) + 180) %% 360 - 180 cumsum(c(angle[1], d)) }) # # Fit a model to the adjusted angles. # fit <- lm(`total angle` ~ year, X) # # Analyze the fit. # b <- coefficients(fit) y.hat <- predict(fit) #--Compute dates the fit must wrap around from 360 to 0: y.breaks <- seq(floor(min(y.hat) / 360)*360, max(y.hat), by=360) year.breaks <- (y.breaks - b[1]) / b[2] #--Make the plots: u <- ceiling(max(X$`total angle`)/360)
par(mfcol=c(1,2))

#--The fits:
plot(X$year, X$angle, pch=19, ylim=c(0, 360), yaxp=c(0, 360, 4),
     col="gray", ylab="Angle (degrees)", xlab="Year",
     main="Raw Data and Fit")
for (x in year.breaks) 
  abline(c(-x * b[2], b[2]), col="Red", lwd=2)

plot(X$year, X$`total angle`, ylim=c(0,u*360),  yaxp=c(0, u*360, u),
     xlab="Year", ylab="Total angle",
     main="Adjusted Data and Fit")
abline(fit, col="Red", lwd=2)

#--The raw data:
plot(X$year, X$angle, ylim=c(0,u*360),  yaxp=c(0, u*360, u),
     pch=19, col="gray", ylab="Angle (degrees)", xlab="Year",
     main="Raw Data")

plot(X$year, X$`total angle`, ylim=c(0,u*360),
     yaxp=c(0, u*360, u),
     xlab="Year", ylab="Total angle",
     main="Adjusted Data")
par(mfcol=c(1,1))