懒码农。。。。。。

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::

 

#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)));
}
posted on 2010-03-09 17:17  阿彬  阅读(562)  评论(0编辑  收藏  举报