本文翻譯自2018年最熱門的Python金融教程 Python For Finance: Algorithmic Trading。
本教程由以下五部分內容構成:
- Python金融入門
- 常見的金融分析方法
- 簡單的動量策略開發
- 回溯測試策略
- 評估交易策略
本文是該教程的第三部分。
既然你已對數據做了初步分析,那么是時候創建你的第一個交易策略了。但在深入研究之前,為什么不先來了解一些最常用的交易策略呢?下面簡短的介紹,無疑會讓你更容易開始交易策略的開發。
常見的交易策略
你應該還記得在本教程開篇的介紹中,我們講了交易策略是在市場上做多或做空的既定計劃,但不僅限于此,你還有更多的信息沒有真正掌握。一般來說,有兩種常見的交易策略:動量策略(momentum strategy)和回歸策略(reversion strategy)。
第一種是動量策略,也被稱為背離(divergence)或趨勢(trend)交易。當遵循該策略時,你相信數值會沿著當前的方向繼續移動。也就是說你相信股票具有慣性,即有上升或下降的趨勢,能被你發現并利用。
這一策略的實例包括移動均線交叉策略(moving average crossover)、雙均線交叉策略(dual moving average crossover)和海龜交易策略(turtle trading)。
- 移動均線交叉是指資產的價格從移動均線的一側移動到了另一側。這一交叉代表了動量的改變,可以作為進入或退出市場的決策點。在本教程的后面你會學到量化交易中使用該策略的入門級案例。
- 雙均線交叉發生在短期均線與長期均線交叉處。這一信號被用來識別在短期均線上的變化。買入信號發生在短期均線從下方上升超越長期均線時,而賣出信號則由短期均線從上方下降越過長期均線而觸發。
- 海龜交易是一個眾所周知的趨勢跟蹤策略,最初是由理查德·丹尼斯教授給大家的。它的基本策略是以20天高點買入期貨,并在20天低點賣出。
第二種是回歸策略,也被稱為收斂(convergence)或循環(cycle)交易。這一策略相信數值的運動最終會逆轉。這可能有點抽象,讓我們來舉個例子。在均值回歸策略(mean reversion strategy)中,你確信股價會回到它的均值,當它偏離均值時就可以利用這一點來做決策。
聽起來已經很實用了,對吧?
除了均值回歸策略以外,另一個例子是與之相似的配對交易均值回歸(pairs trading mean-reversion)。均值回歸策略的基本思想是股價會回到其均值,而配對交易策略對此進行了擴展,認為如果能識別出兩只高度相關的股票,當其中之一偏離了與另一方的相關性,兩者價格差的改變就能作為交易的信號。這意味著如果這兩只股票的相關性下降,那么可以做空價格較高的股票。高價格的股票最終會回到其均值,所以它應該被賣出。另一方面,做多低價格的股票,因為隨著相關性恢復正常,其價格將會上升。
除了這兩種最常用的策略,還有其他你偶爾可能會碰上的策略,比如預測策略,它試圖基于某些歷史因素預測股票在隨后一段時間內的方向和價值。還有高頻交易策略(HFT),它利用了亞毫秒市場的微觀結構來做決策。
這都是今后的樂章了,現在讓我們聚焦于開發你的第一個交易策略。
一個簡單的交易策略
正如上面所述,你將從量化交易的“hello world”:移動均線交叉策略開始。你將要開發的策略很簡單:對時間序列創建兩個獨立的簡單移動均值(Simple Moving Averages, SMA),它們具有不同的回望周期,比如40天和100天。如果短期移動均值超過了長期移動均值,那么就做多;如果長期移動均值超過了短期移動均值,那么就退出。
記住當做多時,你認為股價會上漲,并且將來會以更高的價格賣出(=買入信號);當賣空時,你賣出你的股票,希望將來以更低的價格買回并實現盈利(=賣出信號)。
當你剛開始的時候,這一簡單的策略可能看起來相當復雜。但是讓我們一步一步來:
- 首先,定義兩個不同的回望周期:一個短期窗口和一個長期窗口。設置兩個變量,并對每一個變量賦值一個整數。一定要讓短期窗口的值小于長期窗口的值。
- 接著,創建一個空的數據框
signals
,并確保復制了aapl
數據的索引,以便能開始計算aapl
數據每日的買賣信號。
- 在空的
signals
數據框中創建一列名為signal
的數據,并將其每一行的數值初始化為0.0
。
- 做好了準備工作后,是時候在各自的時間窗口上創建短期和長期的簡單移動均值了。使用
rolling()
函數開始滑動窗口計算:在該函數中,設置window
、min_period
和center
參數。在實際中,窗口大小window
分別設置為short_window
和long_window
。min_period
設為1
,作為窗口中觀測量的最小值。center
的值為False
,這樣標簽就不會設置在窗口的中心。接著,不要忘了在其后連接mean()
函數以便能計算滾動均值。
- 當計算了短期和長期窗口的均值后,你應該在短期移動均線與長期移動均線交叉處創建信號,但這僅針對大于最短移動均值窗口的時期。在Python中,這對應著如下條件不等式:
signals['short_mavg'][short_window:] > signals['long_mavg'][short_window:]
。注意加上[short_window:]
是為了遵守條件“僅針對大于最短移動均值窗口的時期”。當上述條件不等式為真時,signal
列中的初值0.0
將被替換為1.0
。 這樣一個“信號”就產生了。如果條件不等式為假,將繼續保留原始值0.0
,也沒有信號產生。使用 NumPy 中的where()
函數來設置這一條件不等式。就像你剛才讀到的那樣,將這一結果賦值給變量signals['signal'][short_window]
,因為我們僅想針對大于最短移動均值窗口的時期創建信號。
- 最后,計算信號的差值,從而生成實際的交易訂單。換句話說,在
signals
數據框的positions
這一列中,你將能區分多頭和空頭頭寸,無論是買入或賣出股票。
嘗試下方的代碼:
# 導入apple公司股票數據
import pandas_datareader as pdr
import datetime
aapl = pdr.get_data_yahoo('AAPL',
start=datetime.datetime(2006, 10, 1),
end=datetime.datetime(2012, 1, 1))
# 導入pandas,numpy
import pandas as pd
import numpy as np
# 初始化短期和長期窗口
short_window = 40
long_window = 100
# 初始化 `signals` 數據框,增加 `signal` 列
signals = pd.DataFrame(index=aapl.index)
signals['signal'] = 0.0
# 創建短期簡單移動均值
signals['short_mavg'] = aapl['Close'].rolling(window=short_window, min_periods=1, center=False).mean()
# 創建長期簡單移動均值
signals['long_mavg'] = aapl['Close'].rolling(window=long_window, min_periods=1, center=False).mean()
# 生成信號
signals['signal'][short_window:] = np.where(signals['short_mavg'][short_window:]
> signals['long_mavg'][short_window:], 1.0, 0.0)
# 生成交易命令
signals['positions'] = signals['signal'].diff()
# 輸出`signals`
print(signals)
signal short_mavg long_mavg positions
Date
2006-10-02 0.0 10.694285 10.694285 NaN
2006-10-03 0.0 10.638571 10.638571 0.0
2006-10-04 0.0 10.681905 10.681905 0.0
2006-10-05 0.0 10.683928 10.683928 0.0
2006-10-06 0.0 10.667714 10.667714 0.0
2006-10-09 0.0 10.666667 10.666667 0.0
2006-10-10 0.0 10.649184 10.649184 0.0
2006-10-11 0.0 10.625714 10.625714 0.0
2006-10-12 0.0 10.639683 10.639683 0.0
2006-10-13 0.0 10.647429 10.647429 0.0
2006-10-16 0.0 10.658701 10.658701 0.0
2006-10-17 0.0 10.654881 10.654881 0.0
2006-10-18 0.0 10.654286 10.654286 0.0
2006-10-19 0.0 10.699286 10.699286 0.0
2006-10-20 0.0 10.747429 10.747429 0.0
2006-10-23 0.0 10.803036 10.803036 0.0
2006-10-24 0.0 10.848655 10.848655 0.0
2006-10-25 0.0 10.894206 10.894206 0.0
2006-10-26 0.0 10.938797 10.938797 0.0
2006-10-27 0.0 10.966214 10.966214 0.0
2006-10-30 0.0 10.991088 10.991088 0.0
2006-10-31 0.0 11.017987 11.017987 0.0
2006-11-01 0.0 11.030621 11.030621 0.0
2006-11-02 0.0 11.041131 11.041131 0.0
2006-11-03 0.0 11.046857 11.046857 0.0
2006-11-06 0.0 11.059945 11.059945 0.0
2006-11-07 0.0 11.076296 11.076296 0.0
2006-11-08 0.0 11.101378 11.101378 0.0
2006-11-09 0.0 11.129113 11.129113 0.0
2006-11-10 0.0 11.153952 11.153952 0.0
... ... ... ... ...
2011-11-17 1.0 56.485857 54.946829 0.0
2011-11-18 1.0 56.381000 55.005257 0.0
2011-11-21 1.0 56.259000 55.052886 0.0
2011-11-22 1.0 56.177750 55.100386 0.0
2011-11-23 1.0 56.070536 55.125471 0.0
2011-11-25 1.0 55.974107 55.142343 0.0
2011-11-28 1.0 55.955536 55.169371 0.0
2011-11-29 1.0 55.950536 55.188643 0.0
2011-11-30 1.0 55.985179 55.228929 0.0
2011-12-01 1.0 56.019750 55.277757 0.0
2011-12-02 1.0 56.063786 55.323014 0.0
2011-12-05 1.0 56.146679 55.373357 0.0
2011-12-06 1.0 56.154322 55.410543 0.0
2011-12-07 1.0 56.114322 55.432386 0.0
2011-12-08 1.0 56.073143 55.452114 0.0
2011-12-09 1.0 56.020250 55.461714 0.0
2011-12-12 1.0 55.912536 55.468214 0.0
2011-12-13 1.0 55.801179 55.461800 0.0
2011-12-14 1.0 55.651000 55.435643 0.0
2011-12-15 1.0 55.580715 55.400686 0.0
2011-12-16 1.0 55.529679 55.384157 0.0
2011-12-19 1.0 55.491607 55.370429 0.0
2011-12-20 1.0 55.456536 55.378243 0.0
2011-12-21 1.0 55.451822 55.377814 0.0
2011-12-22 1.0 55.444500 55.391586 0.0
2011-12-23 1.0 55.439643 55.406957 0.0
2011-12-27 0.0 55.445286 55.448614 -1.0
2011-12-28 0.0 55.437643 55.490072 0.0
2011-12-29 0.0 55.468393 55.564229 0.0
2011-12-30 0.0 55.495500 55.608500 0.0
[1323 rows x 4 columns]
不是很難,對吧?輸出 signals
數據框并檢查其結果。這里重要的是要掌握該數據框中 positions
和 signal
這兩列的含義。當你繼續下去時,會發現這一點變得非常重要。
當你花時間去理解該交易策略的結果時,可以用 Matplotlib 將短期和長期的移動均線,以及買入和賣出的信號,快速繪制在圖中。
# 導入`pyplot` 模塊
import matplotlib.pyplot as plt
# 初始化圖形
fig = plt.figure(figsize=(12,8))
# 增加子圖并設置y軸標簽
ax1 = fig.add_subplot(111, ylabel='Price in $')
# 繪制收盤價曲線
aapl['Close'].plot(ax=ax1, color='r', lw=2.)
# 繪制短期和長期移動均線
signals[['short_mavg', 'long_mavg']].plot(ax=ax1, lw=2.)
# 繪制買入信號
ax1.plot(signals.loc[signals.positions == 1.0].index,
signals.short_mavg[signals.positions == 1.0],
'^', markersize=10, color='m')
# 繪制賣出信號
ax1.plot(signals.loc[signals.positions == -1.0].index,
signals.short_mavg[signals.positions == -1.0],
'v', markersize=10, color='k')
# 顯示做圖
plt.show()
結果很酷,不是嗎?