Strategi kuantitatif henti untung dan henti rugi dinamik crossover purata bergerak berganda

EMA SMA
Tarikh penciptaan: 2024-12-11 11:23:54 Akhirnya diubah suai: 2024-12-11 11:23:54
Salin: 0 Bilangan klik: 441
1
fokus pada
1617
Pengikut

Strategi kuantitatif henti untung dan henti rugi dinamik crossover purata bergerak berganda

Gambaran keseluruhan

Ini adalah strategi perdagangan kuantitatif berdasarkan isyarat silang dua garis rata, untuk menilai trend pasaran melalui persilangan indeks bergerak cepat (EMA) dan indeks bergerak perlahan (EMA), sambil menggabungkan kawalan hentian hentian dinamik untuk menguruskan risiko. Strategi ini menggunakan pengurusan kedudukan peratusan, secara lalai menggunakan 10% dana untuk berdagang, untuk melindungi keuntungan dan mengawal risiko dengan menetapkan harga hentian dan hentian dinamik.

Prinsip Strategi

Logik teras strategi ini adalah untuk mengenal pasti perubahan trend dengan memantau 20 kitaran dan 50 kitaran crossover rata-rata bergerak indeks (EMA). Apabila EMA yang cepat ke atas melintasi EMA yang perlahan, sistem menghasilkan banyak isyarat. Setiap kali kedudukan dibuka, sistem akan secara automatik menetapkan harga berhenti (STOP) dan harga berhenti (STOP) berdasarkan harga masuk (STOP) dan harga berhenti (STOP) berdasarkan harga masuk (STOP).

Kelebihan Strategik

  1. Stabiliti isyarat yang kuat - menggunakan EMA dan bukan purata bergerak sederhana ((SMA), dapat bertindak balas lebih cepat terhadap perubahan harga, dan juga dapat menyaring sebahagian bunyi pasaran.
  2. Pengurusan risiko yang sempurna - Mekanisme Stop Loss Dinamik digunakan, harga Stop Loss akan disesuaikan dengan perubahan harga masuk.
  3. Pengurusan dana yang munasabah - Menggunakan pengurusan kedudukan peratusan tetap, mengelakkan risiko tinggi operasi keseluruhan.
  4. Tingkat automasi yang tinggi - automasi dari penjanaan isyarat hingga pengurusan kedudukan, mengurangkan campur tangan manusia.
  5. Adaptif - Strategi boleh disesuaikan dengan keadaan pasaran yang berbeza, parameter boleh disesuaikan dengan keadaan sebenar.

Risiko Strategik

  1. Keterlambatan garis rata-rata - EMA bertindak balas dengan cepat, tetapi masih terdapat keterlambatan tertentu, yang mungkin menyebabkan sedikit kelewatan masa masuk.
  2. Tidak berlaku untuk pasaran goyah - Isyarat penembusan palsu yang kerap mungkin berlaku apabila pasaran goyah.
  3. Penangguhan pegangan pengganda tetap - Penangguhan pegangan yang ditetapkan dengan pengganda tetap mungkin tidak sesuai untuk semua keadaan pasaran
  4. Risiko penarikan balik - Dalam pasaran yang bergelombang, 5% stop loss mungkin tidak mencukupi untuk menghadapi turun naik yang besar.

Arah pengoptimuman strategi

  1. Memperkenalkan penunjuk kadar turun naik - Mencadangkan penambahan penunjuk ATR untuk secara dinamik menyesuaikan penggandaan hentian hentian untuk menjadikannya lebih sesuai dengan turun naik pasaran.
  2. Meningkatkan pengesahan trend - boleh digabungkan dengan RSI, MACD dan lain-lain untuk menapis isyarat perdagangan dan meningkatkan peluang kemenangan.
  3. Pengurusan kedudukan yang dioptimumkan - saiz kedudukan boleh disesuaikan secara dinamik mengikut turun naik pasaran, untuk kawalan risiko yang lebih halus.
  4. Menambah penapis masa - pertimbangkan untuk menambah sekatan pada tetingkap masa dagangan, mengelakkan masa yang lebih bergelombang.
  5. Peningkatan mekanisme penangguhan - penangguhan bergerak boleh dilakukan, dan lebih banyak keuntungan akan diperoleh jika keadaan terus bertambah baik.

ringkaskan

Ini adalah strategi untuk mengesan trend yang dirancang dengan logik dan logik yang jelas, menangkap trend dengan penyeberangan dua garis sejajar, menggunakan stop loss dinamik untuk menguruskan risiko. Kelebihan strategi ini adalah bahawa peraturan operasi jelas, risiko dapat dikawal, sesuai sebagai kerangka asas sistem perdagangan jangka menengah dan panjang. Dengan menambahkan lebih banyak syarat penapisan dan mengoptimumkan mekanisme stop loss, strategi ini mempunyai ruang pengoptimuman yang lebih besar.

Kod sumber strategi
/*backtest
start: 2019-12-23 08:00:00
end: 2024-12-09 08:00:00
period: 1d
basePeriod: 1d
exchanges: [{"eid":"Futures_Binance","currency":"BTC_USDT"}]
*/

// This source code is subject to the terms of the Mozilla Public License 2.0 at https://mozilla.org/MPL/2.0/
// © Pineify

//======================================================================//
//                    ____  _            _  __                          //
//                   |  _ \(_)_ __   ___(_)/ _|_   _                    //
//                   | |_) | | '_ \ / _ \ | |_| | | |                   //
//                   |  __/| | | | |  __/ |  _| |_| |                   //
//                   |_|   |_|_| |_|\___|_|_|  \__, |                   //
//                                             |___/                    //
//======================================================================//

//@version=5
strategy(title="TQQQ EMA Strategy", overlay=true)

//#region —————————————————————————————————————————————————— Common Dependence

p_comm_time_range_to_unix_time(string time_range, int date_time = time, string timezone = syminfo.timezone) =>
    int start_unix_time = na
    int end_unix_time = na
    int start_time_hour = na
    int start_time_minute = na
    int end_time_hour = na
    int end_time_minute = na
    if str.length(time_range) == 11
        // Format: hh:mm-hh:mm
        start_time_hour := math.floor(str.tonumber(str.substring(time_range, 0, 2)))
        start_time_minute := math.floor(str.tonumber(str.substring(time_range, 3, 5)))
        end_time_hour := math.floor(str.tonumber(str.substring(time_range, 6, 8)))
        end_time_minute := math.floor(str.tonumber(str.substring(time_range, 9, 11)))
    else if str.length(time_range) == 9
        // Format: hhmm-hhmm
        start_time_hour := math.floor(str.tonumber(str.substring(time_range, 0, 2)))
        start_time_minute := math.floor(str.tonumber(str.substring(time_range, 2, 4)))
        end_time_hour := math.floor(str.tonumber(str.substring(time_range, 5, 7)))
        end_time_minute := math.floor(str.tonumber(str.substring(time_range, 7, 9)))
    start_unix_time := timestamp(timezone, year(date_time, timezone), month(date_time, timezone), dayofmonth(date_time, timezone), start_time_hour, start_time_minute, 0)
    end_unix_time := timestamp(timezone, year(date_time, timezone), month(date_time, timezone), dayofmonth(date_time, timezone), end_time_hour, end_time_minute, 0)
    [start_unix_time, end_unix_time]

p_comm_time_range_to_start_unix_time(string time_range, int date_time = time, string timezone = syminfo.timezone) =>
    int start_time_hour = na
    int start_time_minute = na
    if str.length(time_range) == 11
        // Format: hh:mm-hh:mm
        start_time_hour := math.floor(str.tonumber(str.substring(time_range, 0, 2)))
        start_time_minute := math.floor(str.tonumber(str.substring(time_range, 3, 5)))
    else if str.length(time_range) == 9
        // Format: hhmm-hhmm
        start_time_hour := math.floor(str.tonumber(str.substring(time_range, 0, 2)))
        start_time_minute := math.floor(str.tonumber(str.substring(time_range, 2, 4)))
    timestamp(timezone, year(date_time, timezone), month(date_time, timezone), dayofmonth(date_time, timezone), start_time_hour, start_time_minute, 0)

p_comm_time_range_to_end_unix_time(string time_range, int date_time = time, string timezone = syminfo.timezone) =>
    int end_time_hour = na
    int end_time_minute = na
    if str.length(time_range) == 11
        end_time_hour := math.floor(str.tonumber(str.substring(time_range, 6, 8)))
        end_time_minute := math.floor(str.tonumber(str.substring(time_range, 9, 11)))
    else if str.length(time_range) == 9
        end_time_hour := math.floor(str.tonumber(str.substring(time_range, 5, 7)))
        end_time_minute := math.floor(str.tonumber(str.substring(time_range, 7, 9)))
    timestamp(timezone, year(date_time, timezone), month(date_time, timezone), dayofmonth(date_time, timezone), end_time_hour, end_time_minute, 0)

p_comm_timeframe_to_seconds(simple string tf) =>
    float seconds = 0
    tf_lower = str.lower(tf)
    value = str.tonumber(str.substring(tf_lower, 0, str.length(tf_lower) - 1))
    if str.endswith(tf_lower, 's')
        seconds := value
    else if str.endswith(tf_lower, 'd')
        seconds := value * 86400
    else if str.endswith(tf_lower, 'w')
        seconds := value * 604800
    else if str.endswith(tf_lower, 'm')
        seconds := value * 2592000
    else
        seconds := str.tonumber(tf_lower) * 60
    seconds

p_custom_sources() =>
    [open, high, low, close, volume]

//#endregion —————————————————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Ta Dependence


//#endregion —————————————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Constants

// Input Groups
string P_GP_1      =      ""

//#endregion —————————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Inputs

// Default
int p_inp_1        =      input.int(defval=20, title="Fast EMA Length", group=P_GP_1)
int p_inp_2        =      input.int(defval=50, title="Slow EMA Length", group=P_GP_1)
float p_inp_3      =      input.float(defval=1.3, title="Take Profit Price Multiplier", group=P_GP_1, step=0.01)
float p_inp_4      =      input.float(defval=0.95, title="Stop Loss Price Multiplier", group=P_GP_1, step=0.01)


//#endregion ———————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Price Data



//#endregion ———————————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Indicators

p_ind_1      =      ta.ema(close, p_inp_1) // Fast EMA
p_ind_2      =      ta.ema(close, p_inp_2) // Slow EMA


//#endregion ———————————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Conditions

p_cond_1      =      (ta.crossover(p_ind_1, p_ind_2))


//#endregion ———————————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Strategy

// Strategy Order Variables
string p_st_name_1                       =      "Entry"
string p_st_name_2                       =      "Exit"
var float p_st_name_2_tp                 =      na
var bool p_st_name_2_tp_can_drawing      =      true
var float p_st_name_2_sl                 =      na
var bool p_st_name_2_sl_can_drawing      =      true

// Strategy Global
open_trades_number = strategy.opentrades
pre_bar_open_trades_number = na(open_trades_number[1]) ? 0 : open_trades_number[1]
var p_entry_order_id = 1
p_can_place_entry_order() =>
    strategy.equity > 0
get_entry_id_name(int current_order_id, string name) =>
    "[" + str.tostring(current_order_id) + "] " + name
is_entry_order(string order_id, string name) =>
    str.startswith(order_id, "[") and str.endswith(order_id, "] " + name)
get_open_trades_entry_ids() =>
    int p_open_trades_count = strategy.opentrades
    string[] p_entry_ids = array.new_string(0, "")
    if p_open_trades_count > 0
        for i = 0 to p_open_trades_count - 1
            array.push(p_entry_ids, strategy.opentrades.entry_id(i))
    p_entry_ids

// Entry (Entry)
if p_cond_1 and p_can_place_entry_order()
    p_st_name_1_id = get_entry_id_name(p_entry_order_id, p_st_name_1)
    p_entry_order_id := p_entry_order_id + 1
    string entry_message = ""
    strategy.entry(id=p_st_name_1_id, direction=strategy.long, alert_message=entry_message, comment=p_st_name_1_id)
    
    // TP/SL Exit (Exit)
    float p_st_name_2_limit = close * p_inp_3
    if p_st_name_2_tp_can_drawing
        p_st_name_2_tp_can_drawing := false
        p_st_name_2_tp := p_st_name_2_limit
    float p_st_name_2_stop = close * p_inp_4
    if p_st_name_2_sl_can_drawing
        p_st_name_2_sl_can_drawing := false
        p_st_name_2_sl := p_st_name_2_stop
    string p_st_name_2_alert_message = ""
    strategy.exit(id=p_st_name_1_id + "_0", from_entry=p_st_name_1_id, qty_percent=100, limit=p_st_name_2_limit, stop=p_st_name_2_stop, comment_profit=p_st_name_2 + " - TP", comment_loss=p_st_name_2 + " - SL", alert_message=p_st_name_2_alert_message)
    

if high >= p_st_name_2_tp or (pre_bar_open_trades_number > 0 and open_trades_number == 0)
    p_st_name_2_tp_can_drawing := true
    p_st_name_2_sl_can_drawing := true
    p_st_name_2_tp := na
    p_st_name_2_sl := na
plot(p_st_name_2_tp, title="Exit - TP", color=color.rgb(0, 150, 136, 0), linewidth=1, style = plot.style_circles)
if low <= p_st_name_2_sl or (pre_bar_open_trades_number > 0 and open_trades_number == 0)
    p_st_name_2_sl_can_drawing := true
    p_st_name_2_tp_can_drawing := true
    p_st_name_2_sl := na
    p_st_name_2_tp := na
plot(p_st_name_2_sl, title="Exit - SL", color=color.rgb(244, 67, 54, 0), linewidth=1, style = plot.style_circles)
//#endregion —————————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Indicator Plots

// Fast EMA
plot(p_ind_1, "Fast EMA", color.rgb(33, 150, 243, 0), 1)

// Slow EMA
plot(p_ind_2, "Slow EMA", color.rgb(255, 82, 82, 0), 1)

//#endregion ————————————————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Custom Plots

//#endregion —————————————————————————————————————————————————————————————


//#region —————————————————————————————————————————————————— Alert

//#endregion ——————————————————————————————————————————————————————