百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

4G短信猫发送中文短信(Python)

liuian 2025-03-01 14:38 30 浏览

4G短信猫发送中文短信(Python)

4G短信猫发送中文短信的方式可以使用 TEXT 模式或者 PDU 模式。

1.TEXT模式

在TEXT模式下发送中文短信的指令序列:

AT+CSCS="UCS2"
AT+CSMP=17,167,0,8
AT+CMGF=1
AT+CMGS="<手机号码>"
><短信内容的 UNICODE 编码>

<手机号码> - 必须是手机号码每一位的ASCII码值16进制表示的字符串,如:15148078817 是
00310035003100340038003000370038003800310037

<短信内容的 UNICODE 编码> - 直接将短信内容转换为 UNICODE 编码,可以通过百度搜索“unicode编码转换”或者直接使用站长工具:
https://tool.chinaz.com/Tools/Unicode.aspx,如“测试短信”的 Unicode 编码:16D4B8BD577ED4FE1

也可以自己写一个简单的代码即可实现,我这里用 Python 语言来实现:

def string2unicode(raw_str):
    result = ''
    for c in raw_str:
        result += '%04x' % ord(c)
    return result

通过 Python 语言来实现发送短信,需要使用 pyserial 库来实现,首先,安装:

pip install pyserial

然后,编写发送短信的代码如下:

# coding: utf-8
import serial
import settings
import time
from logger import logger


def string2unicode(raw_str):
    result = ''
    for c in raw_str:
        result += '%04x' % ord(c)
    return result


def send(mobile: str, msg: str):
    ser = serial.Serial(settings.PORTX, settings.BPS, timeout=0.5)
    try:
        if not ser.isOpen(): ser.open()

        ser.write('AT\r\n'.encode('ascii'))
        res = ser.readall()
        logger.info(res)

        ser.write('AT+CMGF=1\r\n'.encode())
        res = ser.readall()
        logger.info(res)

        ser.write('AT+CSCS="UCS2"\r\n'.encode())
        res = ser.readall()
        logger.info(res)

        ser.write('AT+CSMP=17,167,0,8\r\n'.encode())
        res = ser.readall()
        logger.info(res)

        ser.write('AT+CMGS="{0}"\r\n'.format(string2unicode(mobile)).encode())
        res = ser.readall()
        logger.info(res)

        ser.write(string2unicode(msg).encode() + b'\x1a')
        res = ser.readall()
        logger.info(res)

    except Exception as ex:
        logger.error(str(ex))

    finally:
        time.sleep(5)
        res = ser.readall()
        logger.info(res)
        if ser.isOpen(): ser.close()

对于 Ctri + Z 结束符,在写入串口时将 Unicode 编码的末尾加上 b'\x1a' 即可。

调用 send 函数发送短信:

if __name__ == '__main__':
    for i in range(0,1):
        send('15148078818', '{0}.I look for what I miss I know not what it is 寻寻觅觅 冷冷清清,【2023-01-11】'.format(i))

2.PDU模式

PDU 编码规则,我们以例子来说明:

手机号码:15148078818

短信内容:测试短信

PDU 编码:
0011000b815141088718F80008a7086D4B8BD577ED4FE1

PDU 编码长度:22

例子

说明

0011000b81

前缀

5141088718F8

手机号码

0008a7

附加码

08

短信内容的 Unicode 编码字节数,十六进制表示

6D4B8BD577ED4FE1

短信内容“测试短信”的 Unicode 编码

前缀和附加码:直接使用即可,对于一般发送国内的中文短信都可以满足,具体编码规则可网上查找;

手机号码:末尾用 f 将号码串补齐偶数位,字符两两调换位置。

Unicode 编码字节数:短信内容的 Unicode 编码后的字符数量除以2即可,例子中 Unicode 编码“6D4B8BD577ED4FE1”的长度为16,除以2得8,所以,其值为 x08。

PDU 长度:全部 PDU 编码不含最开始的 00 计算出的字符数除以 2,例子中的长度为 22。

在PDU模式下发送中文短信的指令序列:

AT+CSCS="UCS2"
AT+CMGF=0
AT+CMGS=
><短信内容的 PDU 编码>

编写发送短信的代码如下:

# coding: utf-8
import serial
import settings
import time
from logger import logger

def string2unicode(raw_str):
    result = ''
    for c in raw_str:
        result += '%04x' % ord(c)
    return result

# 手机号码编码转换
def mobile2code(mobile: str):
    phone = mobile

    phone = phone[1:] if phone.startswith('+') else phone
    phone = phone if len(phone) % 2 == 0 else phone + 'F'
    lst = []
    for i in range(0, len(phone), 2):
        a, b = phone[i + 1], phone[i]
        lst.append(a)
        lst.append(b)

    return ''.join(lst)

# PDU编码转换
def pducode(mobile: str, msg: str):
    phone = mobile2code(mobile)
    logger.info('Mobile: %s' % phone)

    message = string2unicode(msg)
    logger.info('Unicode: %s' % message)

    msg_length = '%02x' % (len(message) // 2)
    logger.info('Unicode size: %s' % msg_length)

    info = '0011000b81' + phone + '0008a7' + msg_length + message
    logger.info('Info: %s' % info)
    logger.info('Info size: %d' % ((len(info)-2) // 2))

    return info

# 发送
def send(mobile: str, msg: str):
    info = pducode(mobile, msg)

    ser = serial.Serial(settings.PORTX, settings.BPS, timeout=0.5)
    try:
        if not ser.isOpen(): ser.open()

        ser.write('AT\r\n'.encode('ascii'))
        res = ser.readall()
        logger.info(res)

        ser.write('AT+CMGF=0\r\n'.encode())
        res = ser.readall()
        logger.info(res)

        ser.write('AT+CSCS="UCS2"\r\n'.encode())
        res = ser.readall()
        logger.info(res)

        ser.write('AT+CMGS={0}\r\n'.format((len(info)-2) // 2).encode())
        res = ser.readall()
        logger.info(res)

        ser.write(info.encode() + b'\x1a')
        res = ser.readall()
        logger.info(res)

    except Exception as ex:
        logger.error(str(ex))

    finally:
        time.sleep(5)
        res = ser.readall()
        logger.info(res)
        if ser.isOpen(): ser.close()

调用 send 函数发送短信:

if __name__ == '__main__':
    for i in range(0,1):
        send('15148078818', '{0}.I look for what I miss I know not what it is 寻寻觅觅 冷冷清清,【2023-01-11】'.format(i))



相关推荐

git的撤销、删除和版本回退_git撤销删除的文件

备注:本文参考于廖雪峰的博客Git教程。依照其博客进行学习和记录,感谢其无私分享,也欢迎各位查看原文。知识点:1、gitstatus,查看git仓库的状态2、gitdiff查看git修改了的内容...

程序员开发必会之git常用命令,git配置、拉取、提交、分支管理

整理日常开发过程中经常使用的git命令!git配置SSH刚进入项目开发中,我们首先需要配置git的config、配置SSH方式拉取代码,以后就免输入账号密码了!#按顺序执行gitconfig-...

Git使用指南 | 教你轻松学会Git_git用法详解

4000字,教大家学会Git使用。一、Git基础1、Git介绍Git是目前世界上最先进的分布式版本控制系统。版本控制系统:设计师在设计的时候做了很多版本经过了数天去问设计师每个版本都改了些啥,设计师此...

深入浅出 Git_深入浅出 gRPC

git初体验使用git前需设置用户名和Email,这些信息会出现在提交记录中以标识作者。gitconfig--globaluser.name"YeHanlin"gitc...

Git不提交指定文件的方法_git不提交指定文件的方法有哪些

大家在开发项目的时候都很喜欢使用git作为代码管理工具,但是在开发项目的时候我们的本地配置文件不应该覆盖服务器中的配置文件,我们使用命令gitstatus查看待提交文件的时候需要注意不要把本地的配...

相见恨晚的 Git 命令动画演示,一看就懂

虽然Git是一个强大的工具,但是我觉得大部分人都会同意我说的:它也可以是一个……噩梦!我一直觉得,使用Git的时候把操作过程在脑海里视觉化会非常有用:当我执行某个命令的时候,分支之间是如何交互...

GitCode的一些命令_git命令大全

GitCode的一些命令配置工具对所有本地仓库的用户信息进行配置$gitconfig--globaluser.name"[name]"对你的commit操作设置关联的用户名$...

【git】 如何删除所有 tag(本地和远程)

要删除所有本地和远程的Git标签,可以按照以下步骤进行:删除本地标签首先,删除本地标签。你可以使用以下命令删除本地的所有标签:gittag-d$(gittag-l)这将列出并删除所有本地...

互联网大漏洞:每600个网站就有1个暴露了.git文件夹

对于Web开发人员来说,向外界暴露你的.git文件夹绝对是一个菜鸟级错误。因为这样会允许任何人下载你的整个源代码存储库,包括数据库密码、加密盐、Hash和第三方接口密钥API,还有你的用户名和密码。多...

git常用命令整理_git 常用

一、Git仓库完整迁移完整迁移,就是指,不仅将所有代码移植到新的仓库,而且要保留所有的commit记录1.随便找个文件夹,从原地址克隆一份裸版本库gitclone--bare旧的git地址...

项目常用GIT操作命令_git常用操作命令 简书

Git仓库更新依赖的命令:gradle--refresh-dependenciesgradleaR完全编译;./gradlewecomm:packages:telephony:larges...

【超详细】Git 所有常用命令 + 提交规范全指南(建议收藏!)

Git命令大全初始化类命令作用gitinit初始化一个本地Git仓库(当前目录会出现.git文件夹)gitclone<仓库地址>克隆远程仓库到本地,一般用来拉项目提交代...

Git 常用的alias命令大全_git -a

Git的alias(别名)功能可以将常用的复杂命令简化,大幅提升操作效率。以下是一些实用的Gitalias配置和常用示例:一、配置alias的方法通过gitconfig命令设置,分...

Git使用教程:最详细、最傻瓜、最浅显、真正手把手教

导读:因为教程详细,所以行文有些长,新手边看边操作效果出乎你的预料。GitHub虽然有些许改版,但并无大碍。一、Git是什么?Git是目前世界上最先进的分布式版本控制系统。工作原理/流程:Work...

实用干货分享(3)- Git常用操作干货分享

官方学习地址https://git-scm.com/book/zh/v2简单的代码提交流程1.gitstatus查看工作区代码相对于暂存区的差别;2.gitadd.将当前目录下修改的所有...