这个SAPI语法抓住单词名称是一个句子的中间。
<GRAMMAR LANGID="409">
<RULE NAME="SOUNDLOG" TOPLEVEL="ACTIVE">
<OPT>
<DICTATION MAX="INF"/>
</OPT>
<L>
<P>name</P>
</L>
<OPT>
<DICTATION MAX="INF"/>
</OPT>
</RULE>
</GRAMMAR>
因此,如果我说“我的名字是安德烈”,则会标识“姓名”一词。 有更好的方法吗?
答案 0 :(得分:2)
当您获得识别时,与识别相关联的SPPHRASE数据包含子规则和属性数据以及它们在识别中出现的位置。
因此,如果您有“名称”的规则或属性标记,则可以找到与“名称”关联的字词。
例如,给出你的语法
<GRAMMAR LANGID="409">
<RULE NAME="SOUNDLOG" TOPLEVEL="ACTIVE">
<OPT>
<DICTATION MAX="INF"/>
</OPT>
<L>
<P>name</P>
</L>
<OPT>
<DICTATION MAX="INF"/>
</OPT>
</RULE>
</GRAMMAR>
如果您将其更改为
<GRAMMAR LANGID="409">
<RULE NAME="SOUNDLOG" TOPLEVEL="ACTIVE">
<OPT>
<DICTATION MAX="INF"/>
</OPT>
<L PROPNAME='name'>
<P VAL='name'>name</P>
</L>
<OPT>
<DICTATION MAX="INF"/>
</OPT>
</RULE>
</GRAMMAR>
你可以找到与'name'对应的单词:
HRESULT OnRecognition(ISpRecoResult* pResult)
{
SPPHRASE *pPhrase;
HRESULT hr = pResult->GetPhrase(&pPhrase);
if (SUCCEEDED(hr))
{
const SPPHRASEPROPERTY pProp = FindProperty(pPhrase->pProperty, L"name");
if (pRule)
{
LPWSTR text(NULL);
hr = pResult->GetText(pProp->ulFirstElement, pProp->ulCountOfElements, TRUE, &text, NULL);
if (SUCCEEDED(hr))
{
// do something with text
::CoTaskMemFree(text);
}
}
}
return hr;
}
const SPPHRASEPROPERTY* FindProperty(const SPPHRASEPROPERTY* pProp, LPCWSTR what) const
{
while (pProp!=NULL)
{
if (pProp->pFirstChild != NULL)
{
const SPPHRASEPROPERTY* pFoundProp = FindRule(pProp->pFirstChild, what);
if (pFoundProp)
{
return pFoundProp;
}
}
if (pProp->pszName != NULL && wcsstr(pProp->pszName, what) != NULL)
{
return pProp;
}
pProp = pProp->pNextSibling;
}
return NULL;
}
此代码专门针对属性所涵盖的文本。但是,通常最好使用val属性来标识项而不将代码显式绑定到语法。这允许您在不更改代码的情况下调整语法(或添加等效值)。要使用这些值,只需在获取属性后使用SPPHRASEPROPERTY.pszValue或vValue字段(而不是使用ISpPhrase :: GetText)。