很多sms modem的二次开发中间件都是收费的,但最近项目需要控制终端设备用sms通信,作为一个开源阵营追随者实在不想让老板花这钱,所以就自己动手写了一个程序用来读写SMS modem,不过还真给实现了,过程中也借鉴了网友的一些成果,特别感谢他们。我只是实现了读写短信功能,当然如果还需要其他功能也是不难的,只要向SMS发送AT指令就可以了,相信都能够实现。一般的SMS modem使用COM 口通信的,也有使用USB的。发送AT指令是通过StringWriter完成,当SMS得到AT后据会做相应动作,并把完成结果包含有"OK","ERROR"的串返回到COM口,在用StreamReader取得作分析就行。

     在实现过程中遇到的第一个困难就是中英文问题,需要转换编码方式才可发送成功。第二个就是短信长度问题,一般短信中文70,英文140,如果编码方式设置不好,即使是英文SMS也认为是中文而只能发送70长度,这就需要对“AT+CSMP=x,x,x,x”调试,因为对AT不熟悉,我可是TEST了好多次才成功,结果应为使用AT+CSMP=1,173,36,07,中文使用AT+CSMP=1,173,36,08,这方面希望熟悉AT的网友能共享您的知识。

     java的端口通信就要使用的javax.comm.*包中的类,

javax.comm.SerialPort,javax.comm.CommPortIdentifier,当然还有些监听端口类我这里没有用到。使用这些类之前先做个简单的配置,如下:

1.把win32com.dll放到jre/bin 目录 如C:/Program Files/Java/jre6/bin
2.将Javax.comm.properties放到jre/lib目录 如:C:/Program Files/Java/jre6/lib

上面的文件因为这里无法上传附件,所以上传到我的资源中,连同源码都打包成java2sms.rar,可以下载。

 

=================
这样就可以尝试连接端口了
=================
    CommPortIdentifier portId;
    SerialPort serialPort;
    OutputStreamWriter out;
    InputStreamReader in;
    public boolean open(String portName){
        
        try {
            portId = CommPortIdentifier.getPortIdentifier(portName);
            
            try {
                // 打开COM口, open方法的第二个参数是超时时间,单位是毫秒。
                // 在本程序中超时时间是5秒
                serialPort = (SerialPort) portId.open("Serial_Communication", 5000);
            } catch (PortInUseException e) {
                System.out.println("PortInUseException");
                return false;
            }
            
            // 下面是得到用于和COM口通讯的输入、输出流。
            try {
                in = new InputStreamReader(serialPort.getInputStream());
                out =new OutputStreamWriter(serialPort.getOutputStream());
            } catch (IOException e) {
                System.out.println("IOException");
                return false;
            }
            
            // 下面是初始化COM口的传输参数,如传输速率:9600等。
            try {
                serialPort.setSerialPortParams(9600, SerialPort.DATABITS_8,
                        SerialPort.STOPBITS_1, SerialPort.PARITY_NONE);
            } catch (UnsupportedCommOperationException e) {
                System.out.println("UnsupportedCommOperationException");
                return false;
            }
            
        }  catch (NoSuchPortException e) {
            System.out.println("NoSuchPortException");
            return false;
        }
        return true;
    } 
=========
 向COM 写入
=========
 
    public void writeln(String s) throws Exception {
        out.write(s);
        out.write('/r');
        out.flush();
      }
============
    从COM 读出
============

    public String read() throws Exception {
        int n, i;
        char c;
        String answer = new String("");
        for (i = 0; i < 10; i++) { 
          while (in.ready()) {   
            n = in.read();  
            if (n != -1) { 
              c = (char)n;            
              answer = answer + c; 
              Thread.sleep(1);                     phone [Timing]
            } // if
            else break;                 
          } // while
          Thread.sleep(100);            
        }  // for
//        System.out.println("read port: " + answer + "/n");    
        return answer;                  
      }
===========
    发送AT指令
===========

    public synchronized String sendAT(String atcommand) throws java.rmi.RemoteException {
        String s="";

        try {
          Thread.sleep(300);   
          writeln(atcommand);   
          for(int i=0;i<20;i++){
              Thread.sleep(300);   
              s = read();                   //为了等待结果回复
              if(s.indexOf("OK", 0)>0 || s.indexOf("ERROR", 0)>0)
                  break;
          }
        } // try
        catch (Exception e) {
          System.out.println("ERROR: send AT command failed; " + "Command: " + atcommand + "; Answer: " + s + "  " + e);
        } // catch
        return s;
      }  // sendAT

自定义类:

port :如上面代码讲解,用来COM通信,很简单不再啰嗦。

Message:盛放SMS对象,包含了SMS的一些属性。

VirtualObj: sms modem对象,实现发送,接收短信功能,如果需要其他如彩信,电话本...其他功能只要扩充它就可以。这里要说明的是  1.我的短信都是已Text方式发送的,即AT+CMGF=1,没有测试AT+CMGF=0的UDF方式,这个没有测试过,因为时间紧迫。2.中英文方面在发送时就做了判断。

if(msg.getBytes().length==msg.length()){
                //english
                if(msg.length()>140)
                    return false;
                atCommand = "AT+CSMP=1,173,36,07";
            }else{
                //chinese
                if(msg.length()>70)
                    return false;
                atCommand = "AT+CSMP=1,173,36,08";
                msg = StringUtil.encodeHex(msg);
            }

至于编码方式为什么要用07,08这两个数字,我也不知道,测试出来的,希望网友能解决我这个困惑。

好了就这些吧,多说也没什么用,自己看代码最直接了。