#include <sms.h>
#pragma comment(lib,"sms.lib")
void SendSMS1(LPCTSTR lpszMessage, LPCTSTR lpszRecipient,LPCTSTR lpszSMSC,BOOL bSendConfirmation, BOOL bUseDefaultSMSC)
{
SMS_HANDLE smshHandle; //短信句柄
SMS_ADDRESS smsaDestination;//接收号码
TEXT_PROVIDER_SPECIFIC_DATA tpsd;
SMS_MESSAGE_ID smsmidMessageID;
// 打开发送句柄
// 协议字符串, SMS_MODE_SEND:发送模式 psmshHandle:返回的句柄 phMessageAvailableEvent:通知的事件
if(FAILED(SmsOpen(SMS_MSGTYPE_TEXT, SMS_MODE_SEND, &smshHandle, NULL)))
{
return;
}
// Create the destination address
//SMSAT_NATIONAL:国内电话 SMSAT_INTERNATIONAL:国际电话
smsaDestination.smsatAddressType = SMSAT_UNKNOWN;
_tcsncpy(smsaDestination.ptsAddress, lpszRecipient, SMS_MAX_ADDRESS_LENGTH);
// Set up provider specific data
memset(&tpsd, 0, sizeof(tpsd));
//tpsd.dwMessageOptions = PS_MESSAGE_OPTION_NONE;
tpsd.dwMessageOptions = PS_MESSAGE_OPTION_STATUSREPORT;//表示需要状态报告
tpsd.psMessageClass = PS_MESSAGE_CLASSUNSPECIFIED;
//PS_MESSAGE_CLASS0表示短信在被接收后立即显示且不存储在收件箱(称为闪信)
//PS_MESSAGE_CLASS1表示一般的情况,被接收后存储到收件箱并发送一个确认回短信中心,发送方收到一个已被接收的状态报告。
tpsd.psReplaceOption = PSRO_NONE;
//手动清空头信息
ZeroMemory(tpsd.pbHeaderData, sizeof(tpsd.pbHeaderData));
tpsd.dwHeaderDataSize = 0;
tpsd.fMessageContainsEMSHeaders = FALSE;
tpsd.dwProtocolID = SMS_MSGPROTOCOL_UNKNOWN;
// Send the message, indicating success or failure
/*
*smshHandle: SmsOpen打开的句柄
psmsaSMSCAddress: SMSC中心
*psmsaDestinationAddress:发送的目的地址
pstValidityPeriod :发送时间的有效期
*pbData: 信息的内容部分
*dwDataSize: 信息的内容部分的长度
*pbProviderSpecificData:针对运营商的附加数据
*dwProviderSpecificDataSize :附加数据的长度
smsdeDataEncoding :短信编码[SMSDE_OPTIMAL是其
推荐值]表示由短信中心决定字符的格式
dwOptions :其他选项
psmsmidMessageID: 用于得到系统回执的信息
SmsGetMessageStatus
*/
//SMSDE_GSM使用7-bit编码
//SMSDE_UCS2使用16-bit (Unicode)编码
if(SUCCEEDED(SmsSendMessage(smshHandle, NULL,
&smsaDestination, NULL, (PBYTE) lpszMessage,
_tcslen(lpszMessage) * sizeof(TCHAR), (PBYTE) &tpsd,
sizeof(TEXT_PROVIDER_SPECIFIC_DATA), SMSDE_UCS2,
SMS_OPTION_DELIVERY_NONE, &smsmidMessageID)))
{
//Tstr=L"发送完成";
//发送完成
MessageBox( NULL,L"SEND OK",L"MSG",MB_OK);
}
else
{
//发送失败
//Tstr=L"发送失败";
MessageBox(NULL,L"发送失败",L"MSG",MB_OK);
}
// clean up
VERIFY(SUCCEEDED(SmsClose(smshHandle)));
}
注意:LPCTSTR 和 PCWSTR是宽字符,所以用16-bit (Unicode)编码。网上的HelloSMS的程序不够严谨,如果照用可能造成短信中心网关把短信给吞掉的情况,程序虽然调用成功,但是就是目标号码收不到短信。函数参数中的后三个参数可以不用或设默认值都可以。
使用Windows CE的SMS API函数SmsSendMessage可以发送7-bit编码和16-bit编码的短消息(未测试8-bit编码)。单条短消息的最大字符数分别是160和70。发送短消息全部为ASCII字符,SMS_DATA_ENCODING参数选择SMSDE_GSM或SMSDE_OPTIMAL都可以。发送短消息不全为ASCII字符,SMS_DATA_ENCODING参数选择SMSDE_UCS2或SMSDE_OPTIMAL都可以。建议选择SMSDE_OPTIMAL。
发送短消息全部为ASCII字符,且字符个数超过160个时,系统会自动将短消息进行分割,但并不是按每条160进行分割,而是按每条153。发送短消息不全为ASCII字符,且字符个数超过70个时,系统会自动将短消息进行分割,但并不是按每条70进行分割,而是按每条67。
发送短消息的字符数超过160或70时,应该自行进行分割,而不应交由系统进行分割,这样可以最大限度的利用单条短消息的最大字符数分别是160和70的条件.
上面的代码是通过MAPI来实现,任意一部有SIM卡的PDA/PPC上都已运行。
下面的代码是通过串口和AT指令来发送短信,需要有Gms Modem 才可以运行。
HANDLE hSerial;
int res;
unsigned long bytes;
//COM1 COM2 COM4
hSerial = CreateFile(_T("COM1:"), GENERIC_READ|GENERIC_WRITE,FILE_SHARE_READ,NULL,OPEN_EXISTING,0,NULL);
if(hSerial == NULL)
{
///L"串口打开失败";
return;
}
///配置串口
DCB PortDCB;
PortDCB.DCBlength = sizeof(DCB);
// 默认串口参数
GetCommState(hSerial, &PortDCB);
PortDCB.BaudRate =115200 ; // baud
PortDCB.ByteSize = 8; // Number of bits/byte, 4-8
PortDCB.Parity = NOPARITY;
PortDCB.StopBits = ONESTOPBIT;
PortDCB.fBinary=TRUE;
if (! SetCommState(hSerial, &PortDCB))
{
///L"配置串口失败";
return;
}
////配置超时值
COMMTIMEOUTS CommTimeouts;
GetCommTimeouts(hSerial, &CommTimeouts);
CommTimeouts.ReadIntervalTimeout = 100;
CommTimeouts.ReadTotalTimeoutMultiplier = 100;
CommTimeouts.ReadTotalTimeoutConstant = 100;
CommTimeouts.WriteTotalTimeoutMultiplier =100;
CommTimeouts.WriteTotalTimeoutConstant = 100;
if (!SetCommTimeouts(hSerial, &CommTimeouts))
{
return;
}
//设置串口“感兴趣”的事件
SetCommMask(hSerial, EV_RXCHAR);
//设置输入输出缓冲区
SetupComm(hSerial, 1024, 1024);
//初始化缓冲区中的信息
PurgeComm(hSerial, PURGE_TXCLEAR|PURGE_RXCLEAR);
//-------------------------------------串口设置完毕----------------------------------------
UpdateData(TRUE);
char str_last[2000];
int num_Length=0;
int context_length=0;
TCHAR tmp[2];
int i=0;
char addr[100]="0891683108100005F0";//短信中心号码
char phone[100]="11000D91";
char msg[1000];
// unsigned char *msg_tmp;
unsigned char msg_tmp[1000];
char str_tmp[100]="";
char str_tmp2[50]="000800";
num_Length=m_num.GetLength();
context_length=m_context.GetLength();
m_num=L"86"+m_num;
if(num_Length%2){m_num=m_num+L"F";}
for (i=0; i <(m_num.GetLength()); ) //处理 目标电话号码 奇偶换位
{
phone[i+9] = m_num.GetAt(i);
phone[i+8] = m_num.GetAt(i+1);
i+=2;
}
WideCharToMultiByte(CP_ACP, 0, m_context, m_context.GetLength(), str_tmp, 160, NULL, NULL);
// msg_tmp=(unsigned char *)malloc(sizeof(unsigned char ));
gsmEncodeUcs2(str_tmp,msg_tmp,context_length);//UCS2编码
for(i=0;i<context_length*2;i++)//将编后的码转换为16进制以进行传输 前两位预留
{
wsprintf(tmp, L"%02X", msg_tmp[i]);
msg[2*i+2]=tmp[0];
msg[2*i+3]=tmp[1];
}
wsprintf(tmp, L"%02X", context_length*2);//前两位存储msg的长度
msg[0]=tmp[0];
msg[1]=tmp[1];
//-------------------------------addr phone msg 全部就绪----------------------------
// 最终信息为 str_last=addr+phone+000800+msg
for(i=0;i<18 ;i++)
{
str_last[i]=addr[i];
}
for(i=0;i<22 ;i++)
{
str_last[i+18]=phone[i];
}
for(i=0;i<6 ;i++)
{
str_last[i+40]=str_tmp2[i];
}
for(i=0;i<context_length*4+2 ;i++)
{
str_last[i+46]=msg[i];
}
str_last[i+46]='\32';
//--------------------------开始发送--------------------------------------------
char str1[100]="AT+CMGS=25\r";
wsprintf(tmp, L"%02d",(context_length*4+30)/2);//计算需要发送的字节
str1[8]=tmp[0];
str1[9]=tmp[1];
char str2[100]="";
res=WriteFile(hSerial,str1,100,&bytes,NULL);
res=ReadFile(hSerial,str2,100,&bytes,NULL);
char str3[100]="";
res=WriteFile(hSerial,str_last,1000,&bytes,NULL);
res=ReadFile(hSerial,str3,100,&bytes,NULL);
//----------------------------发送结束-------------------------------------------------------
// free(msg_tmp);
CloseHandle(hSerial);
}
BOOL CSendMessage2::OnInitDialog()
{
CDialog::OnInitDialog();
// TODO: Add extra initialization here
// m_num=L"15010657623";
m_num=L"13811916883";
// m_context=L"1192323 392486";
m_context=L"工作愉快!";
UpdateData(false);
return TRUE;
网上的HelloSMS的发送短信代码,有不完善的地方。用它来发短信可能会有问题,下面这段代码是我对它的一个改进,基本可以运行在实际的项目中:
////////////////////////////////////////////////////////////////////////////
// 发送短信
//lpszSMSC:短信中心号码 lpszRecipient:接收号码(目标号码)
//lpszMessage:发送短信内容
//bUseDefaultSMSC:目标地址
//bSendConfirmation:消息发送选项
////////////////////////////////////////////////////////////////////////////
void CCellGuardApp::SendSMS( LPCTSTR lpszMessage, LPCTSTR lpszRecipient,LPCTSTR lpszSMSC,BOOL bSendConfirmation, BOOL bUseDefaultSMSC)
{
SMS_HANDLE smshHandle; //短信句柄
SMS_ADDRESS smsaDestination;//接收号码
TEXT_PROVIDER_SPECIFIC_DATA tpsd;
SMS_MESSAGE_ID smsmidMessageID;
// 打开发送句柄
// 协议字符串, SMS_MODE_SEND:发送模式 psmshHandle:返回的句柄 phMessageAvailableEvent:通知的事件
if(FAILED(SmsOpen(SMS_MSGTYPE_TEXT, SMS_MODE_SEND, &smshHandle, NULL)))
{
return;
}
// Create the destination address
smsaDestination.smsatAddressType = SMSAT_UNKNOWN;
_tcsncpy(smsaDestination.ptsAddress, lpszRecipient, SMS_MAX_ADDRESS_LENGTH);
// Set up provider specific data
memset(&tpsd, 0, sizeof(tpsd));
//tpsd.dwMessageOptions = PS_MESSAGE_OPTION_NONE;
tpsd.dwMessageOptions = PS_MESSAGE_OPTION_STATUSREPORT;
tpsd.psMessageClass = PS_MESSAGE_CLASSUNSPECIFIED;
tpsd.psReplaceOption = PSRO_NONE;
//tpsd.dwHeaderDataSize = 0;
ZeroMemory(tpsd.pbHeaderData, sizeof(tpsd.pbHeaderData));
tpsd.dwHeaderDataSize = 0;
tpsd.fMessageContainsEMSHeaders = FALSE;
tpsd.dwProtocolID = SMS_MSGPROTOCOL_UNKNOWN;
// Send the message, indicating success or failure
if(SUCCEEDED(SmsSendMessage(smshHandle, NULL,
&smsaDestination, NULL, (PBYTE) lpszMessage,
_tcslen(lpszMessage) * sizeof(TCHAR), (PBYTE) &tpsd,
sizeof(TEXT_PROVIDER_SPECIFIC_DATA), SMSDE_UCS2,
SMS_OPTION_DELIVERY_NONE, &smsmidMessageID)))
{
//Tstr=L"发送完成";
//发送完成
;
}
// clean up
VERIFY(SUCCEEDED(SmsClose(smshHandle)));
}