这段时间一直致力于嵌入式IOServer与嵌入式HMI的开发,这中间及牵扯C#与EVC通信的问题,也牵扯EVC本身开发遇到的一些问题。
1、EVC与C#数据传递
我是用EVC做DLL(MFC 扩展DLL),C#直接调用。
这是EVC DLL的接口
- DLLEXPORT long WINAPI IOMReadData(LPTSTR,LPTSTR); //读内存数据 变量名称 数据
- DLLEXPORT long WINAPI IOMWriteData(LPTSTR,LPTSTR); //写内存数据 变量名称
这是C#的接口声明
- [DllImport(@"\Storage Card\YFIOES.dll")]
- public static extern int IOMReadData(string strName, StringBuilder strData);
- [DllImport(@"\Storage Card\YFIOES.dll")]
- public static extern int IOMWriteData(string strName, string strData);
这里面遇到几个有意思的问题:对于字符串类型一直是数据传递的问题频发地带,在EVC接口中声明CString 与 CString & 接口时发现字符串能传递下去,可是返不回来,声明成LPTSTR类型,发现如果C#接口为string ,则messagebox可以显示该字符串,但是传递到textbox控件中则为空,声明为StringBuilder则就没有任何问题。
这里有一个需要注意的问题:当返回的字符串大于16个字长度时,如下声明StringBuilder strData=new StringBuilder(); 要修改为StringBuilder strData=new StringBuilder(255); 否则会报错,我就是在这个问题上调试了半天,我还以为我的DLL出问题了呢。
2、EVC与INI文件
在EVC中是不支持INI读写的API的,幸好有网络,查了一个,还挺好使。
- DWORD CYFIOMEM::GetPrivateProfileString(LPCTSTR lpAppName,LPCTSTR lpKeyName,LPCTSTR lpDefault,LPTSTR lpReturnedString,DWORD nSize,LPCTSTR lpFileName )
- {
- CFile iniFile;
- PBYTE pFileBuf;
- CString szBuf;
- DWORD dwLength;
- if( lpReturnedString == NULL )return 0;
- if( lpDefault ) {_tcscpy( lpReturnedString,lpDefault ); }
- else {_tcscpy( lpReturnedString,TEXT( "\0" )); }
- if (lpFileName == NULL ){return _tcslen( lpReturnedString );}
- if(!iniFile.Open(lpFileName, CFile::modeRead)) {return _tcslen( lpReturnedString );}
- dwLength = iniFile.GetLength();
- if (dwLength == 0) { return _tcslen( lpReturnedString );}
- pFileBuf = new BYTE[dwLength + 2];
- if (pFileBuf == NULL) { return _tcslen( lpReturnedString );}
- memset(pFileBuf, 0x0, dwLength + 2);
- iniFile.Read((void *)pFileBuf, dwLength);
- iniFile.Close();
- if (pFileBuf[0] == 0xFF && pFileBuf[1] == 0xFE) {szBuf = (LPCWSTR)(pFileBuf + 2);}
- else
- {
- PTCHAR pszWideChar = new TCHAR[dwLength + 1];
- MultiByteToWideChar(CP_ACP, 0, (LPCSTR)pFileBuf, dwLength, pszWideChar, dwLength);
- szBuf = pszWideChar;
- delete pszWideChar;
- }
- delete pFileBuf;
- while (1)
- {
- CString szTemp;
- int nPos;
- if (szBuf.IsEmpty()) return _tcslen( lpReturnedString );
- nPos = szBuf.FindOneOf(TEXT("\r\n"));
- if (nPos == -1)
- {
- szTemp = szBuf;
- szBuf.Empty();
- }
- else
- {
- szTemp = szBuf.Left(nPos);
- szBuf = szBuf.Right(szBuf.GetLength() - nPos);
- szBuf.TrimLeft(TEXT("\r\n"));
- }
- szTemp.TrimLeft(TEXT("\t "));
- szTemp.TrimRight(TEXT("\t "));
- if (szTemp.GetAt(0) == TEXT('[') && szTemp.GetAt(szTemp.GetLength() - 1) == TEXT(']'))
- {
- szTemp = szTemp.Right(szTemp.GetLength() - 1);
- szTemp = szTemp.Left(szTemp.GetLength() - 1);
- if (lpAppName == NULL)
- {
- return _tcslen( lpReturnedString );
- }
- else if (szTemp.CompareNoCase(lpAppName) == 0)
- {
- while (1)
- {
- if (szBuf.IsEmpty()) {continue; }
- nPos = szBuf.FindOneOf(TEXT("\r\n"));
- if (nPos == -1)
- {
- szTemp = szBuf;
- szBuf.Empty();
- }
- else
- {
- szTemp = szBuf.Left(nPos);
- szBuf = szBuf.Right(szBuf.GetLength() - nPos);
- szBuf.TrimLeft(TEXT("\r\n"));
- }
- nPos = szTemp.Find(TEXT("="));
- if (nPos == -1) {return _tcslen( lpReturnedString );}
- CString szTemp1;
- szTemp1 = szTemp.Left(nPos);
- szTemp1.TrimLeft(TEXT("\t "));
- szTemp1.TrimRight(TEXT("\t "));
- if (lpKeyName == NULL) {return _tcslen( lpReturnedString );}
- else if (szTemp1.CompareNoCase(lpKeyName) == 0)
- {
- szTemp1 = szTemp.Right(szTemp.GetLength() - nPos - 1);
- szTemp1.TrimLeft(TEXT("\t "));
- szTemp1.TrimRight(TEXT("\t "));
- _tcscpy( lpReturnedString,szTemp1 );
- return _tcslen( lpReturnedString );
- }
- }
- }
- }
- }
- return _tcslen( lpReturnedString );
- }
3、在EVC中宽字符是我最头疼的,建议最好多用CString,能有效回避该问题。
在开发过程中一定还会遇到各种各样的问题,有时间再一一写来... ...
本文转自yefanqiu51CTO博客,原文链接:http://blog.51cto.com/yfsoft/323792,如需转载请自行联系原作者