# [技術指標] CMO
> 來源:https://uqer.io/community/share/5590d8bdf9f06cb5604f1881
## 指標介紹
+ CMO(Chande Momentum Oscillator)動量震蕩指標是由Tushar S. Chande提出的類似于RSI的指標
+ `CMOn`是一個n天滾動指標,在這n天中的第i天計算每天的 收盤價 - 前收盤價 ,如果為正則賦給`upi`(`dni`為0),為負則將絕對值賦給`dni`(`upi`為0)
+ 其計算公式為:

## 策略思路
+ 計算上證50成分股當中所有股票過去n天的CMO
+ CMO大于0時買入,小于0時賣出
+ 根據一定的調倉原則進行調倉,細節見代碼
## 可進一步挖掘的點
+ 考慮CMO的形態,如上/下穿0線作為買賣信號
+ 擴大股票池范圍,觀察CMO與股票池的關系,比如區分大小盤股觀察CMO的有效性
+ 股票權重的分配方式
+ 其他調倉原則
## 先來看看最簡單的情況
```py
import numpy as np
start = '2010-01-01'
end = '2015-06-20'
benchmark = 'SH50'
universe = set_universe('SH50')
capital_base = 1000000
window = 35 # 參數,CMO指標計算周期
def initialize(account):
pass
def handle_data(account):
clp = account.get_attribute_history('closePrice', window)
prc = account.get_attribute_history('preClosePrice', window)
p = account.referencePrice
# 計算CMO
CMO = {}
for s in account.universe:
diff = clp[s] - prc[s]
u = sum(n for n in diff if n > 0)
d = sum(-n for n in diff if n < 0)
if u + d == 0: continue
CMO[s] = (u - d) / (u + d) * 100
# 根據CMO賣出目前持有股票
v = account.cash
for s,a in account.valid_secpos.items():
if CMO.get(s, 0) < 0 and s in account.universe:
order_to(s, 0)
v += a * p[s]
# 根據CMO確定買入列表
buylist = []
for s in account.universe:
if CMO.get(s, 0) > 0 and not np.isnan(p[s]) and s not in account.valid_secpos:
buylist.append(s)
# 根據買入列表和可用現金買入股票
if v > account.referencePortfolioValue * 0.33: # 為了避免調倉過于頻繁,僅當可用現金超過賬戶市值1/3時買入
for s in buylist:
order(s, v / len(buylist) / p[s])
```

上面的策略實現了策略思路所表述的意思,使用了非常簡單的調倉原則,其表現還不錯
但是,其中的關鍵參數 `window` 為什么設置為 35 呢?
這當然不是拍腦袋拍出來的,而是通過參數調試出來的:
```
window annualized_return sharpe max_drawdown
10 0.0052 -0.1377 0.4944
15 0.1817 0.5284 0.3388
20 0.1925 0.5850 0.3404
25 0.1914 0.5835 0.2956
30 0.2094 0.6053 0.4516
35 0.2156 0.6468 0.3856
40 0.0610 0.0970 0.5264
45 0.1980 0.5728 0.4872
50 0.1730 0.4632 0.5166
```
從上面的調試結果中可以看到,當 `window = 35` 時,夏普和最大回撤相對而言最好,因此有了最上面的那個策略
然而調試完了`window`,這個策略就沒有優化空間了嗎?不,我們還可以根據這個策略的表現來分析一下這個策略的缺陷在哪里,并加以改進
因為該策略的調倉原則是買入所有產生的信號,并沒有對持倉進行限制,這會造成兩個方面的影響:
1. 倉位中的股票可能會有很多只,這樣資金會比較分散,削弱信號的效果
1. 如果買入信號比較少,而賣出信號比較多的話,現金的利用率會比較低
那么到底是否存在上述問題呢?我們可以通過最大持倉數量和現金走勢來加以判斷
```py
x = map(len, bt['security_position'].tolist())
max(x)
42
```
```py
bt.cash.plot()
<matplotlib.axes.AxesSubplot at 0x6053cd0>
```

從上面的兩個cell中可以看出這兩個問題還是比較明顯的。為了解決這兩個問題,我們對策略進行優化:一是限制最大持倉位10只股票,二是每次賣出的現金都平均分配給目前倉位中的股票和即將買入的股票
```py
import numpy as np
from heapq import nlargest
start = '2010-01-01'
end = '2015-06-20'
benchmark = 'SH50'
universe = set_universe('SH50')
capital_base = 1000000
max_n = 10 # 參數,最大持倉數量
window = 15 # 參數,CMO指標計算周期
def initialize(account):
pass
def handle_data(account):
clp = account.get_attribute_history('closePrice', window)
prc = account.get_attribute_history('preClosePrice', window)
p = account.referencePrice
# 計算CMO
CMO = {}
for s in account.universe:
diff = clp[s] - prc[s]
u = sum(n for n in diff if n > 0)
d = sum(-n for n in diff if n < 0)
if u + d == 0: continue
CMO[s] = (u - d) / (u + d) * 100
# 根據CMO賣出目前持有股票
n = len(account.valid_secpos)
sellist = []
for s,a in account.valid_secpos.items():
if CMO.get(s, 0) < 0 and s in account.universe:
order_to(s, 0)
n -= 1
sellist.append(s)
if n >= max_n: # 如果超過最大持倉,則不買入
return
# 根據CMO確定買入列表
buylist = []
for s in account.universe:
if CMO.get(s, 0) > 0 and not np.isnan(p[s]) and s not in account.valid_secpos:
buylist.append(s)
# 根據最大持倉數量確定買入列表數量,按CMO排序選較大的部分
if len(buylist) + n > max_n:
buylist = nlargest(max_n - n, buylist, key=CMO.get)
# 將資金重新分配到新買入的與已持有的股票中
buylist += [s for s in account.valid_secpos if s not in sellist]
amount = {}
for s in buylist:
amount[s] = account.referencePortfolioValue / len(buylist) / p[s] - account.valid_secpos.get(s, 0)
# 根據應調數量買賣股票,先賣出后買入
for s in sorted(amount, key=amount.get):
order(s, amount[s])
```

```
window annualized_return sharpe max_drawdown
10 0.0830 0.1789 0.4101
15 0.2606 0.8783 0.3182
20 0.1726 0.5180 0.3689
25 0.2190 0.6762 0.3508
30 0.2067 0.6376 0.3725
35 0.1676 0.5040 0.3550
40 0.1416 0.4086 0.4478
45 0.1927 0.5926 0.4001
50 0.1183 0.3215 0.4030
```
從上面的圖表可以看出其表現相比最初的策略有了不少改善。其中 `window = 15` 是最適合目前調倉原則的參數。
但是這些優化的初衷是為了解決股票數量和資金利用率的問題,我們仍然通過最大持倉數量和現金走勢來判斷
```py
x = map(len, bt['security_position'].tolist())
max(x)
10
```
```py
bt.cash.plot()
<matplotlib.axes.AxesSubplot at 0x5a48fd0>
```
!](img/8DjEAXQMXIa20AAAAASUVORK5CYII=.png)
以上是對CMO這個技術指標進行的一些簡單的回測,并且針對策略本身的特點進行了一定的優化。在最前面列出了一些可挖掘的點,如果想進行更深入的研究還是有很多東西可以做的。
- Python 量化交易教程
- 第一部分 新手入門
- 一 量化投資視頻學習課程
- 二 Python 手把手教學
- 量化分析師的Python日記【第1天:誰來給我講講Python?】
- 量化分析師的Python日記【第2天:再接著介紹一下Python唄】
- 量化分析師的Python日記【第3天:一大波金融Library來襲之numpy篇】
- 量化分析師的Python日記【第4天:一大波金融Library來襲之scipy篇】
- 量化分析師的Python日記【第5天:數據處理的瑞士軍刀pandas】
- 量化分析師的Python日記【第6天:數據處理的瑞士軍刀pandas下篇
- 量化分析師的Python日記【第7天:Q Quant 之初出江湖】
- 量化分析師的Python日記【第8天 Q Quant兵器譜之函數插值】
- 量化分析師的Python日記【第9天 Q Quant兵器譜之二叉樹】
- 量化分析師的Python日記【第10天 Q Quant兵器譜 -之偏微分方程1】
- 量化分析師的Python日記【第11天 Q Quant兵器譜之偏微分方程2】
- 量化分析師的Python日記【第12天:量化入門進階之葵花寶典:因子如何產生和回測】
- 量化分析師的Python日記【第13天 Q Quant兵器譜之偏微分方程3】
- 量化分析師的Python日記【第14天:如何在優礦上做Alpha對沖模型】
- 量化分析師的Python日記【第15天:如何在優礦上搞一個wealthfront出來】
- 第二部分 股票量化相關
- 一 基本面分析
- 1.1 alpha 多因子模型
- 破解Alpha對沖策略——觀《量化分析師Python日記第14天》有感
- 熔斷不要怕, alpha model 為你保駕護航!
- 尋找 alpha 之: alpha 設計
- 1.2 基本面因子選股
- Porfolio(現金比率+負債現金+現金保障倍數)+市盈率
- ROE選股指標
- 成交量因子
- ROIC&cashROIC
- 【國信金工】資產周轉率選股模型
- 【基本面指標】Cash Cow
- 量化因子選股——凈利潤/營業總收入
- 營業收入增長率+市盈率
- 1.3 財報閱讀 ? [米缸量化讀財報] 資產負債表-投資相關資產
- 1.4 股東分析
- 技術分析入門 【2】 —— 大家搶籌碼(06年至12年版)
- 技術分析入門 【2】 —— 大家搶籌碼(06年至12年版)— 更新版
- 誰是中國A股最有錢的自然人
- 1.5 宏觀研究
- 【干貨包郵】手把手教你做宏觀擇時
- 宏觀研究:從估值角度看當前市場
- 追尋“國家隊”的足跡
- 二 套利
- 2.1 配對交易
- HS300ETF套利(上)
- 【統計套利】配對交易
- 相似公司股票搬磚
- Paired trading
- 2.2 期現套利 ? 通過股指期貨的期現差與 ETF 對沖套利
- 三 事件驅動
- 3.1 盈利預增
- 盈利預增事件
- 事件驅動策略示例——盈利預增
- 3.2 分析師推薦 ? 分析師的金手指?
- 3.3 牛熊轉換
- 歷史總是相似 牛市還在延續
- 歷史總是相似 牛市已經見頂?
- 3.4 熔斷機制 ? 股海拾貝之 [熔斷錯殺股]
- 3.5 暴漲暴跌 ? [實盤感悟] 遇上暴跌我該怎么做?
- 3.6 兼并重組、舉牌收購 ? 寶萬戰-大戲開幕
- 四 技術分析
- 4.1 布林帶
- 布林帶交易策略
- 布林帶回調系統-日內
- Conservative Bollinger Bands
- Even More Conservative Bollinger Bands
- Simple Bollinger Bands
- 4.2 均線系統
- 技術分析入門 —— 雙均線策略
- 5日線10日線交易策略
- 用5日均線和10日均線進行判斷 --- 改進版
- macross
- 4.3 MACD
- Simple MACD
- MACD quantization trade
- MACD平滑異同移動平均線方法
- 4.4 阿隆指標 ? 技術指標阿隆( Aroon )全解析
- 4.5 CCI ? CCI 順勢指標探索
- 4.6 RSI
- 重寫 rsi
- RSI指標策略
- 4.7 DMI ? DMI 指標體系的構建及簡單應用
- 4.8 EMV ? EMV 技術指標的構建及應用
- 4.9 KDJ ? KDJ 策略
- 4.10 CMO
- CMO 策略模仿練習 1
- CMO策略模仿練習2
- [技術指標] CMO
- 4.11 FPC ? FPC 指標選股
- 4.12 Chaikin Volatility
- 嘉慶離散指標測試
- 4.13 委比 ? 實時計算委比
- 4.14 封單量
- 按照封單跟流通股本比例排序,剔除6月上市新股,前50
- 漲停股票封單統計
- 實時計算漲停板股票的封單資金與總流通市值的比例
- 4.15 成交量 ? 決戰之地, IF1507 !
- 4.16 K 線分析 ? 尋找夜空中最亮的星
- 五 量化模型
- 5.1 動量模型
- Momentum策略
- 【小散學量化】-2-動量模型的簡單實踐
- 一個追漲的策略(修正版)
- 動量策略(momentum driven)
- 動量策略(momentum driven)——修正版
- 最經典的Momentum和Contrarian在中國市場的測試
- 最經典的Momentum和Contrarian在中國市場的測試-yanheven改進
- [策略]基于勝率的趨勢交易策略
- 策略探討(更新):價量結合+動量反轉
- 反向動量策略(reverse momentum driven)
- 輕松跑贏大盤 - 主題Momentum策略
- Contrarian strategy
- 5.2 Joseph Piotroski 9 F-Score Value Investing Model · 基本面選股系統:Piotroski F-Score ranking system
- 5.3 SVR · 使用SVR預測股票開盤價 v1.0
- 5.4 決策樹、隨機樹
- 決策樹模型(固定模型)
- 基于Random Forest的決策策略
- 5.5 鐘擺理論 · 鐘擺理論的簡單實現——完美躲過股災和精準抄底
- 5.6 海龜模型
- simple turtle
- 俠之大者 一起賺錢
- 5.7 5217 策略 · 白龍馬的新手策略
- 5.8 SMIA · 基于歷史狀態空間相似性匹配的行業配置 SMIA 模型—取交集
- 5.9 神經網絡
- 神經網絡交易的訓練部分
- 通過神經網絡進行交易
- 5.10 PAMR · PAMR : 基于均值反轉的投資組合選擇策略 - 修改版
- 5.11 Fisher Transform · Using Fisher Transform Indicator
- 5.12 分型假說, Hurst 指數 · 分形市場假說,一個聽起來很美的假說
- 5.13 變點理論 · 變點策略初步
- 5.14 Z-score Model
- Zscore Model Tutorial
- 信用債風險模型初探之:Z-Score Model
- user-defined package
- 5.15 機器學習 · Machine Learning 學習筆記(一) by OTreeWEN
- 5.16 DualTrust 策略和布林強盜策略
- 5.17 卡爾曼濾波
- 5.18 LPPL anti-bubble model
- 今天大盤熔斷大跌,后市如何—— based on LPPL anti-bubble model
- 破解股市泡沫之謎——對數周期冪率(LPPL)模型
- 六 大數據模型
- 6.1 市場情緒分析
- 通聯情緒指標策略
- 互聯網+量化投資 大數據指數手把手
- 6.2 新聞熱點
- 如何使用優礦之“新聞熱點”?
- 技術分析【3】—— 眾星拱月,眾口鑠金?
- 七 排名選股系統
- 7.1 小市值投資法
- 學習筆記:可模擬(小市值+便宜 的修改版)
- 市值最小300指數
- 流通市值最小股票(新篩選器版)
- 持有市值最小的10只股票
- 10% smallest cap stock
- 7.2 羊駝策略
- 羊駝策略
- 羊駝反轉策略(修改版)
- 羊駝反轉策略
- 我的羊駝策略,選5只股無腦輪替
- 7.3 低價策略
- 專撿便宜貨(新版quartz)
- 策略原理
- 便宜就是 alpha
- 八 輪動模型
- 8.1 大小盤輪動 · 新手上路 -- 二八ETF擇時輪動策略2.0
- 8.2 季節性策略
- Halloween Cycle
- Halloween cycle 2
- 夏買電,東買煤?
- 歷史的十一月板塊漲幅
- 8.3 行業輪動
- 銀行股輪動
- 申萬二級行業在最近1年、3個月、5個交易日的漲幅統計
- 8.4 主題輪動
- 快速研究主題神器
- recommendation based on subject
- strategy7: recommendation based on theme
- 板塊異動類
- 風險因子(離散類)
- 8.5 龍頭輪動
- Competitive Securities
- Market Competitiveness
- 主題龍頭類
- 九 組合投資
- 9.1 指數跟蹤 · [策略] 指數跟蹤低成本建倉策略
- 9.2 GMVP · Global Minimum Variance Portfolio (GMVP)
- 9.3 凸優化 · 如何在 Python 中利用 CVXOPT 求解二次規劃問題
- 十 波動率
- 10.1 波動率選股 · 風平浪靜 風起豬飛
- 10.2 波動率擇時
- 基于 VIX 指數的擇時策略
- 簡單低波動率指數
- 10.3 Arch/Garch 模型 · 如何使用優礦進行 GARCH 模型分析
- 十一 算法交易
- 11.1 VWAP · Value-Weighted Average Price (VWAP)
- 十二 中高頻交易
- 12.1 order book 分析 · 基于高頻 limit order book 數據的短程價格方向預測—— via multi-class SVM
- 12.2 日內交易 · 大盤日內走勢 (for 擇時)
- 十三 Alternative Strategy
- 13.1 易經、傳統文化 · 老黃歷診股
- 第三部分 基金、利率互換、固定收益類
- 一 分級基金
- “優礦”集思錄——分級基金專題
- 基于期權定價的分級基金交易策略
- 基于期權定價的興全合潤基金交易策略
- 二 基金分析
- Alpha 基金“黑天鵝事件” -- 思考以及原因
- 三 債券
- 債券報價中的小陷阱
- 四 利率互換
- Swap Curve Construction
- 中國 Repo 7D 互換的例子
- 第四部分 衍生品相關
- 一 期權數據
- 如何獲取期權市場數據快照
- 期權高頻數據準備
- 二 期權系列
- [ 50ETF 期權] 1. 歷史成交持倉和 PCR 數據
- 【50ETF期權】 2. 歷史波動率
- 【50ETF期權】 3. 中國波指 iVIX
- 【50ETF期權】 4. Greeks 和隱含波動率微笑
- 【50ETF期權】 5. 日內即時監控 Greeks 和隱含波動率微笑
- 【50ETF期權】 5. 日內即時監控 Greeks 和隱含波動率微笑
- 三 期權分析
- 【50ETF期權】 期權擇時指數 1.0
- 每日期權風險數據整理
- 期權頭寸計算
- 期權探秘1
- 期權探秘2
- 期權市場一周縱覽
- 基于期權PCR指數的擇時策略
- 期權每日成交額PC比例計算
- 四 期貨分析
- 【前方高能!】Gifts from Santa Claus——股指期貨趨勢交易研究