Source code for czsc.signals.pos

# -*- coding: utf-8 -*-
"""
author: zengbin93
email: zeng_bin8888@163.com
create_dt: 2023/4/14 19:27
describe: 
"""
from czsc.analyze import CZSC
from collections import OrderedDict
from czsc.traders.base import CzscTrader
from czsc.utils import create_single_signal
from czsc.objects import Operate, Direction, Mark
from czsc.signals.tas import update_ma_cache


[docs]def pos_ma_V230414(cat: CzscTrader, **kwargs) -> OrderedDict: """判断开仓后是否升破MA均线或跌破MA均线 参数模板:"{pos_name}_{freq1}#{ma_type}#{timeperiod}_持有状态V230414" **信号逻辑:** 多头止损逻辑如下,反之为空头止损逻辑: 1. 从多头开仓点开始,在给定对的K线周期 freq1 上向前找 N 个底分型,记为 F1 2. 将这 N 个底分型的最低点,记为 L1,如果 L1 的价格低于开仓点的价格,则止损 **信号列表:** - Signal('日线三买多头N1_60分钟#SMA#5_持有状态V230414_多头_升破均线_任意_0') - Signal('日线三买多头N1_60分钟#SMA#5_持有状态V230414_空头_跌破均线_任意_0') :param cat: CzscTrader对象 :param kwargs: 参数字典 - pos_name: str,开仓信号的名称 - freq1: str,给定的K线周期 - n: int,向前找的分型个数,默认为 3 :return: """ pos_name = kwargs["pos_name"] freq1 = kwargs["freq1"] ma_type = kwargs.get("ma_type", "SMA").upper() timeperiod = int(kwargs.get("timeperiod", 5)) k1, k2, k3 = f"{pos_name}_{freq1}#{ma_type}#{timeperiod}_持有状态V230414".split("_") v1, v2 = '其他', '其他' key = update_ma_cache(cat.kas[freq1], ma_type=ma_type, timeperiod=timeperiod) # 如果没有持仓策略,则不产生信号 if not hasattr(cat, "positions"): return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) pos = [x for x in cat.positions if x.name == pos_name][0] if len(pos.operates) == 0 or pos.operates[-1]['op'] in [Operate.SE, Operate.LE]: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) c = cat.kas[freq1] op = pos.operates[-1] # 多头止损逻辑 if op['op'] == Operate.LO: bars = [x for x in c.bars_raw[-100:] if x.dt > op['dt']] for x in bars: if x.close > x.cache[key]: v1, v2 = '多头', '升破均线' break # 空头止损逻辑 if op['op'] == Operate.SO: bars = [x for x in c.bars_raw[-100:] if x.dt > op['dt']] for x in bars: if x.close < x.cache[key]: v1, v2 = '空头', '跌破均线' break return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1, v2=v2)
[docs]def pos_fx_stop_V230414(cat: CzscTrader, **kwargs) -> OrderedDict: """按照开仓点附近的分型止损 参数模板:"{freq1}_{pos_name}N{n}_止损V230414" **信号逻辑:** 多头止损逻辑如下,反之为空头止损逻辑: 1. 从多头开仓点开始,在给定对的K线周期 freq1 上向前找 N 个底分型,记为 F1 2. 将这 N 个底分型的最低点,记为 L1,如果 L1 的价格低于开仓点的价格,则止损 **信号列表:** - Signal('日线_日线三买多头N1_止损V230414_多头止损_任意_任意_0') - Signal('日线_日线三买多头N1_止损V230414_空头止损_任意_任意_0') :param cat: CzscTrader对象 :param kwargs: 参数字典 - pos_name: str,开仓信号的名称 - freq1: str,给定的K线周期 - n: int,向前找的分型个数,默认为 3 :return: """ pos_name = kwargs["pos_name"] freq1 = kwargs["freq1"] n = int(kwargs.get('n', 3)) k1, k2, k3 = f"{freq1}_{pos_name}N{n}_止损V230414".split("_") v1 = '其他' # 如果没有持仓策略,则不产生信号 if not hasattr(cat, "positions"): return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) pos = [x for x in cat.positions if x.name == pos_name][0] if len(pos.operates) == 0 or pos.operates[-1]['op'] in [Operate.SE, Operate.LE]: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) c = cat.kas[freq1] op = pos.operates[-1] # 多头止损逻辑 if op['op'] == Operate.LO: fxs = [x for x in c.fx_list if x.mark == Mark.D and x.dt < op['dt']][-n:] if cat.latest_price < min([x.low for x in fxs]): v1 = '多头止损' # 空头止损逻辑 if op['op'] == Operate.SO: fxs = [x for x in c.fx_list if x.mark == Mark.G and x.dt < op['dt']][-n:] if cat.latest_price > max([x.high for x in fxs]): v1 = '空头止损' return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1)
[docs]def pos_bar_stop_V230524(cat: CzscTrader, **kwargs) -> OrderedDict: """按照开仓点附近的N根K线极值止损 参数模板:"{pos_name}_{freq1}N{n}K_止损V2305224" **信号逻辑:** 多头止损逻辑如下,反之为空头止损逻辑: 1. 从多头开仓点开始,在给定对的K线周期 freq1 上向前找 N 个K线,记为 F1 2. 将这 N 个K线的最低点,记为 L1,如果最新价跌破 L1,则止损 **信号列表:** - Signal('日线三买多头_日线N3K_止损V2305224_多头止损_任意_任意_0') - Signal('日线三买多头_日线N3K_止损V2305224_空头止损_任意_任意_0') :param cat: CzscTrader对象 :param kwargs: 参数字典 - pos_name: str,开仓信号的名称 - freq1: str,给定的K线周期 - n: int,向前找的K线个数,默认为 3 :return: """ pos_name = kwargs["pos_name"] freq1 = kwargs["freq1"] n = int(kwargs.get('n', 3)) k1, k2, k3 = f"{pos_name}_{freq1}N{n}K_止损V2305224".split("_") v1 = '其他' assert 20 >= n >= 1, "参数 n 取值范围为 1~20" # 如果没有持仓策略,则不产生信号 if not hasattr(cat, "positions"): return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) pos_ = [x for x in cat.positions if x.name == pos_name][0] if len(pos_.operates) == 0 or pos_.operates[-1]['op'] in [Operate.SE, Operate.LE]: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) c: CZSC = cat.kas[freq1] op = pos_.operates[-1] # 多头止损逻辑 if op['op'] == Operate.LO: bars = [x for x in c.bars_raw[-100:] if x.dt < op['dt']][-n:] if cat.latest_price < min([x.low for x in bars]): v1 = '多头止损' # 空头止损逻辑 if op['op'] == Operate.SO: bars = [x for x in c.bars_raw[-100:] if x.dt < op['dt']][-n:] if cat.latest_price > max([x.high for x in bars]): v1 = '空头止损' return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1)
[docs]def pos_holds_V230414(cat: CzscTrader, **kwargs) -> OrderedDict: """开仓后N根K线涨幅小于M%%,则平仓 参数模板:"{pos_name}_{freq1}N{n}M{m}_趋势判断V230414" **信号逻辑:** 1. 找出开仓后的 N 根K线,计算涨幅,如果涨幅小于 M%%,则平仓 2. 这里面的逻辑是,如果开仓后的 N 根K线涨幅小于 M%%,则说明趋势不明朗,平仓等待 **信号列表:** - Signal('日线三买多头N1_60分钟N5M100_趋势判断V230414_多头存疑_任意_任意_0') - Signal('日线三买多头N1_60分钟N5M100_趋势判断V230414_多头良好_任意_任意_0') :param cat: CzscTrader对象 :param kwargs: 参数字典 - pos_name: str,开仓信号的名称 - freq1: str,给定的K线周期 - n: int,最少持有K线数量,默认为 5,表示5根K线之后开始判断趋势 - m: int,涨幅阈值,默认为 100,表示涨幅小于 100BP 时,平仓 :return: """ pos_name = kwargs["pos_name"] freq1 = kwargs["freq1"] n = int(kwargs.get('n', 5)) m = int(kwargs.get('m', 100)) k1, k2, k3 = f"{pos_name}_{freq1}N{n}M{m}_趋势判断V230414".split("_") v1 = '其他' # 如果没有持仓策略,则不产生信号 if not hasattr(cat, "positions"): return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) pos = [x for x in cat.positions if x.name == pos_name][0] if len(pos.operates) == 0 or pos.operates[-1]['op'] in [Operate.SE, Operate.LE]: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) c = cat.kas[freq1] op = pos.operates[-1] bars = [x for x in c.bars_raw[-100:] if x.dt > op['dt']] if len(bars) < n: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) if op['op'] == Operate.LO: zdf = (bars[-1].close - op['price']) / op['price'] * 10000 v1 = '多头存疑' if zdf < m else '多头良好' if op['op'] == Operate.SO: zdf = (op['price'] - bars[-1].close) / op['price'] * 10000 v1 = '空头存疑' if zdf < m else '空头良好' return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1)
[docs]def pos_fix_exit_V230624(cat: CzscTrader, **kwargs) -> OrderedDict: """固定比例止损,止盈 参数模板:"{pos_name}_固定{th}BP止盈止损_出场V230624" **信号逻辑:** 以多头为例,如果持有收益超过 th 个BP,则止盈;如果亏损超过 th 个BP,则止损。 **信号列表:** - Signal('日线三买多头_固定100BP止盈止损_出场V230624_多头止损_任意_任意_0') - Signal('日线三买多头_固定100BP止盈止损_出场V230624_空头止损_任意_任意_0') :param cat: CzscTrader对象 :param kwargs: 参数字典 - pos_name: str,开仓信号的名称 - freq1: str,给定的K线周期 - n: int,向前找的K线个数,默认为 3 :return: """ pos_name = kwargs["pos_name"] th = int(kwargs.get('th', 300)) k1, k2, k3 = f"{pos_name}_固定{th}BP止盈止损_出场V230624".split("_") v1 = '其他' if not hasattr(cat, "positions"): return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) pos_ = [x for x in cat.positions if x.name == pos_name][0] if len(pos_.operates) == 0 or pos_.operates[-1]['op'] in [Operate.SE, Operate.LE]: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) op = pos_.operates[-1] op_price = op['price'] if op['op'] == Operate.LO: if cat.latest_price < op_price * (1 - th / 10000): v1 = '多头止损' if cat.latest_price > op_price * (1 + th / 10000): v1 = '多头止盈' if op['op'] == Operate.SO: if cat.latest_price > op_price * (1 + th / 10000): v1 = '空头止损' if cat.latest_price < op_price * (1 - th / 10000): v1 = '空头止盈' return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1)
[docs]def pos_profit_loss_V230624(cat: CzscTrader, **kwargs) -> OrderedDict: """开仓后盈亏比达到一定比值,才允许平仓 贡献者:谌意勇 参数模板:"{pos_name}_{freq1}YKB{ykb}N{n}_盈亏比判断V230624" **信号逻辑:** 1. 通过公式 计算盈亏比=abs(现价-开仓价)/abs(开仓价-止损价)* 10,当比值大于一定阀值时才允许平仓 **信号列表:** - Signal('日线通道突破_60分钟YKB20N3_盈亏比判断V230624_空头止损_任意_任意_0') - Signal('日线通道突破_60分钟YKB20N3_盈亏比判断V230624_多头止损_任意_任意_0') - Signal('日线通道突破_60分钟YKB20N3_盈亏比判断V230624_多头达标_任意_任意_0') - Signal('日线通道突破_60分钟YKB20N3_盈亏比判断V230624_空头达标_任意_任意_0') :param cat: CzscTrader对象 :param kwargs: 参数字典 - pos_name: str,开仓信号的名称 - freq1: str,给定的K线周期 - ykb: int,默认为 20, 表示2倍盈亏比,计算盈亏比=abs(现价-开仓价)/abs(开仓价-止损价) - n: int 默认为3 止损取最近n个分型的最低点或最高点 :return: """ pos_name = kwargs["pos_name"] freq1 = kwargs["freq1"] ykb = int(kwargs.get('ykb', 20)) n = int(kwargs.get('n', 3)) k1, k2, k3 = f"{pos_name}_{freq1}YKB{ykb}N{n}_盈亏比判断V230624".split("_") v1 = '其他' # 如果没有持仓策略,则不产生信号 if not hasattr(cat, "positions"): return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) pos = [x for x in cat.positions if x.name == pos_name][0] if len(pos.operates) == 0 or pos.operates[-1]['op'] in [Operate.SE, Operate.LE]: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) c = cat.kas[freq1] op = pos.operates[-1] last_close = c.bars_raw[-1].close if op['op'] == Operate.LO: fxs = [x for x in c.fx_list if x.mark == Mark.D and x.dt < op['dt']][-n:] stop_price = min([x.low for x in fxs]) ykb_ = ((last_close - op['price']) / (op['price'] - stop_price)) * 10 if ykb_ > ykb: v1 = '多头达标' else: if last_close < stop_price: v1 = '多头止损' if op['op'] == Operate.SO: fxs = [x for x in c.fx_list if x.mark == Mark.G and x.dt < op['dt']][-n:] stop_price = max([x.high for x in fxs]) ykb_ = ((last_close - op['price']) / (op['price'] - stop_price)) * 10 if ykb_ > ykb: v1 = '空头达标' else: if last_close > stop_price: v1 = '空头止损' return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1)
[docs]def pos_status_V230808(cat: CzscTrader, **kwargs) -> OrderedDict: """Position策略的持仓状态 参数模板:"{pos_name}_持仓状态_BS辅助V230808" **信号逻辑:** 对指定的持仓策略,有三种状态:持多、持空、持币。 **信号列表:** - Signal('日线三买多头N1_持仓状态_BS辅助V230808_持多_任意_任意_0') - Signal('日线三买多头N1_持仓状态_BS辅助V230808_持空_任意_任意_0') - Signal('日线三买多头N1_持仓状态_BS辅助V230808_持币_任意_任意_0') :param cat: CzscTrader对象 :param kwargs: 参数字典 - pos_name: str,开仓信号的名称 :return: """ pos_name = kwargs["pos_name"] k1, k2, k3 = f"{pos_name}_持仓状态_BS辅助V230808".split("_") # 如果没有持仓策略,则不产生信号 if not hasattr(cat, "positions"): return create_single_signal(k1=k1, k2=k2, k3=k3, v1="其他") pos = cat.get_position(pos_name) v1 = '持币' if len(pos.operates) == 0: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) op = pos.operates[-1] if op['op'] == Operate.LO: v1 = '持多' if op['op'] == Operate.SO: v1 = '持空' return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1)
[docs]def pos_holds_V230807(cat: CzscTrader, **kwargs) -> OrderedDict: """开仓后N根K线收益小于M%%,且当前收益大于T%%,平仓保本 参数模板:"{pos_name}_{freq1}N{n}M{m}T{t}_BS辅助V230807" **信号逻辑:** 1. 针对某个开仓点,如果 N 根K线之后,收益低于 M,则认为开仓失误; 2. 开仓的失误发生后,如果市场给了逃命的机会,不贪心,等待收益大于 T 时,平仓保本; 3. 保本有两种场景:开仓后先亏损后反弹;开仓后先盈利后回落。 **信号列表:** - Signal('日线三买多头N1_60分钟N5M50T10_BS辅助V230807_多头保本_任意_任意_0') - Signal('日线三买多头N1_60分钟N5M50T10_BS辅助V230807_空头保本_任意_任意_0') :param cat: CzscTrader对象 :param kwargs: 参数字典 - pos_name: str,开仓信号的名称 - freq1: str,给定的K线周期 - n: int,最少持有K线数量,默认为 5,表示5根K线之后开始判断 - m: int,收益最小阈值,默认为 100,表示收益小于 100BP 时,需要开始判断保本单 - t: int,保本收益阈值,默认为 10,表示收益大于 10BP 时,可以平仓保本 :return: """ pos_name = kwargs["pos_name"] freq1 = kwargs["freq1"] n = int(kwargs.get('n', 5)) m = int(kwargs.get('m', 50)) t = int(kwargs.get('t', 10)) assert m > t > 0, "参数 m 必须大于 t" k1, k2, k3 = f"{pos_name}_{freq1}N{n}M{m}T{t}_BS辅助V230807".split("_") v1 = '其他' # 如果没有持仓策略,则不产生信号 if not cat.kas or not hasattr(cat, "positions"): return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) pos = [x for x in cat.positions if x.name == pos_name][0] if len(pos.operates) == 0 or pos.operates[-1]['op'] in [Operate.SE, Operate.LE]: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) c = cat.kas[freq1] op = pos.operates[-1] bars = [x for x in c.bars_raw[-100:] if x.dt > op['dt']] if len(bars) < n: return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1) if op['op'] == Operate.LO: zdf = (bars[-1].close - op['price']) / op['price'] * 10000 if t < zdf < m: v1 = '多头保本' if op['op'] == Operate.SO: zdf = (op['price'] - bars[-1].close) / op['price'] * 10000 if t < zdf < m: v1 = '空头保本' return create_single_signal(k1=k1, k2=k2, k3=k3, v1=v1)