在商品期货市场中,交易者面临的不仅是价格的涨跌波动,还有如何在这些波动中找到稳定盈利的机会。而胜率和盈亏比作为两个核心指标,能够帮助交易者量化策略的表现,提升交易的科学性和可持续性。本文将深入探讨胜率与盈亏比的意义、相互关系,以及如何通过量化平台来实现这些指标的计算与分析。
在期货交易中,胜率和盈亏比决定了策略的成败。尽管它们是两个独立的概念,但它们之间的相互作用对于交易结果至关重要。
胜率指的是在一段时间内,盈利交易占总交易次数的百分比。公式如下:
[ 胜率 = \frac{盈利交易次数}{总交易次数} \times 100% ]
举个例子,假设你在一个月内进行了100次交易,其中50次交易是盈利的,则胜率为50%。尽管胜率是衡量交易表现的重要指标,但仅依赖高胜率并不一定能带来稳定的收益。
盈亏比衡量每次盈利交易的平均收益与每次亏损交易的平均损失的比率,公式如下:
[ 盈亏比 = \frac{每次盈利的平均收益}{每次亏损的平均损失} ]
举个例子,假设你平均每次盈利100美元,每次亏损50美元,则盈亏比为2:1。这意味着即便胜率不高,通过较高的盈亏比仍然可以实现盈利。
胜率与盈亏比往往是相辅相成的。当胜率较高时,即使盈亏比相对较低,仍可能实现较好的收益;反之,盈亏比较高也可以在胜率较低的情况下确保盈利。
在现代量化交易平台上,胜率与盈亏比的计算通过自动化算法实现。我们以优宽量化平台为例,展示如何通过编写策略代码来实现这些指标的追踪与计算。
let openContracts = {}; // 追踪所有未平仓的合约
let profitLossHistory = {}; // 追踪每个合约的盈亏历史
let processedOrderIds = new Set(); // 记录已处理的订单 ID
let winLossCount = { win: 0, loss: 0, total: 0 }; // 统计胜败和总交易数量
function updateContracts(newOrders, commissionFee) {
let totalProfit = 0;
let totalLoss = 0;
for (let order of newOrders) {
let symbol = order.Symbol;
let orderId = order.Id;
// 检查订单状态,如果订单未成交(status !== 1),则跳过该订单
if (order.Status !== 1) {
continue;
}
// 如果订单已处理,跳过
if (processedOrderIds.has(orderId)) {
continue;
}
// 初始化合约追踪信息
if (!openContracts[symbol]) {
openContracts[symbol] = {
longPositions: [], // 保存多单开仓的价格和数量
shortPositions: [] // 保存空单开仓的价格和数量
};
}
// 处理开仓和加仓订单
if (order.Type === 0 && order.Offset === 0) { // 买入开仓(多单)
openContracts[symbol].longPositions.push({ amount: order.Amount, price: order.AvgPrice });
} else if (order.Type === 1 && order.Offset === 0) { // 卖出开仓(空单)
openContracts[symbol].shortPositions.push({ amount: order.Amount, price: order.AvgPrice });
}
// 处理平仓订单
if (order.Offset === 1) {
let remainingAmount = order.Amount;
let profitOrLoss = 0;
let isError = false;
// 平多单
if (order.Type === 1) {
if (openContracts[symbol].longPositions.length === 0) {
Log(`错误: 没有多单开仓记录,无法平仓,订单ID ${orderId}`);
isError = true;
} else {
while (remainingAmount > 0 && openContracts[symbol].longPositions.length > 0) {
let position = openContracts[symbol].longPositions[0];
let amountToClose = Math.min(position.amount, remainingAmount);
// 计算盈亏
profitOrLoss += (order.AvgPrice - position.price) * amountToClose;
remainingAmount -= amountToClose;
openContracts[symbol].longPositions[0].amount -= amountToClose;
if (openContracts[symbol].longPositions[0].amount === 0) {
openContracts[symbol].longPositions.shift();
}
}
// 如果仍然有未平仓的数量,记录错误
if (remainingAmount > 0) {
Log(`错误: 平仓数量大于开仓数量,订单ID ${orderId}`);
isError = true;
}
}
}
// 平空单
if (order.Type === 0) {
if (openContracts[symbol].shortPositions.length === 0) {
Log(`错误: 没有空单开仓记录,无法平仓,订单ID ${orderId}`);
isError = true;
} else {
while (remainingAmount > 0 && openContracts[symbol].shortPositions.length > 0) {
let position = openContracts[symbol].shortPositions[0];
let amountToClose = Math.min(position.amount, remainingAmount);
// 计算盈亏
profitOrLoss += (position.price - order.AvgPrice) * amountToClose;
remainingAmount -= amountToClose;
openContracts[symbol].shortPositions[0].amount -= amountToClose;
if (openContracts[symbol].shortPositions[0].amount === 0) {
openContracts[symbol].shortPositions.shift();
}
}
// 如果仍然有未平仓的数量,记录错误
if (remainingAmount > 0) {
Log(`错误: 平仓数量大于开仓数量,订单ID ${orderId}`);
isError = true;
}
}
}
if (!isError) {
// 扣除手续费并更新盈亏
let volumeMultiple = exchange.SetContractType(symbol).VolumeMultiple // 合约乘数
let totalCommissionFee = commissionFee * order.Amount * 2 /volumeMultiple; // 买入和卖出的手续费
profitOrLoss -= totalCommissionFee;
// 更新盈亏历史
if (profitOrLoss > 0) {
totalProfit += profitOrLoss;
winLossCount.win++;
} else {
totalLoss += Math.abs(profitOrLoss);
winLossCount.loss++;
}
winLossCount.total++;
}
}
// 记录订单 ID 为已处理
processedOrderIds.add(orderId);
}
// 记录盈亏历史
profitLossHistory.totalProfit = (profitLossHistory.totalProfit || 0) + totalProfit;
profitLossHistory.totalLoss = (profitLossHistory.totalLoss || 0) + totalLoss;
// 计算盈亏比
let winLossRatio = profitLossHistory.totalLoss > 0
? profitLossHistory.totalProfit > 0 ? profitLossHistory.totalProfit / profitLossHistory.totalLoss : -1
: profitLossHistory.totalProfit > 0 ? 1 : 0; // 避免除以0
// 计算胜率
let winRate = winLossCount.total > 0
? winLossCount.win / winLossCount.total
: 0;
return [winLossRatio, winRate];
}
在期货交易中,追踪每一笔成交订单是计算胜率和盈亏比的基础。通过 exchange.GetHistoryOrders()
函数,平台可以获取当日所有的已成交订单,包括开仓和平仓的详细信息。
exchange.GetHistoryOrders()
该函数用于获取当日的所有已成交订单,返回的订单信息包含了订单类型、成交价格、成交数量等关键信息。通过分析这些订单数据,交易者可以精准地计算出每笔交易的盈亏情况。
let orders = exchange.GetHistoryOrders();
for (let i = 0; i < orders.length; i++) {
let order = orders[i];
// 输出每个订单的详细信息
Log("Order ID:", order.Id,
"合约符号:", order.Symbol,
"订单类型 (Type):", order.Type === 0 ? "买入" : "卖出",
"开/平仓 (Offset):", order.Offset === 0 ? "开仓" : "平仓",
"状态 (Status):", order.Status === 1 ? "已成交" : "未成交",
"数量:", order.Amount,
"成交价格:", order.AvgPrice
}
通过遍历 GetHistoryOrders
返回的订单数据,可以获取当日所有的已成交订单,并进行进一步的盈亏计算。
在实际交易中,每笔交易通常由多个订单组成,包括开仓订单与平仓订单。要准确计算每笔交易的盈亏,首先需要追踪每个未平仓合约,并在平仓时根据开仓与平仓的价差计算盈亏。
let openContracts = {}; // 追踪所有未平仓的合约
let profitLossHistory = {}; // 追踪每个合约的盈亏历史
let processedOrderIds = new Set(); // 记录已处理的订单 ID
let winLossCount = { win: 0, loss: 0, total: 0 }; // 统计胜败和总交易数量
当收到新的开仓订单时,系统会根据合约类型(多单或空单)将订单价格与数量加入对应的未平仓合约列表中:
if (order.Type === 0 && order.Offset === 0) { // 买入开仓(多单)
openContracts[symbol].longPositions.push({ amount: order.Amount, price: order.AvgPrice });
} else if (order.Type === 1 && order.Offset === 0) { // 卖出开仓(空单)
openContracts[symbol].shortPositions.push({ amount: order.Amount, price: order.AvgPrice });
}
当接收到平仓订单时,系统会根据 FIFO(First In, First Out)规则,匹配相应的开仓合约来计算盈亏。这一规则保证了最早的合约被优先平仓。结合代码中的逻辑:
处理未平仓的多单:
while
循环,系统会尝试平掉所有未平仓的多单头寸。openContracts[symbol].longPositions[0]
表示获取该合约最早的开仓多单。这正是 FIFO 规则的体现。逐步平仓:
amountToClose
变量用于确定本次平仓可以平掉多少仓位,取未平仓数量 remainingAmount
与当前持仓数量 position.amount
的最小值。这样可以保证一次性处理平仓量不超过开仓量。盈亏计算:
(order.AvgPrice - position.price) * amountToClose
来计算平仓的盈亏。这里 order.AvgPrice
是平仓的价格,position.price
是最早开仓时的价格,两者之差乘以平掉的仓位数量得到这笔交易的盈亏。更新剩余平仓量和开仓头寸:
remainingAmount
被减少,而该开仓头寸的数量 openContracts[symbol].longPositions[0].amount
也相应减少。如果当前头寸被完全平掉(即 amount
变为 0),则该头寸会从 longPositions
队列中移除(通过 shift()
)。错误处理:
remainingAmount > 0
),则说明出现了平仓数量大于开仓数量的错误,系统会记录日志并标记 isError = true
。这段代码的核心在于通过 FIFO 原则,确保按照开仓时间的顺序,逐步处理多单平仓,同时准确计算每笔平仓操作的盈亏并更新持仓情况。注意代码中对盈亏还进行了手续费的处理。
在每次交易结束后,系统会动态更新胜率和盈亏比。胜率的计算基于盈利交易次数与总交易次数的比率,盈亏比则是盈利总金额与亏损总金额的比值。
let winLossRatio = profitLossHistory.totalLoss > 0
? profitLossHistory.totalProfit > 0 ? profitLossHistory.totalProfit / profitLossHistory.totalLoss : -1
: profitLossHistory.totalProfit > 0 ? 1 : 0;
let winRate = winLossCount.total > 0
? winLossCount.win / winLossCount.total
: 0;
无论是在商品期货交易还是在其他市场,胜率与盈亏比都是决定策略成败的核心因素。通过量化平台如优宽量化,交易者可以方便地追踪这些指标,并实时调整策略。在量化平台的帮助下,复杂的盈亏计算变得直观透明,为交易者提供了重要的决策依据。