| # 循环评测 | |
| ## 背景 | |
| 对于选择题而言,当 LLM 给出正确的选项,并不一定代表着它能真正地理解题意并经过推理得出答案,它也有可能是蒙对的。为了将这两种情形区分开,同时也为了降低 LLM 对选项的偏见,我们可以尝试使用循环评测 (CircularEval)。我们会将一道选择题按照打乱选项的方式进行增广,若 LLM 可以在增广后的每道题上均得到正确的答案,那么我们认为在循环评测的意义下,这道题被做对了。 | |
| ## 新增自己的循环评测数据集 | |
| 一般来说,为了将一个数据集使用循环评测的方式进行评测,它的加载方式和评测方式是需要被重写的,OpenCompass 主库和配置文件均需要进行修改。后续我们以 C-Eval 为例进行讲解。 | |
| OpenCompass 主库: | |
| ```python | |
| from opencompass.datasets.ceval import CEvalDataset | |
| from opencompass.datasets.circular import CircularDatasetMeta | |
| class CircularCEvalDataset(CEvalDataset, metaclass=CircularDatasetMeta): | |
| # 被重载的数据集类 | |
| dataset_class = CEvalDataset | |
| # 若原 load 方法得到一 DatasetDict,其哪些 split 需要被循环评测。CEvalDataset load 得到 [dev, val, test],我们只需要对 val 和 test 进行循环评测,dev 不需要 | |
| default_circular_splits = ['val', 'test'] | |
| # 需要被打乱的 key 列表 | |
| default_option_keys = ['A', 'B', 'C', 'D'] | |
| # 若 answer_key 的内容属于是 ['A', 'B', 'C', 'D'] 之一,并表示正确答案。该字段表示打乱选项后,需要如何更新正确答案。与 default_answer_key_switch_method 二选一 | |
| default_answer_key = 'answer' | |
| # 如果 answer_key 的内容不属于 ['A', 'B', 'C', 'D'] 之一,那么可以使用函数的方式来指定打乱选项后的正确答案。与 default_answer_key 二选一 | |
| # def default_answer_key_switch_method(item, circular_pattern): | |
| # # item 是原本的数据项 | |
| # # circular_pattern 是一个 tuple,表示打乱选项后的顺序,例如 ('D', 'A', 'B', 'C') 表示原来的 A 选项变成了 D,原来的 B 选项变成了 A,以此类推 | |
| # item['answer'] = circular_pattern['ABCD'.index(item['answer'])] | |
| # return item | |
| ``` | |
| `CircularCEvalDataset` 会接受 `circular_pattern` 参数,它有两个取值: | |
| - `circular`: 表示单项循环。默认为该值。ABCD 会被扩充为 ABCD, BCDA, CDAB, DABC, 共 4 种 | |
| - `all_possible`: 表示全排列。ABCD 会被扩充为 ABCD, ABDC, ACBD, ACDB, ADBC, ADCB, BACD, ..., 共 24 种 | |
| 另外我们提供了一个 `CircularEvaluator` 用于替换 `AccEvaluator`,该 Evaluator 同样接受 `circular_pattern`,该参数应与上述保持一致。它会产出以下指标: | |
| - `acc_{origin|circular|all_possible}`: 将打乱后选项顺序后的题目视作多道单独的题目,计算准确率 | |
| - `perf_{origin|circular|all_possible}`: 按照 circular 的逻辑,若选项打乱后的题目都回答正确,才会视为这道题正确,计算准确率 | |
| - `more_{num}_{origin|circular|all_possible}`: 按照 circular 的逻辑,若选项打乱后的题目回答正确的数量大于等于 num,就会视为这道题正确,计算准确率 | |
| OpenCompass 配置文件: | |
| ```python | |
| from mmengine.config import read_base | |
| from opencompass.datasets.circular import CircularCEvalDataset | |
| with read_base(): | |
| from .datasets.ceval.ceval_gen_5f30c7 import ceval_datasets | |
| for d in ceval_datasets: | |
| # 重载 load 方法 | |
| d['type'] = CircularCEvalDataset | |
| # 为了与非循环评测版本做区分而进行改名 | |
| d['abbr'] = d['abbr'] + '-circular-4' | |
| # 重载评测方法 | |
| d['eval_cfg']['evaluator'] = {'type': CircularEvaluator} | |
| # 上述操作后的 dataset 形如下: | |
| # dict( | |
| # type=CircularCEvalDataset, | |
| # path='./data/ceval/formal_ceval', # 未改变 | |
| # name='computer_network', # 未改变 | |
| # abbr='ceval-computer_network-circular-4', | |
| # reader_cfg=dict(...), # 未改变 | |
| # infer_cfg=dict(...), # 未改变 | |
| # eval_cfg=dict(evaluator=dict(type=CircularEvaluator), ...), | |
| # ) | |
| ``` | |
| 另外评测时为了针对循环评测有更良好的结果呈现,建议考虑使用以下 summarizer | |
| ```python | |
| from mmengine.config import read_base | |
| from opencompass.summarizers import CircularSummarizer | |
| with read_base(): | |
| from ...summarizers.groups.ceval import ceval_summary_groups | |
| new_summary_groups = [] | |
| for item in ceval_summary_groups: | |
| new_summary_groups.append( | |
| { | |
| 'name': item['name'] + '-circular-4', | |
| 'subsets': [i + '-circular-4' for i in item['subsets']], | |
| } | |
| ) | |
| summarizer = dict( | |
| type=CircularSummarizer, | |
| # 选择具体看哪些指标 | |
| metric_types=['acc_origin', 'perf_circular'], | |
| dataset_abbrs = [ | |
| 'ceval-circular-4', | |
| 'ceval-humanities-circular-4', | |
| 'ceval-stem-circular-4', | |
| 'ceval-social-science-circular-4', | |
| 'ceval-other-circular-4', | |
| ], | |
| summary_groups=new_summary_groups, | |
| ) | |
| ``` | |
| 更多复杂的评测案例可以参考这个样例代码: https://github.com/open-compass/opencompass/tree/main/examples/eval_circular.py | |