Funcat 将同花顺、通达信、文华财经等的指标公式移植到了 Python 中。
Funcat 适合做股票、期货、合约、加密数字货币的量化分析与量化交易。
基于funcat实现选股策略,在每个交易日14:00左右推送当日推荐股票。感兴趣可以关注本人微信推送服务
- tushare pro
- akshare (只支持python >= 3.9)
- okex加密数字货币
- rqdata
- tencent实时A股接口
推荐通过miniconda管理环境。
创建python 3.9.12的虚拟环境:
conda create -n py39 python==3.9.12
激活虚拟环境:
conda activate py39
- 安装科学计算库(必选,用于funcat计算指标)
- 虚拟环境安装(推荐)
conda install -c conda-forge ta-lib
- pip安装(不推荐,需要前置安装libta_lib.so,在大多数操作系统不提供该库)
pip install ta-lib
- 安装funcat:
pip install funcat3 -i https://pypi.org/simple
如果上述命令在安装ta-lib时出现报错,请重新pip install ta-lib后再执行funcat3安装命令
如有其他安装问题,请提issue,并附上足够的环境信息。
- 开盘价:
OPEN
O
- 收盘价:
CLOSE
C
- 最高价:
HIGH
H
- 最低价:
LOW
L
- 成交量:
VOLUME
V
VOL
- n天前的数据:
REF
REF(C, 10) # 10天前的收盘价
- 金叉判断:
CROSS
CROSS(MA(C, 5), MA(C, 10)) # 5日均线上穿10日均线
- 两个序列取最小值:
MIN
MIN(O, C) # K线实体的最低价
- 两个序列取最大值:
MAX
MAX(O, C) # K线实体的最高价
- n天都满足条件:
EVERY
EVERY(C > MA(C, 5), 10) # 最近10天收盘价都大于5日均线
- n天内满足条件的天数:
COUNT
COUNT(C > O, 10) # 最近10天收阳线的天数
- n天内最大值:
HHV
HHV(MAX(O, C), 60) # 最近60天K线实体的最高价
- n天内最小值:
LLV
LLV(MIN(O, C), 60) # 最近60天K线实体的最低价
- 求和n日数据
SUM
SUM(C, 10) # 求和10天的收盘价
- 求绝对值
ABS
ABS(C - O)
- 条件
IF
IF(OPEN > CLOSE, OPEN, CLOSE)
因为语法的问题,我们需要使用 &
代替 and
「和」,用 |
代替 or
「或」。
# 收盘价在10日均线上 且 10日均线在20日均线上
(C > MA(C, 10)) & (MA(C, 10) > MA(C, 20))
# 收阳线 或 收盘价大于昨收
(C > O) | (C > REF(C, 1))
- 均线:
MA
MA(C, 60) # 60日均线
其他更多请见:指标库
还有更多的技术指标还在实现中,欢迎提交pr一起实现。
KDJ指标。随机指标(KDJ)由 George C.Lane 创制。它综合了动量观念、强弱指标及移动平均线的优点,用来度量股价脱离价格正常范围的变异程度。
N, M1, M2 = 9, 3, 3
RSV = (CLOSE - LLV(LOW, N)) / (HHV(HIGH, N) - LLV(LOW, N)) * 100
K = EMA(RSV, (M1 * 2 - 1))
D = EMA(K, (M2 * 2 - 1))
J = K * 3 - D * 2
print(K, D, J)
DMI指标。动向指数又叫移动方向指数或趋向指数。是属于趋势判断的技术性指标,其基本原理是通过分析股票价格在上升及下跌过程中供需关系的均衡点,即供需关系受价格变动之影响而发生由均衡到失衡的循环过程,从而提供对趋势判断的依据。
对于 DMI 这个指标,你会发现 TALib 算出来的结果,和同花顺等软件的结果不一样,我对比了下实现方式,发现,是因为同花顺的公式和 TALib 的计算公式不一样,对于这种情况,我们把同花顺的公式搬过来,就可以算出和同花顺一样的结果。
M1, M2 = 14, 6
TR = SUM(MAX(MAX(HIGH - LOW, ABS(HIGH - REF(CLOSE, 1))), ABS(LOW - REF(CLOSE, 1))), M1)
HD = HIGH - REF(HIGH, 1)
LD = REF(LOW, 1) - LOW
DMP = SUM(IF((HD > 0) & (HD > LD), HD, 0), M1)
DMM = SUM(IF((LD > 0) & (LD > HD), LD, 0), M1)
DI1 = DMP * 100 / TR
DI2 = DMM * 100 / TR
ADX = MA(ABS(DI2 - DI1) / (DI1 + DI2) * 100, M2)
ADXR = (ADX + REF(ADX, M2)) / 2
print(DI1, DI2, ADX, ADXR)
from funcat import *
# 选出涨停股
select(
lambda : C / C[1] - 1 >= 0.0995,
start_date=20161231,
end_date=20170104,
)
'''
[20170104]
20170104 000017.XSHE 000017.SZ[深中华A]
20170104 000026.XSHE 000026.SZ[飞亚达A]
20170104 000045.XSHE 000045.SZ[深纺织A]
20170104 000585.XSHE 000585.SZ[东北电气]
20170104 000595.XSHE 000595.SZ[宝塔实业]
20170104 000678.XSHE 000678.SZ[襄阳轴承]
...
'''
# 选出最近30天K线实体最高价最低价差7%以内,最近100天K线实体最高价最低价差大于25%,
# 最近10天,收盘价大于60日均线的天数大于3天
select(
lambda : ((HHV(MAX(C, O), 30) / LLV(MIN(C, O), 30) - 1 < 0.07)
& (HHV(MAX(C, O), 100) / LLV(MIN(C, O), 100) - 1 > 0.25)
& (COUNT(C > MA(C, 60), 10) > 3)
),
start_date=20161220,
)
'''
[20170104]
20170104 600512.XSHG 600512.SH[腾达建设]
[20170103]
[20161230]
20161230 000513.XSHE 000513.SZ[丽珠集团]
...
'''
# 选出最近3天每天的成交量小于20日成交量均线,最近3天最低价低于20日均线,最高价高于20日均线
# 自定义选股回调函数
def callback(date, order_book_id, symbol):
print("Cool, 在", date, "选出", order_book_id, symbol)
select(
lambda : (EVERY(V < MA(V, 20) / 2, 3) & EVERY(L < MA(C, 20), 3) & EVERY(H > MA(C, 20), 3)),
start_date=20161231,
callback=callback,
)
'''
[20170104]
Cool, 在 20170104 选出 002633.SZ 002633.XSHE[申科股份]
Cool, 在 20170104 选出 600857.SH 600857.XSHG[宁波中百]
...
'''
from funcat import *
from funcat.data.tushare_backend import TushareDataBackend
set_data_backend(TushareDataBackend())
# 设置目前天数为2017年1月4日
T("20170104")
# 设置关注股票为上证指数
S("000001.SH")
# 打印 Open High Low Close
>>> print(O, H, L, C)
3133.79 3160.1 3130.11 3158.79
# 当天涨幅
>>> C / C[1] - 1
0.0072929156356
# 打印60日均线
>>> MA(C, 60)
3154.78333333
# 判断收盘价是否大于60日均线
>>> C > MA(C, 60)
True
# 30日最高价
>>> HHV(H, 30)
3301.21
# 最近30日,收盘价 Close 大于60日均线的天数
>>> COUNT(C > MA(C, 60), 30)
17
# 10日均线上穿
>>> CROSS(MA(C, 10), MA(C, 20))
False
import numpy as np
from funcat import *
from funcat.data.tushare_backend import TushareDataBackend
from sklearn.linear_model import LinearRegression
def select_macd_cross_up():
diff = EMA(C, 12) - EMA(C, 26)
dea = EMA(diff, 9)
macd = 2 * (diff - dea)
x_train = []
y_train = []
# 获取最近三次MACD金叉的diff值和索引位置
for i in range(100):
if macd[i] > 0 and macd[i + 1] < 0:
x_train.append(i)
y_train.append(diff[i].value)
if len(x_train) == 3:
break
if len(x_train) != 3:
return -np.nan
x_train.reverse()
y_train.reverse()
x_train = list(map(lambda i: -i + max(x_train), x_train))
# 线性回归拟合趋势
model = LinearRegression()
model.fit(np.array(x_train).reshape(-1, 1), np.array(y_train).reshape(-1, 1))
# 返回趋势线的斜率
return model.coef_
set_data_backend(TushareDataBackend())
# 设置目前天数为2021年5月19日
T("20210519")
# 设置关注股票为300298.SZ
S("300298.SZ")
# 输出结果[[-0.03154982]]
# 表明最近2021/05/19之前3次macd金叉趋势向下
# 趋势时刻有可能发生变化,该股在2021/08/02的趋势开始向上
print(select_macd_cross_up())