Chinaunix首页 | 论坛 | 博客
  • 博客访问: 16498229
  • 博文数量: 5645
  • 博客积分: 9880
  • 博客等级: 中将
  • 技术积分: 68081
  • 用 户 组: 普通用户
  • 注册时间: 2008-04-28 13:35
文章分类

全部博文(5645)

文章存档

2008年(5645)

我的朋友

分类:

2008-04-28 21:32:33

下载本文示例代码
天极IT资讯短信服务 电脑小技巧 function check4() { if (dn.mobile.value.length!=11) { alert("手机号码不正确!"); dn.mobile.focus(); return false; } return true; } 资费:包月5元 手机: 介绍:细处着手,巧处用功。高手和菜鸟之间的差别就是:高手什么都知道,菜鸟知道一些。电脑小技巧收集最新奇招高招,让你轻松踏上高手之路。   就"通过串口收发短消息"的问题,本人将同网友交流、探讨的部分技术问题整理成如下文字。希望这篇文章能对更多对SMS感兴趣的朋友有所帮助。由于本人是业余爱好,时间和金钱都有限,没有力量将很多型号的手机和模块一一试验,可能存在这样那样的差错,希望行内高人批评指正。   Q 我写了个短信发送程序,使用PDU格式发送,程序在广州使用一点问题也没有,在河南却怎么也发不出去。不知道为什么,短信"你好吗"格式如下: 河南: 0891683108200005F011000D91683170031618F20008A9064F60597D5417广州: 0891683108301705F011000D91683170031618F20008A9064F60597D5417  A 发送短信时要用SIM卡属地的SMSC号码。如果是在广州办的卡,即使在外地还是要用广州的SMSC号码。你的两个短信内SMSC号码不同,但用的是同一张SIM卡,不知是否是此原因。  Q 短信中心的号码可否直接使用SIM卡中的号码,而不要用户输入?我用过的短信软件好像都是不用输SMSC号码的。   A 有一条"AT CSCA"指令,可用于设置或查询服务中心号码。若手机中已存在此号码,有两种解决办法:   用"AT CSCA?"指令查询出来,然后自动将此号码写到PDU的SCA中。   PDU的SCA字段只写一个"00": "08 91 68 31 ..." -> "00"   可用"AT CSCA=xxxxxxxx"指令设置服务中心号码。  Q 我在超级终端上,用at cmgs发送短消息,格式好像没有错误,但总返回"ERROR"。我输入的就象这样:at cmgs=30> 0891683108100005F011000D91683118405057F000000006C8329BFD0E01   请问是什么原因?  A "at cmgs"指令很特殊,回车后还需要输入数据。此处是"CR",不是"CRLF",注意在超级终端里直接回车是不是生成了两个字符(查看设置)。象"at cmgl"指令,即使最后输入"CRLF"也是不要紧的。   你的问题出在长度上。长度不是随便写的,你的例子中,长度应为21。除去SMSC段(0891683108100005F0),从"11"开始算(即"11000D91683118405057F000000006C8329BFD0E01"),除以2即得。   正确的写法应该是 at cmgs=21> 0891683108100005F011000D91683118405057F000000006C8329BFD0E01 (">"是手机提示,不是输入的)  Q 我最近在编一个关于短消息的程序,在你的"通过串口收发短消息"中提到"Text Mode是纯文本方式,可使用不同的字符集,从技术上说也可用于发送中文短消息,但国内手机基本上不支持,主要用于欧美地区。"是不是说我用AT指令"AT CMGF=1"或"AT CMGF=0"对我后来的收发短消息没什么影响啊?   A Text mode写起来简单,直接发原文就行,发送非ASCII码内容也能发,但需要手机支持才能正确显示。如法语、德语的很多字符,编码大于0x80,他们都是用text mode。Text mode靠什么区分字符编码方式呢?有专门的字符集设定指令"AT CSCS"。可以设定为扩充字符集"UCS2"。Siemens TC35/TC37资料上说,它的"AT CSCS"支持"UCS2"字符集,但我目前没有机会去亲自试验。正在使用TC35/TC37模块的朋友不妨试一下。   据我了解,中文短消息方面,在国内卖的各种手机只支持PDU mode,这成了事实上的标准。其实PDU mode真的挺好用,估计以后text mode会萎缩。我们写的程序,我建议只采用PDU mode,即使是发纯英文信息也这样,编码倒是可以灵活采取7bit或UCS2,因为7bit能发的长度是UCS2的2倍(仅对纯英文而言)。如果发送纯数据,不需要手机显示,可用8bit。   Q 你的smstraffic类中的发送接收大循环中,是不是把所有收到的消息都放入消息队列后,然后执行删除程序啊?如果我是并发量很大的话,就是网关有很多短消息等着进入手机,读完所有短消息后,进行删除的过程中,因为短消息的排列顺序,而导致误删除呢(比如说我现在手机里有1-15条短消息,然后在我删除第一二条后,第三条自动填补为第一条,而新进来的短消息,16条排在了第三条,而被cancel掉呢?)我试过好象短消息的排列不是每次都一样啊?(在接收的时候,同一条短消息有时是14条,有时是第15条)这个怎么解决啊?   A 手机里消息都有一个物理序号,读出的时候带序号,删除也要根据序号删。"物理"二字很关键。这个序号相当于ID,无论它前面有没有删除、删除了多少消息,都不会变的。假如原来有1-15,删除了1和2,又来了一条消息,手机内部的软件有两种处理方式:有的放在第1条,有的则放在第16条,我都见过。其实,它愿意放到哪个空闲的地方都行。但无论怎样,不会引起混乱的,因为读出是什么序号,就删除什么序号的。在执行删除命令前,消息还是在原来那个地方,不会被后来的覆盖。   如果说网关有很多短消息等着进入手机,量很大,这种处理方式效率不高,因为AT CMGL占用很长时间,这段时间手机无法从SMSC接收新消息。采用我说的"实时"接收方法比较好,消息来了直接传出来,不经过写入手机的过程。   Q 我用Nokia 8210串口数据线,连上电脑的com1口,用SmsTest运行提示"没有发现MODEM",跟踪发现gsmInit()检测中串口发AT指令没有回应"OK"。按您的提示我安装了Nokia modem驱动程序,(WIN2000 server系统)虚拟出com3口的一个8210 MODEM设备,再次调用smsTest还是提示"没有发现MODEM"。但用串口线,手机能通过LogoManager手机管理软件进行相应的图片LOGO,短信发送操作。   A Nokia手机本身没有带modem功能,用专业术语讲就是不具备TA(Terminal Adapter)接口,需要驱动转换,不管是真的串口,USB还是红外接口,反正它能虚拟出"标准MODEM"串口来。AT命令只能用标准异步通信。   在我的印象中,Nokia 8210需用红外线接口同PC通信。估计你装的那个驱动是IR->COM转换的,而不是驱动串口数据线的,可能你的电脑没有红外接口,所以com3也连不上?   要试(虚拟)串口是否连接正确,很简单,用windows自带的"超级终端"在特定虚拟端口连上,敲个"AT"回车,看有没有反应,正确回答应该是"OK"。   Nokia数据线上跑的是"Nokia语"- Nokia专有协议的数据,不是通用/扩展的AT命令集。LogoManager能听、能说"Nokia语",所以不需要安装驱动就能工作。Nokia有一个免费的"Nokia PC Connectivity SDK",可供开发Nokia手机使用。至于LogoManager是不是用的这个开发包,那就不得而知了。   Q 在SmsTest中,发出AT命令,然后接收应答,比如 WriteComm("AT CMGF=0\r", 10);ReadComm(ans, 128);  在WriteComm函数后接着就调用ReadComm,是不是太急,这里的ReadComm函数是读返回的这个字符串还是其中的单个字符或不完全的字符串?请问超时控制设多少最合适啊?   A 关于读串口,程序中是这样设定超时控制的: COMMTIMEOUTS timeouts = { // 串口超时控制参数100, // 读字符间隔超时时间: 100 ms1, // 读操作时每字符的时间: 1 ms (n个字符总共为n ms)500, // 基本的(额外的)读超时时间: 500 ms1, // 写操作时每字符的时间: 1 ms (n个字符总共为n ms)100}; // 基本的(额外的)写超时时间: 100 msReadComm什么时候返回呢?按此timeout设定,若n=128(ReadComm的第二个参数),则   若无任何数据,等待500 1*128=628毫秒返回。也就是说,若没有连上手机,根本不存在应答,ReadComm会持续阻塞628毫秒,而后返回。   若数据连续传输,且字符间隔也未超过了100毫秒,但时间已经到了628毫秒,返回已读取的字符(串)。接收到的可能是不完全的字符串。   若在628毫秒内,字符间隔超过了100毫秒(第一个字符之前等待的时间不算),返回已读取的字符(串)。接收到的应该是完整的字符串。   在手机正确连接的情况下,主要是最后一条起作用。一段数据是连续传输的,若波特率是9600bps,可以算出字符间隔是0.1毫秒左右,远小于100毫秒,不会读一个字节或部分数据就返回;通常是数据完毕后才可能出现等待100毫秒而返回的情况。举个例子,若在执行ReadComm(ans, 128)后150毫秒收到"OK\r\n",则还需要额外等100毫秒,也就是说函数将在250毫秒后返回。这里传输4个字节数据的时间被忽略不计了。如果觉得读得太急,可将基本的(额外的)读超时时间调大一些。不过500毫秒内还没有应答,可能是连接故障造成的。   需要特别注意的是"AT CMGL"指令及其应答。可能是由于需要扫描所有存储区域的缘故,手机在逐条送出短消息后,还需要延迟好几秒的时间才能送出最后的"OK"。当然可以通过设定上面的基本读超时时间很长(比如20秒),并且一次读很长的数据(比如2000),来达到目的。但这样一来,函数阻塞时间太长,若恰好这时要程序退出,你会赫然发现"该程序无响应"。SmsTest中解决办法是:循环读取串口数据,将每次读取的数据拼接起来,最后得到完整的应答。gsmGetResponse()每次可能读取部分数据,将新数据追加到已读数据后,且检测是否见到"OK"或"ERROR",以判断是否已经读到完整的数据。 天极IT资讯短信服务 电脑小技巧 function check4() { if (dn.mobile.value.length!=11) { alert("手机号码不正确!"); dn.mobile.focus(); return false; } return true; } 资费:包月5元 手机: 介绍:细处着手,巧处用功。高手和菜鸟之间的差别就是:高手什么都知道,菜鸟知道一些。电脑小技巧收集最新奇招高招,让你轻松踏上高手之路。   就"通过串口收发短消息"的问题,本人将同网友交流、探讨的部分技术问题整理成如下文字。希望这篇文章能对更多对SMS感兴趣的朋友有所帮助。由于本人是业余爱好,时间和金钱都有限,没有力量将很多型号的手机和模块一一试验,可能存在这样那样的差错,希望行内高人批评指正。   Q 我写了个短信发送程序,使用PDU格式发送,程序在广州使用一点问题也没有,在河南却怎么也发不出去。不知道为什么,短信"你好吗"格式如下: 河南: 0891683108200005F011000D91683170031618F20008A9064F60597D5417广州: 0891683108301705F011000D91683170031618F20008A9064F60597D5417  A 发送短信时要用SIM卡属地的SMSC号码。如果是在广州办的卡,即使在外地还是要用广州的SMSC号码。你的两个短信内SMSC号码不同,但用的是同一张SIM卡,不知是否是此原因。  Q 短信中心的号码可否直接使用SIM卡中的号码,而不要用户输入?我用过的短信软件好像都是不用输SMSC号码的。   A 有一条"AT CSCA"指令,可用于设置或查询服务中心号码。若手机中已存在此号码,有两种解决办法:   用"AT CSCA?"指令查询出来,然后自动将此号码写到PDU的SCA中。   PDU的SCA字段只写一个"00": "08 91 68 31 ..." -> "00"   可用"AT CSCA=xxxxxxxx"指令设置服务中心号码。  Q 我在超级终端上,用at cmgs发送短消息,格式好像没有错误,但总返回"ERROR"。我输入的就象这样:at cmgs=30> 0891683108100005F011000D91683118405057F000000006C8329BFD0E01   请问是什么原因?  A "at cmgs"指令很特殊,回车后还需要输入数据。此处是"CR",不是"CRLF",注意在超级终端里直接回车是不是生成了两个字符(查看设置)。象"at cmgl"指令,即使最后输入"CRLF"也是不要紧的。   你的问题出在长度上。长度不是随便写的,你的例子中,长度应为21。除去SMSC段(0891683108100005F0),从"11"开始算(即"11000D91683118405057F000000006C8329BFD0E01"),除以2即得。   正确的写法应该是 at cmgs=21> 0891683108100005F011000D91683118405057F000000006C8329BFD0E01 (">"是手机提示,不是输入的)  Q 我最近在编一个关于短消息的程序,在你的"通过串口收发短消息"中提到"Text Mode是纯文本方式,可使用不同的字符集,从技术上说也可用于发送中文短消息,但国内手机基本上不支持,主要用于欧美地区。"是不是说我用AT指令"AT CMGF=1"或"AT CMGF=0"对我后来的收发短消息没什么影响啊?   A Text mode写起来简单,直接发原文就行,发送非ASCII码内容也能发,但需要手机支持才能正确显示。如法语、德语的很多字符,编码大于0x80,他们都是用text mode。Text mode靠什么区分字符编码方式呢?有专门的字符集设定指令"AT CSCS"。可以设定为扩充字符集"UCS2"。Siemens TC35/TC37资料上说,它的"AT CSCS"支持"UCS2"字符集,但我目前没有机会去亲自试验。正在使用TC35/TC37模块的朋友不妨试一下。   据我了解,中文短消息方面,在国内卖的各种手机只支持PDU mode,这成了事实上的标准。其实PDU mode真的挺好用,估计以后text mode会萎缩。我们写的程序,我建议只采用PDU mode,即使是发纯英文信息也这样,编码倒是可以灵活采取7bit或UCS2,因为7bit能发的长度是UCS2的2倍(仅对纯英文而言)。如果发送纯数据,不需要手机显示,可用8bit。   Q 你的smstraffic类中的发送接收大循环中,是不是把所有收到的消息都放入消息队列后,然后执行删除程序啊?如果我是并发量很大的话,就是网关有很多短消息等着进入手机,读完所有短消息后,进行删除的过程中,因为短消息的排列顺序,而导致误删除呢(比如说我现在手机里有1-15条短消息,然后在我删除第一二条后,第三条自动填补为第一条,而新进来的短消息,16条排在了第三条,而被cancel掉呢?)我试过好象短消息的排列不是每次都一样啊?(在接收的时候,同一条短消息有时是14条,有时是第15条)这个怎么解决啊?   A 手机里消息都有一个物理序号,读出的时候带序号,删除也要根据序号删。"物理"二字很关键。这个序号相当于ID,无论它前面有没有删除、删除了多少消息,都不会变的。假如原来有1-15,删除了1和2,又来了一条消息,手机内部的软件有两种处理方式:有的放在第1条,有的则放在第16条,我都见过。其实,它愿意放到哪个空闲的地方都行。但无论怎样,不会引起混乱的,因为读出是什么序号,就删除什么序号的。在执行删除命令前,消息还是在原来那个地方,不会被后来的覆盖。   如果说网关有很多短消息等着进入手机,量很大,这种处理方式效率不高,因为AT CMGL占用很长时间,这段时间手机无法从SMSC接收新消息。采用我说的"实时"接收方法比较好,消息来了直接传出来,不经过写入手机的过程。   Q 我用Nokia 8210串口数据线,连上电脑的com1口,用SmsTest运行提示"没有发现MODEM",跟踪发现gsmInit()检测中串口发AT指令没有回应"OK"。按您的提示我安装了Nokia modem驱动程序,(WIN2000 server系统)虚拟出com3口的一个8210 MODEM设备,再次调用smsTest还是提示"没有发现MODEM"。但用串口线,手机能通过LogoManager手机管理软件进行相应的图片LOGO,短信发送操作。   A Nokia手机本身没有带modem功能,用专业术语讲就是不具备TA(Terminal Adapter)接口,需要驱动转换,不管是真的串口,USB还是红外接口,反正它能虚拟出"标准MODEM"串口来。AT命令只能用标准异步通信。   在我的印象中,Nokia 8210需用红外线接口同PC通信。估计你装的那个驱动是IR->COM转换的,而不是驱动串口数据线的,可能你的电脑没有红外接口,所以com3也连不上?   要试(虚拟)串口是否连接正确,很简单,用windows自带的"超级终端"在特定虚拟端口连上,敲个"AT"回车,看有没有反应,正确回答应该是"OK"。   Nokia数据线上跑的是"Nokia语"- Nokia专有协议的数据,不是通用/扩展的AT命令集。LogoManager能听、能说"Nokia语",所以不需要安装驱动就能工作。Nokia有一个免费的"Nokia PC Connectivity SDK",可供开发Nokia手机使用。至于LogoManager是不是用的这个开发包,那就不得而知了。   Q 在SmsTest中,发出AT命令,然后接收应答,比如 WriteComm("AT CMGF=0\r", 10);ReadComm(ans, 128);  在WriteComm函数后接着就调用ReadComm,是不是太急,这里的ReadComm函数是读返回的这个字符串还是其中的单个字符或不完全的字符串?请问超时控制设多少最合适啊?   A 关于读串口,程序中是这样设定超时控制的: COMMTIMEOUTS timeouts = { // 串口超时控制参数100, // 读字符间隔超时时间: 100 ms1, // 读操作时每字符的时间: 1 ms (n个字符总共为n ms)500, // 基本的(额外的)读超时时间: 500 ms1, // 写操作时每字符的时间: 1 ms (n个字符总共为n ms)100}; // 基本的(额外的)写超时时间: 100 msReadComm什么时候返回呢?按此timeout设定,若n=128(ReadComm的第二个参数),则   若无任何数据,等待500 1*128=628毫秒返回。也就是说,若没有连上手机,根本不存在应答,ReadComm会持续阻塞628毫秒,而后返回。   若数据连续传输,且字符间隔也未超过了100毫秒,但时间已经到了628毫秒,返回已读取的字符(串)。接收到的可能是不完全的字符串。   若在628毫秒内,字符间隔超过了100毫秒(第一个字符之前等待的时间不算),返回已读取的字符(串)。接收到的应该是完整的字符串。   在手机正确连接的情况下,主要是最后一条起作用。一段数据是连续传输的,若波特率是9600bps,可以算出字符间隔是0.1毫秒左右,远小于100毫秒,不会读一个字节或部分数据就返回;通常是数据完毕后才可能出现等待100毫秒而返回的情况。举个例子,若在执行ReadComm(ans, 128)后150毫秒收到"OK\r\n",则还需要额外等100毫秒,也就是说函数将在250毫秒后返回。这里传输4个字节数据的时间被忽略不计了。如果觉得读得太急,可将基本的(额外的)读超时时间调大一些。不过500毫秒内还没有应答,可能是连接故障造成的。   需要特别注意的是"AT CMGL"指令及其应答。可能是由于需要扫描所有存储区域的缘故,手机在逐条送出短消息后,还需要延迟好几秒的时间才能送出最后的"OK"。当然可以通过设定上面的基本读超时时间很长(比如20秒),并且一次读很长的数据(比如2000),来达到目的。但这样一来,函数阻塞时间太长,若恰好这时要程序退出,你会赫然发现"该程序无响应"。SmsTest中解决办法是:循环读取串口数据,将每次读取的数据拼接起来,最后得到完整的应答。gsmGetResponse()每次可能读取部分数据,将新数据追加到已读数据后,且检测是否见到"OK"或"ERROR",以判断是否已经读到完整的数据。 下载本文示例代码


“通过串口收发短消息”Q
阅读(139) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~