MQL5 Wizard(1): 不需手寫程式碼建立EA

更新於 發佈於 閱讀時間約 41 分鐘
  • 文內如有投資理財相關經驗、知識、資訊等內容,皆為創作者個人分享行為。
  • 有價證券、指數與衍生性商品之數據資料,僅供輔助說明之用,不代表創作者投資決策之推介及建議。
  • 閱讀同時,請審慎思考自身條件及自我決策,並應有為決策負責之事前認知。
  • 方格子希望您能從這些分享內容汲取投資養份,養成獨立思考的能力、判斷、行動,成就最適合您的投資理財模式。

引言

紀錄閱讀MQL5論壇上EA交易工具相關的心得或翻譯

由於後面有一系列文章是使用mql5 wizard這個功能來寫EA

因此先有一個前導系列來研究mql5 wizard這個功能

至於mql5基礎語言由於筆者本身有基礎背景

因此就先不介紹了,後續如果有需要再考慮


文章出處

MQL5 Wizard: Creating Expert Advisors without Programming


介紹

當在建立自動交易系統時

必須要編寫有關"交易訊號""追蹤平倉""資金風險管理"等邏輯

當各個模組編寫完成,最困難的任務就是組裝各個模組並調適算法

而在這個基礎之上還需要有一個讓各個模組互動正常的架構

若架構規劃的不好,則還需要花時間排查架構錯誤,替換模組時還有可能要改寫架構

在MQL5中,使用了物件導向的方法來簡化自動交易系統的編寫和測試流程,

尤其MetaQuotes Software Corp.還內建了基本的交易訊號(20個)、追蹤平倉(4)和資金風險管理模組(5)

透過組合這些模組就可以立即得到一個可用的EA

接下來會介紹如何自動產生一套EA且不需要自己手寫程式碼


使用MQL5 Wizard建立EA

接下來的步驟將在MetaEditor中完成

其中內建的交易模組位於"\<client_terminal_directory>\MQL5\Include\Expert\"路徑的資料夾內

當在開啟MetaEditor後,MQL5 Wizard會到這個路徑解析這些模組並新增到選單中

之後在新增EA時就可以透過選單來選擇要組合哪些模組


  • 接下來到工具>MetaQuotes語言編輯器,或按(F4)來開啟MetaEditor,
  • 接著一樣左上角按新增EA的按鈕(或ctrl+n),並選擇EA交易(生成)
raw-image
  • 輸入名稱等其他資訊
raw-image
  • 增加選擇交易訊號
    • 交易訊號會決定什麼條件下開倉,平倉,反轉部位,圖中是內建的交易訊號
raw-image
    • 之後可以設定交易訊號的相關參數
      參數具有兩種模式,圖中灰色底的(如shift,applied),代表生成EA之後無法在調適EA的階段再次修改參數值,用於加速EA調適的速度,反之則為白色底,可以透過滑鼠左鍵雙擊切換
raw-image
    • 加入之後如下圖
raw-image
  • 增加追蹤平倉模組
raw-image
  • 增加資金風險管理模組
raw-image
  • 之後按下完成就會生成EA
raw-image



解析EA程式碼與架構

  • 生成的程式碼如下
    //+------------------------------------------------------------------+
    //| Test001.mq5 |
    //| Copyright 2024, MetaQuotes Ltd. |
    //| <https://www.mql5.com> |
    //+------------------------------------------------------------------+
    #property copyright "Copyright 2024, MetaQuotes Ltd."
    #property link "<https://www.mql5.com>"
    #property version "1.00"
    //+------------------------------------------------------------------+
    //| Include |
    //+------------------------------------------------------------------+
    #include <Expert\\Expert.mqh>
    //--- available signals
    #include <Expert\\Signal\\SignalMA.mqh>
    //--- available trailing
    #include <Expert\\Trailing\\TrailingFixedPips.mqh>
    //--- available money management
    #include <Expert\\Money\\MoneyFixedLot.mqh>
    //+------------------------------------------------------------------+
    //| Inputs |
    //+------------------------------------------------------------------+
    //--- inputs for expert
    input string Expert_Title ="Test001"; // Document name
    ulong Expert_MagicNumber =13666; //
    bool Expert_EveryTick =false; //
    //--- inputs for main signal
    input int Signal_ThresholdOpen =10; // Signal threshold value to open [0...100]
    input int Signal_ThresholdClose =10; // Signal threshold value to close [0...100]
    input double Signal_PriceLevel =0.0; // Price level to execute a deal
    input double Signal_StopLevel =50.0; // Stop Loss level (in points)
    input double Signal_TakeLevel =50.0; // Take Profit level (in points)
    input int Signal_Expiration =4; // Expiration of pending orders (in bars)
    input int Signal_MA_PeriodMA =12; // Moving Average(12,0,...) Period of averaging
    input int Signal_MA_Shift =0; // Moving Average(12,0,...) Time shift
    input ENUM_MA_METHOD Signal_MA_Method =MODE_SMA; // Moving Average(12,0,...) Method of averaging
    input ENUM_APPLIED_PRICE Signal_MA_Applied =PRICE_CLOSE; // Moving Average(12,0,...) Prices series
    input double Signal_MA_Weight =1.0; // Moving Average(12,0,...) Weight [0...1.0]
    //--- inputs for trailing
    input int Trailing_FixedPips_StopLevel =30; // Stop Loss trailing level (in points)
    input int Trailing_FixedPips_ProfitLevel=50; // Take Profit trailing level (in points)
    //--- inputs for money
    input double Money_FixLot_Percent =10.0; // Percent
    input double Money_FixLot_Lots =0.1; // Fixed volume
    //+------------------------------------------------------------------+
    //| Global expert object |
    //+------------------------------------------------------------------+
    CExpert ExtExpert;
    //+------------------------------------------------------------------+
    //| Initialization function of the expert |
    //+------------------------------------------------------------------+
    int OnInit()
    {
    //--- Initializing expert
    if(!ExtExpert.Init(Symbol(),Period(),Expert_EveryTick,Expert_MagicNumber))
    {
    //--- failed
    printf(__FUNCTION__+": error initializing expert");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Creating signal
    CExpertSignal *signal=new CExpertSignal;
    if(signal==NULL)
    {
    //--- failed
    printf(__FUNCTION__+": error creating signal");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //---
    ExtExpert.InitSignal(signal);
    signal.ThresholdOpen(Signal_ThresholdOpen);
    signal.ThresholdClose(Signal_ThresholdClose);
    signal.PriceLevel(Signal_PriceLevel);
    signal.StopLevel(Signal_StopLevel);
    signal.TakeLevel(Signal_TakeLevel);
    signal.Expiration(Signal_Expiration);
    //--- Creating filter CSignalMA
    CSignalMA *filter0=new CSignalMA;
    if(filter0==NULL)
    {
    //--- failed
    printf(__FUNCTION__+": error creating filter0");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    signal.AddFilter(filter0);
    //--- Set filter parameters
    filter0.PeriodMA(Signal_MA_PeriodMA);
    filter0.Shift(Signal_MA_Shift);
    filter0.Method(Signal_MA_Method);
    filter0.Applied(Signal_MA_Applied);
    filter0.Weight(Signal_MA_Weight);
    //--- Creation of trailing object
    CTrailingFixedPips *trailing=new CTrailingFixedPips;
    if(trailing==NULL)
    {
    //--- failed
    printf(__FUNCTION__+": error creating trailing");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Add trailing to expert (will be deleted automatically))
    if(!ExtExpert.InitTrailing(trailing))
    {
    //--- failed
    printf(__FUNCTION__+": error initializing trailing");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Set trailing parameters
    trailing.StopLevel(Trailing_FixedPips_StopLevel);
    trailing.ProfitLevel(Trailing_FixedPips_ProfitLevel);
    //--- Creation of money object
    CMoneyFixedLot *money=new CMoneyFixedLot;
    if(money==NULL)
    {
    //--- failed
    printf(__FUNCTION__+": error creating money");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Add money to expert (will be deleted automatically))
    if(!ExtExpert.InitMoney(money))
    {
    //--- failed
    printf(__FUNCTION__+": error initializing money");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Set money parameters
    money.Percent(Money_FixLot_Percent);
    money.Lots(Money_FixLot_Lots);
    //--- Check all trading objects parameters
    if(!ExtExpert.ValidationSettings())
    {
    //--- failed
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Tuning of all necessary indicators
    if(!ExtExpert.InitIndicators())
    {
    //--- failed
    printf(__FUNCTION__+": error initializing indicators");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- ok
    return(INIT_SUCCEEDED);
    }
    //+------------------------------------------------------------------+
    //| Deinitialization function of the expert |
    //+------------------------------------------------------------------+
    void OnDeinit(const int reason)
    {
    ExtExpert.Deinit();
    }
    //+------------------------------------------------------------------+
    //| "Tick" event handler function |
    //+------------------------------------------------------------------+
    void OnTick()
    {
    ExtExpert.OnTick();
    }
    //+------------------------------------------------------------------+
    //| "Trade" event handler function |
    //+------------------------------------------------------------------+
    void OnTrade()
    {
    ExtExpert.OnTrade();
    }
    //+------------------------------------------------------------------+
    //| "Timer" event handler function |
    //+------------------------------------------------------------------+
    void OnTimer()
    {
    ExtExpert.OnTimer();
    }
    //+------------------------------------------------------------------+

  • 程式碼的基本資料描述
    //+------------------------------------------------------------------+
    //| Test001.mq5 |
    //| Copyright 2024, MetaQuotes Ltd. |
    //| <https://www.mql5.com> |
    //+------------------------------------------------------------------+
    #property copyright "Copyright 2024, MetaQuotes Ltd."
    #property link "<https://www.mql5.com>"
    #property version "1.00"
  • 引入的參考檔案
    //+------------------------------------------------------------------+
    //| Include |
    //+------------------------------------------------------------------+
    #include <Expert\\Expert.mqh>
    //--- available signals
    #include <Expert\\Signal\\SignalMA.mqh>
    //--- available trailing
    #include <Expert\\Trailing\\TrailingFixedPips.mqh>
    //--- available money management
    #include <Expert\\Money\\MoneyFixedLot.mqh>
    • Expert.mqh:用來組合各個模組
    • 其餘三個檔案是剛剛選擇的各個模組
  • 接來來是之後可以被調適的EA參數
    //+------------------------------------------------------------------+
    //| Inputs |
    //+------------------------------------------------------------------+
    //--- inputs for expert
    input string Expert_Title ="Test001"; // Document name
    ulong Expert_MagicNumber =13666; //
    bool Expert_EveryTick =false; //
    //--- inputs for main signal
    input int Signal_ThresholdOpen =10; // Signal threshold value to open [0...100]
    input int Signal_ThresholdClose =10; // Signal threshold value to close [0...100]
    input double Signal_PriceLevel =0.0; // Price level to execute a deal
    input double Signal_StopLevel =50.0; // Stop Loss level (in points)
    input double Signal_TakeLevel =50.0; // Take Profit level (in points)
    input int Signal_Expiration =4; // Expiration of pending orders (in bars)
    input int Signal_MA_PeriodMA =12; // Moving Average(12,0,...) Period of averaging
    input int Signal_MA_Shift =0; // Moving Average(12,0,...) Time shift
    input ENUM_MA_METHOD Signal_MA_Method =MODE_SMA; // Moving Average(12,0,...) Method of averaging
    input ENUM_APPLIED_PRICE Signal_MA_Applied =PRICE_CLOSE; // Moving Average(12,0,...) Prices series
    input double Signal_MA_Weight =1.0; // Moving Average(12,0,...) Weight [0...1.0]
    //--- inputs for trailing
    input int Trailing_FixedPips_StopLevel =30; // Stop Loss trailing level (in points)
    input int Trailing_FixedPips_ProfitLevel=50; // Take Profit trailing level (in points)
    //--- inputs for money
    input double Money_FixLot_Percent =10.0; // Percent
    input double Money_FixLot_Lots =0.1; // Fixed volume
    • 起頭的三個是基本參數(Expert_Title/Expert_MagicNumber/Expert_EveryTick)
      Expert_Title:EA名稱
      Expert_MagicNumber:用於管理EA的交易訂單
      Expert_EveryTick:是否在每次價格tick跳動時都執行EA邏輯
    • 後面參數則會跟選擇的模組不同而變動
  • 宣告組合模組的EA變數
    //+------------------------------------------------------------------+
    //| Global expert object |
    //+------------------------------------------------------------------+
    CExpert ExtExpert;
  • 設定組合各模組到EA變數內
    //+------------------------------------------------------------------+
    //| Initialization function of the expert |
    //+------------------------------------------------------------------+
    int OnInit()
    {
    ...
    • 在OnInit()中進行初始化,此函數只會在EA開啟時執行一次
  • 接下來設定ExtExpert的初始參數
    //--- Initializing expert
    if(!ExtExpert.Init(Symbol(),Period(),Expert_EveryTick,Expert_MagicNumber))
    {
    //--- failed
    printf(__FUNCTION__+": error initializing expert");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    • 透過ExtExpert.Init(Symbol(),Period(),Expert_EveryTick,Expert_MagicNumber)
      把基本參數配置到ExtExpert, 如果初始化失敗則會回傳-1,然後進入if判斷執行結束EA相關程序
  • 設定交易訊號模組的參數並配置到ExtExpert
    //--- Creating signal
    CExpertSignal *signal=new CExpertSignal;
    if(signal==NULL)
    {
    //--- failed
    printf(__FUNCTION__+": error creating signal");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //---
    ExtExpert.InitSignal(signal);
    signal.ThresholdOpen(Signal_ThresholdOpen);
    signal.ThresholdClose(Signal_ThresholdClose);
    signal.PriceLevel(Signal_PriceLevel);
    signal.StopLevel(Signal_StopLevel);
    signal.TakeLevel(Signal_TakeLevel);
    signal.Expiration(Signal_Expiration);
    //--- Creating filter CSignalMA
    CSignalMA *filter0=new CSignalMA;
    if(filter0==NULL)
    {
    //--- failed
    printf(__FUNCTION__+": error creating filter0");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    signal.AddFilter(filter0);
    //--- Set filter parameters
    filter0.PeriodMA(Signal_MA_PeriodMA);
    filter0.Shift(Signal_MA_Shift);
    filter0.Method(Signal_MA_Method);
    filter0.Applied(Signal_MA_Applied);
    filter0.Weight(Signal_MA_Weight);
    • 交易訊號分為兩個步驟
    • 建立訊號變數signal,配置到ExtExpert
    • 把各個交易模組當成一層的filter(過濾器),
      透過加入到signal.AddFilter(filter0),
      只有當價格行為通過層層過濾器後,才能發出交易訊號
  • 設定追蹤平倉變數
    //--- Creation of trailing object
    CTrailingFixedPips *trailing=new CTrailingFixedPips;
    if(trailing==NULL)
    {
    //--- failed
    printf(__FUNCTION__+": error creating trailing");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Add trailing to expert (will be deleted automatically))
    if(!ExtExpert.InitTrailing(trailing))
    {
    //--- failed
    printf(__FUNCTION__+": error initializing trailing");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Set trailing parameters
    trailing.StopLevel(Trailing_FixedPips_StopLevel);
    trailing.ProfitLevel(Trailing_FixedPips_ProfitLevel);
    • 建立變數trailing,配置到ExtExpert
  • 設定資金風險管理變數
    //--- Creation of money object
    CMoneyFixedLot *money=new CMoneyFixedLot;
    if(money==NULL)
    {
    //--- failed
    printf(__FUNCTION__+": error creating money");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Add money to expert (will be deleted automatically))
    if(!ExtExpert.InitMoney(money))
    {
    //--- failed
    printf(__FUNCTION__+": error initializing money");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Set money parameters
    money.Percent(Money_FixLot_Percent);
    money.Lots(Money_FixLot_Lots);
    • 建立變數money,配置到ExtExpert
  • 設定好ExtExpert之後驗證所有參數設置,並把剛剛配置的變數都跑一次初始化
    //--- Check all trading objects parameters
    if(!ExtExpert.ValidationSettings())
    {
    //--- failed
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
    //--- Tuning of all necessary indicators
    if(!ExtExpert.InitIndicators())
    {
    //--- failed
    printf(__FUNCTION__+": error initializing indicators");
    ExtExpert.Deinit();
    return(INIT_FAILED);
    }
  • 接下來就同一般的EA一樣,執行MQL5基本參數,然後再透過ExtExpert觸發各模組的程式邏輯
    //+------------------------------------------------------------------+
    //| Deinitialization function of the expert |
    //+------------------------------------------------------------------+
    void OnDeinit(const int reason)
    {
    ExtExpert.Deinit();
    }
    //+------------------------------------------------------------------+
    //| "Tick" event handler function |
    //+------------------------------------------------------------------+
    void OnTick()
    {
    ExtExpert.OnTick();
    }
    //+------------------------------------------------------------------+
    //| "Trade" event handler function |
    //+------------------------------------------------------------------+
    void OnTrade()
    {
    ExtExpert.OnTrade();
    }
    //+------------------------------------------------------------------+
    //| "Timer" event handler function |
    //+------------------------------------------------------------------+
    void OnTimer()
    {
    ExtExpert.OnTimer();
    }
  • 編譯EA確定沒有錯誤和警告
raw-image


結語

透過MQL5 Wizard可以簡化交易想法的建立和測試

也可以自己建立各個模組

並且可以重複使用,組合不同的模組來建立自己獨有的交易程式

且開發組來的

  • 補充說明:內文提供的Moving Average EA只是範例,並無實際盈利能力,具體邏輯還需使用者自行思考設計
avatar-img
2會員
5內容數
RRRRRRRR
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
你可能也想看
Google News 追蹤
Thumbnail
隨著理財資訊的普及,越來越多台灣人不再將資產侷限於台股,而是將視野拓展到國際市場。特別是美國市場,其豐富的理財選擇,讓不少人開始思考將資金配置於海外市場的可能性。 然而,要參與美國市場並不只是盲目跟隨標的這麼簡單,而是需要策略和方式,尤其對新手而言,除了選股以外還會遇到語言、開戶流程、Ap
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
本篇文章分享了自行開發的臺指期當沖策略,並使用XQ全球贏家進行自動化交易買賣。主要內容包括今日交易重點、當日損益、自動交易損益計算與績效圖以及各交易策略說明。
Thumbnail
股市交易中,若能透過程式來輔助進出場的訊號,當手上的持股或者股池數量較多時,可以提供投資人較有效率的判斷,因此非常建議投資人花費一點時間了解程式交易的邏輯,比較常見的就是XQ全球贏家的"盤中量化模組(1000NT/月)" 或者比較簡易型的量化積木模組(500NT/月)。 每個投資者會因學習的過
Thumbnail
歡迎加入學習選擇權交易行列,請參考以下網址,加入後便可以得知完整交易策略(未來還會持續優化策略) https://vocus.cc/optioncreatemoney/introduce 要透過選擇權賺錢必須要事先規劃行情發展,因此願意學習的付費讀者務必隨時追蹤專欄。現在來複習本專欄特殊的買方策
Thumbnail
本篇文章分享了自行開發的臺指期當沖策略,並使用XQ全球贏家進行自動化交易買賣。主要內容包括今日交易重點、當日損益、自動交易損益計算與績效圖以及各交易策略說明。
Thumbnail
借助 MT5 Webtrader,您无需安装任何应用程序即可通过手机进行交易,受到了交易者的高度热情欢迎。
Thumbnail
  本篇文章分享了自行開發的臺指期當沖策略,並使用XQ全球贏家進行自動化交易買賣。主要內容包括今日交易重點、當日損益、自動交易損益計算與績效圖以及各交易策略說明。文章強調策略的執行和分享交易績效的重要性。
Thumbnail
  本篇文章分享了自行開發的臺指期當沖策略,並使用XQ全球贏家進行自動化交易買賣。主要內容包括今日交易重點、當日損益、自動交易損益計算與績效圖以及各交易策略說明。文章強調策略的執行和分享交易績效的重要性。
Thumbnail
Exness 提供多种可靠的交易软件以及多款热门软件,帮助交易者开启专业、便捷的交易体验。其中,MetaTrader 4 和 MetaTrader 5 是最常见且用户量最为庞大的两大平台。
Thumbnail
隨著理財資訊的普及,越來越多台灣人不再將資產侷限於台股,而是將視野拓展到國際市場。特別是美國市場,其豐富的理財選擇,讓不少人開始思考將資金配置於海外市場的可能性。 然而,要參與美國市場並不只是盲目跟隨標的這麼簡單,而是需要策略和方式,尤其對新手而言,除了選股以外還會遇到語言、開戶流程、Ap
Thumbnail
嘿,大家新年快樂~ 新年大家都在做什麼呢? 跨年夜的我趕工製作某個外包設計案,在工作告一段落時趕上倒數。 然後和兩個小孩過了一個忙亂的元旦。在深夜時刻,看到朋友傳來的解籤網站,興致勃勃熬夜體驗了一下,覺得非常好玩,或許有人玩過了,但還是想寫上來分享紀錄一下~
Thumbnail
本篇文章分享了自行開發的臺指期當沖策略,並使用XQ全球贏家進行自動化交易買賣。主要內容包括今日交易重點、當日損益、自動交易損益計算與績效圖以及各交易策略說明。
Thumbnail
股市交易中,若能透過程式來輔助進出場的訊號,當手上的持股或者股池數量較多時,可以提供投資人較有效率的判斷,因此非常建議投資人花費一點時間了解程式交易的邏輯,比較常見的就是XQ全球贏家的"盤中量化模組(1000NT/月)" 或者比較簡易型的量化積木模組(500NT/月)。 每個投資者會因學習的過
Thumbnail
歡迎加入學習選擇權交易行列,請參考以下網址,加入後便可以得知完整交易策略(未來還會持續優化策略) https://vocus.cc/optioncreatemoney/introduce 要透過選擇權賺錢必須要事先規劃行情發展,因此願意學習的付費讀者務必隨時追蹤專欄。現在來複習本專欄特殊的買方策
Thumbnail
本篇文章分享了自行開發的臺指期當沖策略,並使用XQ全球贏家進行自動化交易買賣。主要內容包括今日交易重點、當日損益、自動交易損益計算與績效圖以及各交易策略說明。
Thumbnail
借助 MT5 Webtrader,您无需安装任何应用程序即可通过手机进行交易,受到了交易者的高度热情欢迎。
Thumbnail
  本篇文章分享了自行開發的臺指期當沖策略,並使用XQ全球贏家進行自動化交易買賣。主要內容包括今日交易重點、當日損益、自動交易損益計算與績效圖以及各交易策略說明。文章強調策略的執行和分享交易績效的重要性。
Thumbnail
  本篇文章分享了自行開發的臺指期當沖策略,並使用XQ全球贏家進行自動化交易買賣。主要內容包括今日交易重點、當日損益、自動交易損益計算與績效圖以及各交易策略說明。文章強調策略的執行和分享交易績效的重要性。
Thumbnail
Exness 提供多种可靠的交易软件以及多款热门软件,帮助交易者开启专业、便捷的交易体验。其中,MetaTrader 4 和 MetaTrader 5 是最常见且用户量最为庞大的两大平台。