【譯】Python 金融:算法交易 (3)用Python構建交易策略

本文翻譯自2018年最熱門的Python金融教程 Python For Finance: Algorithmic Trading

本教程由以下五部分內容構成:

本文是該教程的第三部分。


既然你已對數據做了初步分析,那么是時候創建你的第一個交易策略了。但在深入研究之前,為什么不先來了解一些最常用的交易策略呢?下面簡短的介紹,無疑會讓你更容易開始交易策略的開發。

常見的交易策略

你應該還記得在本教程開篇的介紹中,我們講了交易策略是在市場上做多或做空的既定計劃,但不僅限于此,你還有更多的信息沒有真正掌握。一般來說,有兩種常見的交易策略:動量策略(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() 函數開始滑動窗口計算:在該函數中,設置 windowmin_periodcenter 參數。在實際中,窗口大小window 分別設置為 short_windowlong_windowmin_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 數據框并檢查其結果。這里重要的是要掌握該數據框中 positionssignal 這兩列的含義。當你繼續下去時,會發現這一點變得非常重要。

當你花時間去理解該交易策略的結果時,可以用 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()

結果很酷,不是嗎?

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,182評論 6 543
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,489評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,290評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,776評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,510評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,866評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,860評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,036評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,585評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,331評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,536評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,058評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,754評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,154評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,469評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,273評論 3 399
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,505評論 2 379

推薦閱讀更多精彩內容

  • 運用三遍閱讀法,在2小時內閱讀完《加速》,并閱讀書中某一個片段,用RIA法進行拆解輸出。 一、閱讀時間記錄和分析 ...
    度半32閱讀 168評論 0 0
  • 你來我往一起玩,氣氛熱烈不孤單。 但愿今生長相伴,冬天來到也溫暖。
    蠻力閱讀 423評論 3 7
  • 三十歲,曾經是一個那么遙遠的年齡,在2011年,他悄悄的來了,來的讓我措手不及,難以接受。 回想青蔥歲月里,什么也...
    做不一樣的我閱讀 380評論 0 0