# SSIM功能使用指南

## 概述

CPS防御系统现在支持两种特征一致性计算方法：
1. **梯度一致性方法（Gradient Consistency）** - 原有方法，通过计算梯度的余弦相似度来评估特征一致性
2. **SSIM方法（Structural Similarity Index）** - 新增方法，通过结构相似性指数来评估特征一致性

## 使用方法

在调用`cps_defense`函数时，通过`use_ssim`参数来选择使用哪种方法：

### 使用梯度一致性方法（默认）

```python
pred_box, pred_score, gt_box, cps_score, defense_info = cps_defense(
    batch_data=batch_data,
    model=model,
    dataset=dataset,
    perturbation=perturbation,
    compute_gradients=True,  # 开启一致性计算
    use_ssim=False,          # 使用梯度一致性（默认值）
    # ... 其他参数
)
```

### 使用SSIM方法

```python
pred_box, pred_score, gt_box, cps_score, defense_info = cps_defense(
    batch_data=batch_data,
    model=model,
    dataset=dataset,
    perturbation=perturbation,
    compute_gradients=True,  # 开启一致性计算
    use_ssim=True,           # 【使用SSIM方法】
    # ... 其他参数
)
```

## 参数说明

- `compute_gradients`: `bool`
  - `True`: 开启特征一致性计算（梯度或SSIM）
  - `False`: 关闭特征一致性计算（此时`use_ssim`参数无效）

- `use_ssim`: `bool`
  - `True`: 使用SSIM方法计算特征一致性
  - `False`: 使用梯度一致性方法（默认）
  - 注意：此参数仅在`compute_gradients=True`时生效

## 日志输出识别

系统会在日志中明确标识使用的方法：

### 梯度一致性方法日志示例：
```
[MDAG] 【使用梯度一致性方法】进行特征一致性计算
[CPS] 使用梯度一致性方法
[MDAG Final] 使用梯度一致性方法
```

### SSIM方法日志示例：
```
[MDAG] 【使用SSIM方法】进行特征一致性计算
[CPS] 使用SSIM方法计算特征一致性
[MDAG Final] 使用SSIM方法计算特征一致性
```

## 方法对比

### 梯度一致性方法（Gradient Consistency）
- **优点**：
  - 考虑了模型的因果关系
  - 能够捕捉特征对模型输出的影响
  - 理论基础扎实（CPS论文原方法）

- **缺点**：
  - 计算成本较高（需要两次前向传播和反向传播）
  - 对GPU内存要求较高

### SSIM方法（Structural Similarity Index）
- **优点**：
  - 计算速度快（不需要反向传播）
  - GPU内存占用小
  - 专注于结构相似性，对噪声不敏感

- **缺点**：
  - 不考虑特征对模型输出的因果影响
  - 可能在某些情况下不如梯度方法精确

## 性能对比

| 方法 | 计算时间 | GPU内存 | 精度 |
|-----|---------|---------|------|
| 梯度一致性 | 较慢 | 较高 | 高 |
| SSIM | 快 | 低 | 中-高 |

## 推荐使用场景

1. **使用梯度一致性**：
   - GPU资源充足
   - 对防御精度要求极高
   - 需要考虑特征的因果影响

2. **使用SSIM**：
   - GPU资源有限
   - 需要快速推理
   - 对实时性要求高

## 代码示例

### 在训练/推理脚本中使用

```python
# 示例1：使用梯度一致性（原方法）
results = cps_defense(
    batch_data=batch_data,
    model=fusion_net,
    dataset=opencood_dataset,
    perturbation=adv_perturbation,
    attacker_idx=1,
    sampling_budget=10,
    lambda1=1.0,
    lambda2=1.0,  # 梯度一致性权重
    lambda3=1.0,
    tau=0.68,
    compute_gradients=True,
    use_ssim=False,  # 梯度一致性
    use_mdag=True
)

# 示例2：使用SSIM（新方法）
results = cps_defense(
    batch_data=batch_data,
    model=fusion_net,
    dataset=opencood_dataset,
    perturbation=adv_perturbation,
    attacker_idx=1,
    sampling_budget=10,
    lambda1=1.0,
    lambda2=1.0,  # SSIM权重
    lambda3=1.0,
    tau=0.68,
    compute_gradients=True,
    use_ssim=True,  # SSIM
    use_mdag=True
)
```

## 技术细节

### SSIM计算公式

SSIM在特征图上的计算公式：

```
SSIM(x, y) = (2μ_x μ_y + C1)(2σ_xy + C2) / ((μ_x² + μ_y² + C1)(σ_x² + σ_y² + C2))
```

其中：
- `μ_x`, `μ_y`: 特征图的均值
- `σ_x`, `σ_y`: 特征图的方差
- `σ_xy`: 特征图的协方差
- `C1`, `C2`: 稳定性常数

### 返回值范围

- 梯度一致性：返回值范围 `[-1, 1]`，值越大表示越一致
- SSIM：返回值范围 `[-1, 1]`，值越大表示结构越相似

两种方法的返回值范围一致，因此可以直接替换使用，不需要修改CPS计算公式。

## 故障排查

如果遇到问题，请检查：

1. 确认`compute_gradients=True`
2. 检查日志中是否有明确的方法标识
3. 如果SSIM计算失败，会自动fallback到0.0，不会中断程序
4. 查看是否有内存不足的错误（梯度方法内存占用较大）

## 相关文件

- 主实现文件：`defense/cps_defense.py`
  - `compute_gradient_consistency()` - 梯度一致性方法
  - `compute_ssim()` - SSIM方法
  - `compute_group_cps()` - CPS分数计算
  - `cps_defense()` - 主防御函数

