# UA-AT (Uncertainty-Aware Adaptive Thresholding) 使用指南

## 概述

`UncertaintyAwareAdaptiveThreshold` 是一个改进的自适应阈值计算器，结合了：
1. **时间维度**：基于3-Sigma准则的历史统计
2. **空间维度**：基于Ego特征不确定性的动态调节

公式：`T_final = T_stat * (1 + β * Norm(U_ego))`，其中 `T_stat = μ_t + γ * σ_t`

---

## 快速开始

### 1. 在 `cp_attack.py` 中初始化

**位置**: `cp_attack.py` 第 190-200 行附近

**原始代码**:
```python
# Initialize threshold calculator for CPS defense with dynamic threshold
threshold_calculator = None
if opt.cps_defense and opt.use_dynamic_threshold:
    from defense.mdag_grouping import AdaptiveThresholdCalculator
    threshold_calculator = AdaptiveThresholdCalculator(
        window_size=10, 
        k=2.5, 
        base_threshold=0.68,
        max_sigma_ratio=2.3
    )
    logger.info(f"[CPS Defense] Initialized AdaptiveThresholdCalculator with window_size=10")
```

**新增选项 - 使用 UA-AT**:
```python
# Initialize threshold calculator for CPS defense with dynamic threshold
threshold_calculator = None
if opt.cps_defense and opt.use_dynamic_threshold:
    # 选项1: 使用原始的 AdaptiveThresholdCalculator (时间维度)
    from defense.mdag_grouping import AdaptiveThresholdCalculator
    threshold_calculator = AdaptiveThresholdCalculator(
        window_size=10, 
        k=2.5, 
        base_threshold=0.68,
        max_sigma_ratio=2.3
    )
    logger.info(f"[CPS Defense] Initialized AdaptiveThresholdCalculator")
    
    # 选项2: 使用新的 UncertaintyAwareAdaptiveThreshold (时间+空间维度)
    # from defense.mdag_grouping import UncertaintyAwareAdaptiveThreshold
    # threshold_calculator = UncertaintyAwareAdaptiveThreshold(
    #     window_size=10,           # 滑动窗口大小
    #     gamma=3.0,                # 3-Sigma系数 (覆盖99.7%正常分布)
    #     beta=0.5,                 # 不确定性调节系数
    #     base_threshold=0.68,      # 基础阈值
    #     min_threshold=0.5,        # 阈值下限
    #     max_threshold=0.95,       # 阈值上限
    #     warmup_frames=3,          # 预热帧数
    #     uncertainty_method='entropy'  # 'entropy' 或 'variance'
    # )
    # logger.info(f"[CPS Defense] Initialized UncertaintyAwareAdaptiveThreshold (UA-AT)")
```

---

### 2. 在 `cps_defense.py` 中修改调用

**需要修改的地方**: `cps_defense()` 函数中更新阈值的部分

#### 2.1 修改函数签名（可选）

**位置**: `cps_defense.py` 第 363-366 行

如果你想让 UA-AT 成为正式选项，可以添加参数：

```python
def cps_defense(batch_data, model, dataset, perturbation, attacker_idx=1, sampling_budget=10,
                lambda1=1.0, lambda2=1.0, lambda3=1.0, tau=0.68, compute_gradients=False,
                use_mdag=False, use_dynamic_threshold=False, threshold_sensitivity=1.0,
                threshold_calculator=None,
                use_uncertainty_aware=False):  # 新增参数
```

#### 2.2 修改阈值更新逻辑

**位置**: `cps_defense.py` 第 673-685 行

**原始代码**:
```python
# Update threshold calculator with final fusion CPS score for next frame
if use_dynamic_threshold and threshold_calculator is not None:
    # Pass the fusion CPS score as a list (single value) to update calculator
    if is_ego_only:
        threshold_calculator.update([], is_ego_only=True)
        logger.info(f"[CPS Defense] Skipped updating threshold calculator (ego-only)")
    else:
        threshold_calculator.update([best_cps_score], is_ego_only=False)
        # Update defense_info with latest statistics after update
        stats = threshold_calculator.get_statistics()
        defense_info['mu'] = stats['mu']
        defense_info['sigma'] = stats['sigma']
        logger.info(f"[CPS Defense] Updated threshold calculator with final CPS score: {best_cps_score:.4f}, mu={stats['mu']}, sigma={stats['sigma']}")
```

**修改后的代码**:
```python
# Update threshold calculator with final fusion CPS score for next frame
if use_dynamic_threshold and threshold_calculator is not None:
    # 检查是否为 UncertaintyAwareAdaptiveThreshold
    from defense.mdag_grouping import UncertaintyAwareAdaptiveThreshold
    
    if isinstance(threshold_calculator, UncertaintyAwareAdaptiveThreshold):
        # UA-AT 需要 ego_features 参数
        # 获取 ego 特征 (spatial_features 的第一个元素)
        ego_features = cav_content['spatial_features'][0]  # Shape: [C, H, W]
        
        # 检测是否有恶意车辆
        detected_malicious = len(defense_info.get('malicious_agents', [])) > 0
        
        if is_ego_only:
            # ego-only 情况
            threshold_calculator.update(
                current_cps_scores=[],
                ego_features=ego_features,
                is_ego_only=True,
                detected_malicious=False
            )
            logger.info(f"[CPS Defense UA-AT] Skipped updating (ego-only)")
        else:
            # 正常情况：传入 CPS 分数和 ego 特征
            threshold_calculator.update(
                current_cps_scores=[best_cps_score],
                ego_features=ego_features,
                is_ego_only=False,
                detected_malicious=detected_malicious
            )
            # 更新 defense_info
            stats = threshold_calculator.get_statistics()
            defense_info['mu'] = stats['mu_t']
            defense_info['sigma'] = stats['sigma_t']
            defense_info['T_stat'] = stats['T_stat']
            defense_info['U_ego'] = stats['U_ego']
            defense_info['T_final'] = stats['T_final']
            logger.info(f"[CPS Defense UA-AT] Updated with CPS={best_cps_score:.4f}, "
                       f"U_ego={stats['U_ego']:.4f}, T_final={stats['T_final']:.4f}")
    else:
        # 原始 AdaptiveThresholdCalculator
        if is_ego_only:
            threshold_calculator.update([], is_ego_only=True)
            logger.info(f"[CPS Defense] Skipped updating threshold calculator (ego-only)")
        else:
            threshold_calculator.update([best_cps_score], is_ego_only=False)
            stats = threshold_calculator.get_statistics()
            defense_info['mu'] = stats['mu']
            defense_info['sigma'] = stats['sigma']
            logger.info(f"[CPS Defense] Updated threshold calculator with final CPS score: {best_cps_score:.4f}")
```

---

### 3. 同样修改 MDAG 分支中的阈值更新

**位置**: `cps_defense.py` 第 430-470 行附近（MDAG 分支开始处）

在 MDAG 分支中，也需要类似的修改来获取和使用当前阈值：

```python
# Get current threshold for this frame
if use_dynamic_threshold and threshold_calculator is not None:
    from defense.mdag_grouping import UncertaintyAwareAdaptiveThreshold
    
    if isinstance(threshold_calculator, UncertaintyAwareAdaptiveThreshold):
        # UA-AT: 需要先计算一次来获取当前阈值
        # 注意：这里只是获取当前阈值，不更新窗口
        tau = threshold_calculator.T_final if threshold_calculator.T_final is not None else threshold_calculator.base_threshold
        logger.info(f"[MDAG] Using UA-AT threshold: {tau:.4f}")
    else:
        # 原始计算器
        tau = threshold_calculator.current_threshold
        logger.info(f"[MDAG] Using dynamic threshold: {tau:.4f}")
else:
    logger.info(f"[MDAG] Using base threshold: {tau:.4f}")
```

---

## 完整示例：端到端使用

### 示例 1: 在 cp_attack.py 中启用 UA-AT

```python
# 1. 初始化 UA-AT
from defense.mdag_grouping import UncertaintyAwareAdaptiveThreshold

threshold_calculator = UncertaintyAwareAdaptiveThreshold(
    window_size=10,
    gamma=3.0,              # 3-Sigma准则
    beta=0.5,               # 不确定性影响权重
    base_threshold=0.68,
    min_threshold=0.5,
    max_threshold=0.95,
    warmup_frames=3,
    uncertainty_method='entropy'  # 或 'variance'
)

# 2. 在主循环中调用 cps_defense
for i, batch_data in enumerate(data_loader):
    # ... 准备数据和攻击 ...
    
    pred_box_tensor, pred_score, gt_box_tensor, cps_score, defense_info = cps_defense(
        batch_data, model, opencood_dataset, perturbation,
        attacker_idx=attacker_idx,
        sampling_budget=10,
        lambda1=1.0, lambda2=1.0, lambda3=1.0,
        tau=0.68,  # 初始阈值
        compute_gradients=False,
        use_mdag=True,  # 或 False
        use_dynamic_threshold=True,  # 必须为 True
        threshold_calculator=threshold_calculator  # 传入 UA-AT 实例
    )
    
    # 3. 查看统计信息
    stats = threshold_calculator.get_statistics()
    print(f"Frame {i}:")
    print(f"  T_stat (统计阈值): {stats['T_stat']:.4f}")
    print(f"  U_ego (不确定性): {stats['U_ego']:.4f}")
    print(f"  T_final (最终阈值): {stats['T_final']:.4f}")
```

---

## 参数调优指南

### 核心参数说明

| 参数 | 默认值 | 说明 | 调优建议 |
|------|--------|------|----------|
| `window_size` | 10 | 滑动窗口大小 | 增大→更平滑但反应慢；减小→更敏感但不稳定 |
| `gamma` | 3.0 | 3-Sigma系数 | 3.0覆盖99.7%正常分布；减小→更严格 |
| `beta` | 0.5 | 不确定性权重 | 增大→恶劣环境更宽容；减小→更严格 |
| `base_threshold` | 0.68 | 基础阈值 | 根据CPS分数分布调整 |
| `uncertainty_method` | 'entropy' | 不确定性计算方法 | 'entropy'更理论化；'variance'更简单 |

### 场景推荐配置

#### 1. 保守配置（低误报）
```python
threshold_calculator = UncertaintyAwareAdaptiveThreshold(
    window_size=15,      # 更大窗口，更稳定
    gamma=2.5,           # 更严格的统计阈值
    beta=0.3,            # 降低不确定性影响
    base_threshold=0.65,
    uncertainty_method='variance'
)
```

#### 2. 激进配置（高检测率）
```python
threshold_calculator = UncertaintyAwareAdaptiveThreshold(
    window_size=5,       # 更小窗口，更敏感
    gamma=3.5,           # 更宽松的统计阈值
    beta=0.7,            # 增加不确定性影响
    base_threshold=0.70,
    uncertainty_method='entropy'
)
```

#### 3. 平衡配置（推荐）
```python
threshold_calculator = UncertaintyAwareAdaptiveThreshold(
    window_size=10,
    gamma=3.0,
    beta=0.5,
    base_threshold=0.68,
    uncertainty_method='entropy'
)
```

---

## 与原始 AdaptiveThresholdCalculator 的对比

| 特性 | AdaptiveThresholdCalculator | UncertaintyAwareAdaptiveThreshold |
|------|----------------------------|-----------------------------------|
| **输入** | CPS分数列表 | CPS分数 + Ego特征 |
| **维度** | 仅时间维度 | 时间 + 空间维度 |
| **公式** | `T = μ + k*σ` | `T = (μ + γ*σ) * (1 + β*U_ego)` |
| **环境适应** | 否 | 是（根据场景不确定性） |
| **复杂度** | 低 | 中等 |
| **适用场景** | 稳定环境 | 复杂/动态环境 |

---

## 调试和监控

### 1. 查看实时统计

```python
stats = threshold_calculator.get_statistics()
print(f"帧数: {stats['frame_count']}")
print(f"窗口大小: {stats['window_size']}")
print(f"历史均值 μ_t: {stats['mu_t']}")
print(f"历史标准差 σ_t: {stats['sigma_t']}")
print(f"统计阈值 T_stat: {stats['T_stat']}")
print(f"Ego不确定性 U_ego: {stats['U_ego']}")
print(f"最终阈值 T_final: {stats['T_final']}")
```

### 2. 绘制阈值演化曲线

```python
import matplotlib.pyplot as plt

stats = threshold_calculator.get_statistics()

plt.figure(figsize=(12, 6))
plt.subplot(2, 1, 1)
plt.plot(stats['threshold_history'], label='T_final')
plt.axhline(y=threshold_calculator.base_threshold, color='r', linestyle='--', label='Base Threshold')
plt.ylabel('Threshold')
plt.legend()
plt.title('Threshold Evolution')

plt.subplot(2, 1, 2)
plt.plot(stats['uncertainty_history'], label='U_ego', color='orange')
plt.ylabel('Uncertainty')
plt.xlabel('Frame')
plt.legend()
plt.tight_layout()
plt.savefig('ua_at_evolution.png')
```

---

## 常见问题

### Q1: 如何判断应该使用哪种不确定性方法？

**A**: 
- **Entropy**: 更适合特征分布复杂的场景（如多模态特征）
- **Variance**: 更简单直接，计算更快，适合大多数场景

建议先用 `variance`，如果效果不理想再尝试 `entropy`。

### Q2: 阈值一直很高/很低怎么办？

**A**: 检查以下几点：
1. `base_threshold` 是否合理（应该在CPS分数的正常范围内）
2. `gamma` 是否过大/过小
3. `beta` 是否需要调整
4. 查看 `benign_window` 中的历史分数是否正常

### Q3: 能否在运行时切换阈值计算器？

**A**: 可以，但需要注意：
```python
# 保存旧计算器的统计信息
old_stats = old_calculator.get_statistics()

# 创建新计算器
new_calculator = UncertaintyAwareAdaptiveThreshold(...)

# 可选：迁移部分历史数据
new_calculator.benign_window = old_stats['window_values'][-5:]  # 保留最近5帧
```

---

## 总结

`UncertaintyAwareAdaptiveThreshold` 是一个更智能的阈值计算器，能够：
1. ✅ 根据历史CPS分数自适应调整（时间维度）
2. ✅ 根据场景复杂度动态放宽/收紧（空间维度）
3. ✅ 在恶劣环境下避免误报
4. ✅ 在清晰环境下保持严格检测

**推荐使用场景**：
- 多样化的环境条件（晴天、雨天、夜晚混合）
- 需要平衡检测率和误报率
- 对环境鲁棒性要求高的应用

**不推荐使用场景**：
- 环境非常稳定（原始计算器足够）
- 计算资源极度受限（需要计算特征不确定性）
- 需要极简实现

