MSComm

 
一、关于MSCOMM控件的一些说明
   VB5.0/6. MSComm 通信控件提供了一系列标准通信命令的接口,它允许建立串口连接,可以连接到其他通信设备(如 Modem ).
还可以发送命令、进行数据交换以及监视和响应在通信过程中可能发生的各种错误和事件,从而可以用它创建全双工 、事件驱
动的、高效实用的通信程序。但在实际通信软件设计过程中, MSComm 控件并非像想像中那样完美和容易控制.特别是在中文 Wln
 95 98 下通信时更会出现问题。下面就从基础开始介绍,然后逐步讨沦 MSComm控件在编程中出现的问题以及编程技巧。
一、用MSComm控件通信
1 .串口通信基础知识
    一般悦来,计算机都有一个或多个串行端口,它们依次为 com1 Com2 、…,这些串口还提供了外部设备与 pC 进行数据传输和
皿信的通道。这些串口在 CPU 和外设之间充当解释器的角色。当字符数据从 CPU 发送给外设时,这些字符数据将被转换成串行比特
流数据;当接收数据时,比特流数据被转换为字符数据传递给 CPU ,再进一步说,在操作系统方面, Windows 用通信驱动程序
COMM.DRV )调用 API 函数发送和接收数据,当用通信控件或声明调用 API 函数时,它门由 COMM. DRV 解释并传递给设备驱动程序,
作为一个 vB 程序员,要编写通信程序.只需知道通信控件提供给 Windows 通信 AP1 函数的接口即可.换句话说,只需设定和监视通
信控件的属性和事件即可。
2 .使用 Mscomm 控件
在开始使用 MSComm 控件之前。需要先了解其属性、事件或错误
属性             描述
CommPort    设置或返回通信端口号
Settings    以字符串的形式设置或返回波特率、奇偶校验、数据位和停止位
PortOpen    设置或返回通信端口的状态。也可以打开和关闭端口
Input       返回和删除接收缓冲区中的字符
Output      将字符串写入发送缓冲区
 
CommEvent 属性为通信事件或错误返回下列值之一。在该控件的对象库中也可以找到这些常量。
常量                    描述
ComEventBreak   1001    收到了断开信号
ComEventCTSTO   1002    Clear To Send Timeout 。在发送字符时,在系统指定的事 1 件内, CTS Clear To Send )线是低电平
ComEventDSRTO   1003    Data Set Ready Timeout 。在发送字符时,在系统指定的事件内, DSR Data Set Ready )线是低电平
ComEventFrame   1004    数据帧错误。硬件检测到一个数据帧错误
ComEventOverrun 1006    端口溢出。硬件中的字符尚未读,下一个字符又到达,并且丢失
ComEventCDTO    1007    Carrier Detect Time 。在发送字符时,在系统指定的事件内, CD Carrier Detect )线是低电平。 CD
                        也称为 RLSD Receive Line Singal Detect ,接收线信号检测)
ComEventRxOver 1008    接收缓冲区溢出。在接收缓冲区中没有空间
ComEventRxParity 1009   奇偶校验错。硬件检测到奇偶校验错误 7
ComEventTxFull 1010    发送缓冲区满。在对发送字符排队时,发送缓冲区满
ComEventDCB     1011    检取端口 DCB Device Control Blick )时发生了没有预料到的错误
 
通信事件包含了下面的设置:
  常量                  描述
ComEvSend      1    发送缓冲区中的字符数比 Sthreshold 值低
ComEvReceive   2    接收到了 Rthreshold 个字符。持续产生该事件,直到使用了 Input 属性删除了接收缓冲区中的数据
ComEvCTS       3    CTS Clear To Send )线改变
ComEvDSR       4    DSR Data Set Ready )线改变。当 DSR 1 0 改变时,该事件发生
ComEvCD        5    CD Carrier Detect )线改变 ComEvRing6 检测到响铃信号。一些 URAT Universal AsynchronousReciver-
                    -Transmitters, 通用异步收发器)不支持该事件
ComEvEOF       7    收到了 EOF 字符( ASCII 字符 26
 
Error 消息( MSComm 控件)下表列出了 MSComm 控件可捕获的错误消息:
常量                               描述
ComInvalidPropertyValue    380   无效的属性值
ComSetNotSupported         383   属性只读
ComGetNotSupported         394   属性只读
ComPortOpen               8000   端口打开时该存在无效
                          8001   超时设置必须比 0 值大
ComPortInvalid            8002   无效的端口号
                          8003   属性只在运行时有效
                          8004   属性在运行时是只读的
ComPortAleadyOpen         8005   端口已经打开
                          8006   设备标识符无效或不支持
                          8007   不支持设备的波特率
                          8008   指定的字节大小无效
                          8009   缺省参数错误
                          8010   硬件不可用(被其他设备锁住)
                          8011   函数不能分配队列
ComNoOpen                 8012   设备没有打开
                          8013   设备已经打开
                          8014   不能使用通信通知
ComSetCommStateFailed     8015   不能设置通信状态
                          8016   不能设置通信事件屏蔽
ComPortNotOpen            8018   该存在只在端口打开是有效
                          8019   设备忙
ComReadError              8020   通信设备读错误
ComDCBError               8021   检取端口设备控制块时出现内部错误
 
搞清楚以上基本属性后,就可以开始编写通信许程序了。在 VB5.0/6.0 中新建一个工程文件。添加 Microsoft Comm Control 5.0
件,在简体 Form1 中加入 Command 命令按钮并取名为 CmdTest MSComm 控件取名为 MSComm1 ,加入如下程序代码。
Private Sub cmdTestClick ( )            ' 打开串口
MSComml.CommPort =2                     ' 设定 Com2
If MSComml.PortOpen = False Then
MSComm1.Settings = "9600,n,8,1"         '9600 波特率,无校验, 8 位数据位, 1 位停止位
MSComm1.PortOpen = True                 ' 打开串口
End if
MSComm1.OutBufferCount = 0              ' 清空发送缓冲区
MSComm1.InBufferCount = 0               ' 滑空接收缓冲区
 
' 发送字符数据时注意必须用回车符( vbcr )结束
MSComm1.Output="This is a qood book ! " &vbCr
 
' 泼打电话号码或发送 AT 命令
MSComm1.Output = "ATDT 05778191898 , & vbCr
 
' 发送字符数组数据时注意 ByteArray 必须事先定义赋值
Dim ByteArray as byte( )
 
' 定义动态数组
ReDim ByteArray(1)
 
' 重定义数组大小
ByteArray ( 0 ) =0
ByteArray ( 1 ) = 1
MSComm1.Output = ByteArray
End Sub
 
private Sub MScommEvent( )
Select Case MSComm1.CommEvent
Case comEvReceive
Dim Buffer As Variant
MSComm1.InputLen = 0
' 接收二进制数据
MSComm1.InputMode= ComInputModeBinary
Buffer=MSComm1.Input
' 接收字符数据
MSComm1.InputMode=comInputModeText
Buffer = MSComml.Input
Case else
End Select
End sub
( 程序 1)
 
二、中文Win 9598下的通信问题与解决方法
1 .接收的数据少于发送的数据
    如果通过 MSComm 控件一次性传送较多的二进制数据,那么,很可能收到的数据不足。例如在设置为 24oobps 传输率的情况下,
一次性可以传输 2048 个字符数据 那么在大多数情况下。一次只能收到 1200 个字符左右,这址出为新版的 MSComm32 OCX 中存在一
个影响传输二进制数据的臭虫( bug ).注意这不是特性。
    32 Windows API 函数(以下简称 API )使用了几个用 COMMTIMEOUTS 结构表示的限时变量, WriteTotalTimeOutConstant 即是其
中的一个,它被 Windows 内部设定为 5000 (即 5 秒),这个常量决定了在通信驱动程序停止传输之前花费在发送缓冲区中数据的时间
的长短, 5 秒钟意味着通信速度为 1200bps 情况下仅能发送 600 个字符, 24oobps 情况下仅能发送 1200 个左右的字符。事实上,在一个
缓冲区内一次性发送更多的数据是非常可能的。这个 bug 同样也能引发问题,甚至在高速串口门通信情况下,即使系统在使用流控
制,无论丛软件流( Xon XofI )还是硬件流( CTS RTS )。假如数据在发送缓冲区中时,流控制停止了传输,如果停止时间超过 5
秒钟.则数据就会丢失。在某些环境下, 5 秒钟可能相当短.不过也不必担心, VB 5.0/6.0 版本的 MSComm 控件有一个新增的重要的
属性称为 CommID CommID 指的是当串口被打开时,被 API 所调用的串口句柄或称标志,这也意味着能利用 API 接口函数去修改这个
常量。每次串口关闭后, Windows 会自动将之恢复为 5000 ,所以,每次打开串口后需要重斩设定以下 API 声明,其代码见下程序。
Type COMMTIMEOUTS
ReadIntervalTimeout As Long
ReadTotalTimeoutMultiplier As Long
ReadTotalTimeoutConstant As Long
WriteTotalTimeoutMultiplier As Long
WriteTotalTimeoutConstant As Long
End Type
Declare Function SetCommTimeouts Lib "Kernel32"
(BYVal hFile As Long, lpComm TimeoutsAs COMMTIMEOUTS) As Long
Declare Function GetCommTimeouts Lib "Kernel32"
(ByVal hFile As Long, lpCommTimeouts As COMMTIMEOUTS) As Long
Dim timeouts As COMMEOUTS
Dim Ret As Long
If Comm1.PortOpen = False Then
Comm1.PortOpen = True
End if
Ret=GetCommTimeouts ( Comm1.CommID , timeouts )
'Set some default timeouts
timeOuts.ReadIntervalTimeout = 1
timeouts.ReadTotalTimeoutMultiplier =1
timeouts.ReadTotalTimeoutConstant =1
timeouts.WriteTotalTimeoutMultiplier =1
timeouts.WriteTotalTimeoutConstant=
( Comm1.OutBufferSize/Val(Comm1.Settings))*10000+1000
Ret=SetCommTimeouts( Comm1.CommID , timeouts )
( 程序 2)
 
2 .如何发送大于 128 的字符数据
    在通信程序中,以单字符方式逐个发送数据时,每一个数据范围 0-255 (即十六进制的 00-FF )。在单字符版本的英文 Win95
DOS 版的 BASIC 程序中,只需要将相应的数据转换成相应的字符发送到通信端口即可。但在中文 Win95/98 下却行不通,假设在中文
Win95 98 下运行以下程序:
Dim i
For i="0" to 255
MSComm1.Output=chr(i)
Next i
 
    希望在接收端得到预期的 0-255 之间的数据,结果却是:前 129 个数据接收正确,为 0-128 ,后面 127 个数据为 126 0 和一个 255
造成这种给果的原因在于中文 Windows 使用的是双字节字符集( DBCS )系统。 DBCS 系统使用 0-128 之间的数字表示 ASCII 字符,大于
128 的数字仅作为前导字符,它只是显示是一个非拉丁语系的字符,而并不代表实际意义。上述程序在调用 CHR ()函数时用到了
DBCS 字符集,冈此产生了此类错误。那么,如何发送人于 128 的数据呢?答案是使用字符数组,将以上程序改为:
Dim cc(255) As Byte
For i = 0 To 255
cc(i) = i
Next i
MSComm1.Output = cc
Do
DoEvents
Loop Until MSComm1.OutBufferCount = 0
' 接收过程 MSComm1_OnComm()
Select Case MSComm1.CommEvent
Case comEvReceive
Dim Buffer As Variant, b1,i
MSComm1.InputMode=comInputModeBinery
MSComm1.InputLen = 0
Buffer = MSComm1.Input
For i="LBound" (Buffer) To UBound (Buffer )
Debug.Print Buffer ( i ) ;
Next i
Case . . . . .
 
    3 .如何发送 0 字符( 00H NULL
VisuaI C ++中使用串口控件发送 0 字符有些麻烦,但在 VB5.0/6.0 中只要注意以下两点即可:
1 )设置 MSComm 控件的属性 NullDiscard="False" ;。
2 )使用二进制接收,即用 MSComm1.InputMode=ComInputModeBinary 便可以解决问题;
 
    4 .如何发送递中文字符串( DBcS 字符)
VB5.0/6.0 的各种参考书上均指明 MSComm 通信控件不能发送或接收双字节字符集系统 DBCS )的二进制数据,这对于我国及亚洲一些
使用 DBCS 字符集的国家不能不说是一大人遗憾。但是我在实践中发现,用 MSComm 控件也可以发送中文字符,具体方法有以下两种:
1 )直接发送
    直接发送即把中文字符等同于英文字符。如: MSComm1.Intput= " 这是一行中文数据! " ,但这种方法发送的中文数据不能太
长,发送缓冲区和接收缓冲区的大小需设定为中文字符的两倍以上,而且发送与接收系统所处的操作系统版本最好要一致,否则会
出现接收或发送缓冲区溢出之类的错误。这种方法时用于一般要求不太高的场合。
2 )间接发送
    在发送端将汉字或字符转换为机器内码或区位码数据数组,然后将咏转换后的数据发送到串口,在接收端接收到数据后,按照
相反的顺序得到的数据转换为相应的汉字或字符,在转换过程中.要用到位运算,如取得汉字的内码后需要将高字节和低字节分开,
VB5.0/6.0 中并没有提供此类函数,以下是求整数高、低字节的函数。
 
Public Function HiByte(a As Integer )
Dim b
b= a And &HFF00
b = b / 256
If b<0 Then b = b + 256
HiByte = b
End Function
 
Public Function LowByte(a As Integ`er)
Dim b
b = a And &HFF
LowByte = b
End Function
 
5 .如何用单机进行通信测试
    通常在写好了通信程序后需要两台 PC 或一台 Pc 、一台单片机.将通信口连接后进行测试,但很多时侯因条件限制仅有单台 PC 机,
测试项目很简单,那么能否测试呢?当然可以,而且方法也很简单。对于九针的串口,找一个废弃的串口鼠标,剥外鼠标线,将连
2 3 针的线对接即可;对于 25 针的串口,找一枚曲别针(最好有塑料外套的)将它扯直,剥削去两头的塑料后在两头各弯一个圆
圈,中间对忻后直接套接在串口的 2 3 针上即可。如果但心不够安全,则可以将 5 针按地。
'''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''''
关于 mscomm 的用法,提高篇 ......[mgwmj]©
 
MSCOMM 控件是个好东西,如果您能够充分了解他,他会为您衷心的效劳。
 
    大致看了一下下午有关讨论 MSCOMM 的话题,觉得有必要说说我的心得,我一般只做硬件,没有系统的学过软件,只是业余时间
学学用用,多少掌握了一点,也在此拿出来玩玩,不知有错没有,我可是以为我已经做的很好了 ^_^
 
这是一个 VB 通用串口事件驱动接收程序。一次性接收一个数据包,数据包可以为任意字节,保证不会丢失一个数据!
Private Sub MSComm_OnComm()
    Dim S() As Byte
    Dim SS(1024) As Byte
    Static N As Long
    Static T As Variant
 
    If (MSComm.CommEvent = comEvReceive) Then
        S = MSComm.Input                      ' 只要有数据就收进来,哪怕只是一个
        If (Timer - T > 0.01) Then            ' 间隔 10MS 以上就认为是一个新的包
            text1=""                          'text1 用于搜集和显示接收 (HEX 格式 )
            N = 0
        End If
        T = Timer
        For i = 0 To UBound(S)               ' 一个数据包可能产生若干个 oncomm 事件
            Text1.Text = Text1.Text & Right("0" & Hex(S(i)) & "H", 3) + " "
            SS(N+i)=S(i)                     ' 接收数据包缓存于 SS()
            N=N+UBound(S)
        Next i
    End If
End Sub
 
1CDHolding 属性
通过查询 Carrier Detect (CD)(载波检测) 线的状态确定当前是否有传输。Carrier Detect 是从调制解调器发送到相联计算机的一个信号,指示调制解调器正在联机。该属性在设计时无效,在运行时为只读。
语法
object.CDHolding
CDHolding 属性包括下列部分:
部分
描述
object
对象表达式,其值是“应用于”列表中的对象。
 
设置值
CDHolding 属性的设置值为:
设置
描述
True
Carrier Detect 线为高电平
False
Carrier Detect 线为低电平
 
说明
注意当 Carrier Detect 线为高电平 ( CDHolding = True) 且超时时, MSComm 控件设置 CommEvent 属性为 comEventCDTO(Carrier Detect 超时错误),并产生 OnComm 事件。
注意   在主机应用程序中捕获一个丢失的传输是特别重要的,例如一个公告板,因为呼叫者可以随时挂起(放弃传输)。
Carrier Detect 也被称为 Receive Line Signal Detect (RLSD)。
数据类型
Boolean
2CommEvent 属性
返回最近的通讯事件或错误。该属性在设计时无效,在运行时为只读。
语法
object.CommEvent
CommEvent 属性语法包括下列部分:
部分
描述
object
对象表达式,其值是“应用于”列表中的对象。
 
说明
只要有通讯错误或事件发生时都会产生 OnComm 事件, CommEvent 属性存有该错误或事件的数值代码。要确定引发 OnComm 事件的确切的错误或事件,请参阅 CommEvent 属性。
CommEvent 属性返回下列值之一来表示不同的通讯错误或事件。这些常数可以在该控件的 对象库中找到。通讯错误包括下列设置值:
常数
描述
comEventBreak
1001
接收到一个中断信号。
comEventCTSTO
1002
Clear To Send 超时。在系统规定时间内传输一个字符时,Clear To Send 线为低电平。
comEventDSRTO
1003
Data Set Ready 超时。在系统规定时间内传输一个字符时,Data Set Ready 线为低电平。
comEventFrame
1004
帧错误。硬件检测到一帧错误。
comEventOverrun
1006
端口超速。没有在下一个字符到达之前从硬件读取字符,该字符丢失。
comEventCDTO
1007
载波检测超时。在系统规定时间内传输一个字符时,Carrier Detect 线为低电平。Carrier Detect 也称为 Receive Line Signal Detect (RLSD)。
comEventRxOver
1008
接受缓冲区溢出。接收缓冲区没有空间。
comEventRxParity
1009
奇偶校验。硬件检测到奇偶校验错误
comEventTxFull
1010
传输缓冲区已满。传输字符时传输缓冲区已满
comEventDCB
1011
检索端口的设备控制块 (DCB) 时的意外错误
 
通讯事件包括下列设置值:
常数
描述
comEvSend
1
在传输缓冲区中有比 Sthreshold 数少的字符。
comEvReceive
2
收到 Rthreshold 个字符。该事件将持续产生直到用 Input 属性从接收缓冲区中删除数据。
comEvCTS
3
Clear To Send 线的状态发生变化。
comEvDSR
4
Data Set Ready 线的状态发生变化。该事件只在 DST 从 1 变到 0 时才发生。
comEvCD
5
Carrier Detect 线的状态发生变化。
comEvRing
6
检测到振铃信号。一些 UART(通用异步接收— 传输)可能不支持该事件。
comEvEOF
7
收到文件结束(ASCII 字符为 26)字符。
 
数据类型
Integer
3CommPort 属性
设置并返回通讯端口号。
语法
object.CommPort[ = value ]
CommPort 属性语法包括下列部分:
部分
描述
object
对象表达式,其值是“应用于”列表中的对象。
value
一整型值,说明端口号。
 
说明
在设计时, value 可以设置成从 1 到 16 的任何数(缺省值为 1)。但是如果用 PortOpen 属性打开一个并不存在的端口时,MSComm 控件会产生错误68(设备无效)。
警告   必须在打开端口之前设置 CommPort 属性。
数据类型
Integer
4Handshaking 属性
设置并返回硬件握手协议。
语法
object.Handshaking [ = value ]
Handshaking 属性的语法包括下列部分:
部分
描述
object
对象表达式,其值是“应用于”列表中的对象。
value
整型表达式,说明握手约定,如“设置值”中所描述。
 
设置值
value 设置值是:
设置值
描述
comNone
0
(缺省)没有握手。
comXOnXOff
1
(XON/XOFF) 握手。
comRTS
2
RTS/CTS (Request To Send/Clear To Send) 握手。
comRTSXOnXOff
3
Request To Send 和 XON/XOFF 握手皆可。
 
说明
Handshaking 是指内部通讯协议,通过该协议,数据从硬件端口传输到接收缓冲区。当一个数据字符到达串行端口,通讯设备就把它移到接收缓冲区以使程序可以读它。如果没有接受缓冲区,程序需要直接从硬件读取每一个字符,这很可能会造成数据丢失,因为字符到达的速度可以非常快。
握手协议保证在缓冲区过载时数据不会丢失,缓冲区过载为数据到达端口太快而使通讯设备来不及将它移到接收缓冲区。
数据类型
Integer
//串行口初始化
 if(m_ctrlComm.GetPortOpen())   //如果串口已经打开,则关闭。
  m_ctrlComm.SetPortOpen(FALSE);
 m_ctrlComm.SetCommPort(1);    //选择com1
 if( !m_ctrlComm.GetPortOpen())
  m_ctrlComm.SetPortOpen(TRUE);  //打开串口
 else
  AfxMessageBox("不能打开串行口!");
 m_ctrlComm.SetSettings("9600,n,8,1"); //波特率9600,无校验,8个数据位,1个停止位
 m_ctrlComm.SetInputMode(1);    //1:表示以二进制方式检取数据
 m_ctrlComm.SetRThreshold(1);    //参数1表示每当串口接收缓冲区中有多于或等于1个字符时将引发一个接收数据的OnComm事件
 m_ctrlComm.SetInputLen(0);    //设置当前接收区数据长度为0
 m_ctrlComm.GetInput();     //先预读缓冲区以清除残留数据 
 
……………..
//此段设置其他对象在串口打开情况下的状态
if(m_ctrlComm.GetPortOpen())
{
   GetDlgItem(IDC_OPEN_CLOSE)->SetWindowText(“ 关闭串口 ”)
}
return TRUE;
 
四、SetOutput()发送字节型数组。
我做出来了 , 不过还是十分十分感谢大家  
 
我的程序如下 :  
  void   CCommPortDlg::OnSelchangeComboDay()    
  {  
  BYTE   bytSend[7];//
发送字符串  
  int   i,temp;  
  CByteArray   arraySend;  
   
   
  //   TODO:   Add   your   control   notification   handler   code   here  
          //
这里就是说确定选择的天数 ,0 表示当天 ,1 表示前一天 , 等等  
  Day="m"_SelDay.GetCurSel();//
选择哪一天  
   
  //------
构造发送字符串 --------  
  bytSend[0]=0xEB;  
  bytSend[1]=0x90;  
  bytSend[2]=0x80+Day;  
  bytSend[3]=0x01;  
  bytSend[4]=0x00;  
  bytSend[5]=0x07;  
   
  for(i=0,temp=0;i<6;i++)  
  temp="temp"+bytSend[i];  
  temp="temp"%256;  
  bytSend[6]=temp;  
   
  //----------------------------  
   
  //-----
转化为 CByteArray 格式 ---  
  arraySend.RemoveAll();  
  arraySend.SetSize(7);  
  for(i=0;i<7;i++)  
  {  
  arraySend.SetAt(i,bytSend[i]);  
  // temp="arraySend".GetAt(i);     //
通过这里的测试说明这里的数据确实是存在 arraySend 里面了  
  }  
          //----------------------------  
  OpenPort();//
打开串口  
   
  wan="0";  
  total="0";  
          totalReceiveLen="0";  
  intReceiveLen="0";  
  LENGTH="0";  
  Door1=false;  
  Door="false";  
  Success="false";  
   
  m_Com.SetOutput(COleVariant(arraySend));  
  // m_Com.SetOutput(bytSend);  
   
  }  
   
 
程序已经调通 , 可以串口查询的了 .  
 
但是还是十分十分谢谢大家
 
VARIANT Var_Data; // 结果
SAFEARRAY * psa;
SAFEARRAYBOUND rgsabound;

Var_Data.vt = VT_UI1 | VT_ARRAY;
rgsabound.lLbound = 0;
rgsabound.cElements = len;
if (psa = SafeArrayCreate(VT_UI1, 1, &rgsabound))
{
memcpy(psa->pvData, Data, len);
Var_Data.parray = psa;
}
else
VariantInit(&Var_Data);

//
此时如果 Var_Data 不为空的话就发送 Var_Data 即可,
//
发送完了释放内存 SafeArrayDestroy(Var_Data.parray);

这是段程序用在我公司所有的有关串口发数的程序上,是我独立写的,决没有抄别人的,没有使用 MFC 类,使用 API ,目的是提高效率。这段程序可以放心使用。
 
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值