1

当您拥有标准 fst 时,您可以将其加载为 MutableFst 并执行以下操作:

MutableArcIterator<StdMutableFst> aiter(graph, state_id);
aiter.Seek(position)

auto arc = aiter.Value();
arc.weight = fst::TropicalWeight(NEW_WEIGHT);

aiter.SetValue(arc);

这将改变内存中弧的值。

但是,当我有 ConstFst 时,我从 GrammarFst.instances_ 向量中获得了类似的东西:

GrammarFst &fst = const_cast<GrammarFst&>(fst_in);
int32 instance_id = s >> 32;
BaseStateId base_state = static_cast<int32>(s);
const GrammarFst::FstInstance &instance = fst.instances_[instance_id];
const ConstFst<StdArc> *base_fst = instance.fst; // const fst::ConstFst<fst::ArcTpl<fst::TropicalWeightTpl<float> > >* const

我无法base_fst使用 MutableArcIterator 进行索引,因为它是 ConstFst。如果不对 Kaldi/Openfst 进行重大修改,这是否可能?

当我尝试这样做时:

MutableArcIterator<StdConstFst > aiter(base_fst, base_state);

我得到:

test.cpp:91:72: error: invalid conversion from ‘const fst::ConstFst<fst::ArcTpl<fst::TropicalWeightTpl<float> > >*’ to ‘fst::ConstFst<fst::ArcTpl<fst::TropicalWeightTpl<float> > >*’ [-fpermissive]
             MutableArcIterator<StdConstFst > aiter(base_fst, base_state);

我的问题:

我需要instances_在运行时访问 GrammarFst 对象并更改弧权重。我目前不能这样做,因为 GrammarFst.instance_ 是 ConstFsts。

可行的解决方案:

  • 在运行时将 GrammarFst.instances_ 转换为 StdFst
  • 从磁盘加载时将 GrammarFst.instances_ 转换为 StdFst
  • 保存 GrammarFst 时将 GrammarFst.instances_ 保存为 StdFst

不起作用的解决方案:

  • 将整个 GrammarFst 保存为 StdFst,(因此取消引用 instances_ 并将它们复制到它们出现的主 fst 中的每个位置。这大大增加了文件大小(~10X)并且对我不起作用
4

1 回答 1

1

在这里结束循环:GrammarFsts 只能由 ConstFsts 组成。我最终向 kaldi 做了一个 PR 来改变它,这样你就可以用 VectorFsts 组合它们,这反过来又让你可以使用标准的 MutableArcIterator 修改它们。https://github.com/kaldi-asr/kaldi/pull/4067

于 2020-12-08T20:11:11.793 回答