用c语言编写计算器

板书:

1、在编写控制台程序的时候一切流程都是有先后关系、并行的,而且所有函数都是由我们来调用的,比如下面的实例性代码:

printf("确定请输入y,取消输入n");

char c = getchar();

if(c=='y')

{

///

}

else if(c=='n')

{

///

}

我们可以用getchar来等待用户输入一个值。但是到了Windows编程中就不一样了,同一时刻用户即可能点击【OK】按钮,又可能点击【Cancel】按钮,又可能在文本框中输入几个字,还可能在窗口上双击几下,这样就无法同时等待用户的这些动作。为了解决这个问题,Windows引入了消息机制(也可以叫做回调机制或者事件机制)。在程序启动的时候把函数func1要响应【OK】按钮1的点击动作、函数func2要响应【Cancel】按钮的点击动作、函数func3要响应窗口的双击动作等等这些信息告诉Windows,然后当用户执行相应操作的时候Windows就会来主动调用你注册的函数,主动通知你。不再是程序调用操作系统的函数,而是操作系统反过来调用你的函数。Don't call me ,I'll call you!(也被人称为“好莱坞法则”)。

2、关于上面的这个问题要慢慢来理解,下面就来通过第一个例子来初步理解这个Don't call me ,I'll call you!

创建一个对话框程序,然后来分析代码。看Main_OnCommand方法,初探windows的消息机制。

void Main_OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)

{

switch(id)

{

case IDC_OK:

MessageBox(hwnd,"You click OK!","Test003",MB_OK);

EndDialog(hwnd, id);

break;

case IDC_CANCEL:

MessageBox(hwnd,"You click Cancel!","Test003",MB_OK);

EndDialog(hwnd, id);

break;

default:break;

}

}

按钮被按下的时候Main_OnCommand方法被调用,hwnd是对话框句柄(什么是句柄后面讲,通俗的说就是通过它能够操纵对话框),id是控件的id,后两个参数暂时不关心。

Main_OnCommand方法中根据id,也就是被点击按钮的名字来决定不同的动作,EndDialog用来关闭对话框。

3、定制自己的对话框,向世界问好

首先打开资源编辑器并且打开对话框IDD_MAIN,然后删除对话框上的两个按钮和标签,同时删除main.cpp中的两个Case语句变成:

void Main_OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)

{

switch(id)

{

default:break;

}

}

然后重新添加一个自己的按钮,在属性中修改Name属性为IDC_BTNHELLO。

控件名字的潜规则:所有控件的Name都以“IDC_”开头,然后后边跟着控件类型的简称(按钮简称BTN、文本框简称EDT等等),最后才是控件的真正的名字。修改按钮的Caption属性(也就是按钮上显示的文字)为“问好”。

在代码中怎么得到IDC_BTNHELLO呢?刚才被删掉的IDC_OK是什么东东呢?回忆配置ResEd的时候配置的“名称输出格式”和“默认输出文件名”以及“保存时自动输出”。每次保存对话框的时候ResEd都会帮我们把控件的名字输出到rsrc.inc文件中,打开工程文件夹下的rsrc.inc,内容如下:

#define IDC_BTNHELLO

1001

Dialog编辑器会自动递增id的取值。然后生成rsrc.inc,其实就是h头文件,取这些定义的时候要先include这个inc文件。可以看到rsrc.inc文件中就是这些控件名字的定义,使用的时候只要include这些文件就可以。“rsrc.inc”和头文件一样。因此首先在main.cpp中添加“#include "rsrc.inc"”

编辑Main_OnCommand方法:

void Main_OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)

{

switch(id)

{

case IDC_BTNHELLO:

MessageBox(NULL,TEXT("世界你好"),TEXT("问好"),MB_OK);

break;

default:break;

}

}

运行程序,点击“问好”按钮,咱们第一个响应按钮动作的程序就做好了。

4、自己动手写计算器1.0版

(1)、得到控件中用户输入的文本

GetDlgItemText(hwnd,IDC_EDTNAME,str,sizeof(str));

第一个参数是对话框的句柄;第二参数就是控件的id(name);第三个参数就是字符串数组的指针;

疑问:为什么不能像MessageBox一样把字符串做为返回值返回呢???

这就涉及到C语言中函数返回指针的问题了

int i=20;

char cStr[3];

itoa(i,cStr,10);

char* f1()

{

return "adsfadsfasd";

}

在函数内部返回函数内构建的指针有可能出现数据混乱.

当f1函数在执行的时候,这段内存是被占用的,一旦函数返回,那么内存就标记为可以被其他人、函数占用。

如果你把这段内存中的指针返回了,那么一旦其他地方用了这段内存,那么你引用的就是错误的数据了。

用GetWindowText、GetDlgItemText的时候为什么要传数组名呢?复习:数组名就是指针,函数只有得到指针,才能直接操作数组中的数据。

(2)c语言中字符串连接:strcat(name,"你好");

但是在编写windows程序的时候最好使用lstrcat代替strcat。

lstrcat(name,"你好");

同样代替的有:strlen→lstrlen;strcmp→lstrcmp

(3)向用户问好

TCHAR name[256];

GetDlgItemText(hwnd,IDC_EDTNAME,name,sizeof(name)/sizeof(TCHAR));

lstrcat(name,"你好");

MessageBox(NULL,name,TEXT("问好"),MB_OK);

(4)

C语言里边字符串转换为数字:atoi:ascii to int

(5)

编写windows程序的时候最好用TCHAR来代替char,可以避免中文的问题

case IDC_BTNHELLO:

TCHAR name[256];

GetDlgItemText(hwnd,IDC_EDTNAME,name,sizeof(name)/sizeof(TCHAR));

int i = atoi(name);

int j=i*2;

TCHAR result[256];

itoa(j,result,10);

MessageBox(NULL,result,TEXT("问好"),MB_OK);

(6)Get、Set:配对的,设置对话框的值用SetDlgItemText:

代码:

case IDC_BTNADD:

//取第一个文本框的字符串,然后得到整数表示

// 取第二个文本框的字符串,然后得到整数表示

//计算两个整数的和

//把和重新转换成字符串,然后Messagebox出来。

TCHAR cNumber1[256];

GetDlgItemText(hwnd,IDC_EDTNUM1,cNumber1,sizeof(cNumber1)/sizeof(TCHAR));

int n1 = atoi(cNumber1);

TCHAR cNumber2[256];

GetDlgItemText(hwnd,IDC_EDTNUM2,cNumber2,sizeof(cNumber2)/sizeof(TCHAR));

int n2 = atoi(cNumber2);

int n3 = n1+n2;

TCHAR cResult[256];

itoa(n3,cResult,10);

SetDlgItemText(hwnd,IDC_EDTRESULT, cResult);

break;

(7)代码中有UINT、TCHAR、LRESULT、HWND之类的新的数据类型,其实它们只是一些类型的别名而已,可以通过宏定义看出来。但是考虑到可移植性,尽量不要使用它们的真实类型

(8)思考:这个计算两个数的和程序有什么缺陷?没有阻止用户输入非数字

(9)作业:做一个面积计算器,用户输入半径,在另外一个文本框中显示面积。

5、得到系统中有哪些逻辑驱动器

DWORD GetLogicalDrives(VOID);

返回值的二进制位标志着存在哪些驱动器。其中,位0设为1表示驱动器A存在于系统中;位1设为1表示存在B驱动器;以次类推。

比如:

00001100:有C盘,D盘

000011100:c、d、e

1101:A(软驱)、C、D

(1)写程序中的错误排除

"dwDrives"was not declared in the scope

declare:声明;scope:范围

"dwDrives"没有被声明在这个范围内。

(2)以二进制显示GetLogicalDrives的返回值:

stdlib.h

DWORD ds = GetLogicalDrives();

char str[33];

itoa(ds,str,2);

MessageBox(NULL,str,"",MB_OK);

DWORD是什么类型?在DWORD上点击右键,选择“转到DWORD的定义”,其实DWORD是“unsigned long”。因此str定义成33位(还有最后一位的“\0”)。

(3)课后习题:判断是否存在D盘。

(4)课后习题:依次显示系统中所有的盘符。比如显示出“CDEF”。

(5)课后习题:显示系统中有多少个逻辑驱动器。

这三道课后作业都是在锻炼位运算。一定要重视,不要一位没啥意思,很多公司的笔试面试都会考查这一点,这也是继续深入研究C和其他语言的基础。

Main_OnCommand 函数的响应按钮段:

CODE:

void Main_OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)

{

TCHAR cDriveExists[]="ABCDEFGHIJKLMNOPQRSTUVWXYZ";

int nLen=0;

switch(id)

{

case IDC_BTNRESULT: //响应计算圆面积按钮

TCHAR cBanjing[256];

GetDlgItemText(hwnd,IDC_EDTRADIUS,cBanjing,sizeof(cBanjing)/sizeof(TCHAR));

double fBanjing,fZhouchang;

fBanjing = atof(cBanjing);

fZhouchang = 3.1415926*(fBanjing*fBanjing);

TCHAR cZhouchang[256];

gcvt(fZhouchang,10,cZhouchang);

//SetDlgItemText(hwnd,IDC_EDTZHOUCHANG,cZhouchang);

SetDlgItemText(hwnd,IDC_STC,cZhouchang);

break;

case IDC_BTNCALCHELP: //响应圆面积帮助按钮

MessageBox(hwnd,TEXT("\n圆形面积计算公式:\n\n圆面积(s)=圆周率π(3.14)*半径(r)的平方。\t"),TEXT("计算公式"),MB_OK|MB_ICONASTERISK);

break;

case IDC_BTNGETDRIVERS: //获取用户磁盘信息按钮

DWORD dwMydrivers;

dwMydrivers = GetLogicalDrives();

TCHAR cDrivers[256];

itoa(dwMydrivers,cDrivers,2); //转换为字符串

strrev(cDrivers); //反转字符串

TCHAR cDriverscn[256];

//MessageBox(hwnd,cDrivers,TEXT("调试"),MB_OK);

//列出所有存在盘符

int i;

for(i=0;i<strlen(cDrivers);i++)

{

if(cDrivers[i]=='1')

{

cDriverscn[nLen] = cDriveExists[i];

nLen++;

}

}

cDriverscn[nLen]='\0';

SetDlgItemText(hwnd,IDC_EDTMYDRIVERS,cDriverscn);

//判断是否存在D盘

if(cDrivers[4]='1')

{

SetDlgItemText(hwnd,IDC_EDTYESNOD,TEXT("有"));

}

else

{

SetDlgItemText(hwnd,IDC_EDTYESNOD,TEXT("无"));

}

//计算存在磁盘总数

TCHAR cSUM[256];

itoa(nLen,cSUM,10);

SetDlgItemText(hwnd,IDC_EDTDRIVERSNUM,cSUM);

break;

default:break;

}

}

#include "rsrc.inc"     rsrc.inc文件里面的内容

CODE:

#define IDC_GRPCALC             1001

#define IDC_GRP1                1014

#define IDC_EDTRADIUS           1002

#define IDC_BTNRESULT           1003

#define IDC_BTNCALCHELP         1006

#define IDC_BTNGETDRIVERS       1008

#define IDC_STC1                1005

#define IDC_EDTMYDRIVERS        1007

#define IDC_EDTYESNOD           1009

#define IDC_EDTDRIVERSNUM       1010

#define IDC_STC2                1011

#define IDC_STC3                1012

#define IDC_STC4                1013

#define IDC_STC                 1015

dialogs.dlg  文件的内容

CODE:

#define IDC_GRPCALC 1001

#define IDC_GRP1 1014

#define IDC_EDTRADIUS 1002

#define IDC_BTNRESULT 1003

#define IDC_BTNCALCHELP 1006

#define IDC_BTNGETDRIVERS 1008

#define IDC_STC1 1005

#define IDC_EDTMYDRIVERS 1007

#define IDC_EDTYESNOD 1009

#define IDC_EDTDRIVERSNUM 1010

#define IDC_STC2 1011

#define IDC_STC3 1012

#define IDC_STC4 1013

#define IDC_STC 1015

#include "dialogs.h"

#include <windows.h>

IDD_MAIN DIALOGEX 122,75,245,193

LANGUAGE 4,2

CAPTION "calc"

FONT 8,"Tahoma",0,0,0

STYLE WS_POPUP|WS_VISIBLE|WS_CAPTION|WS_SYSMENU|WS_THICKFRAME|WS_MAXIMIZEBOX

BEGIN

  CONTROL "",IDC_EDTRADIUS,"Edit",WS_CHILD|WS_VISIBLE|WS_TABSTOP,102,29,54,13,WS_EX_CLIENTEDGE

  CONTROL "计  算",IDC_BTNRESULT,"Button",WS_CHILD|WS_VISIBLE|WS_TABSTOP,162,29,42,13

  CONTROL "帮  助",IDC_BTNCALCHELP,"Button",WS_CHILD|WS_VISIBLE|WS_TABSTOP,162,46,42,13

  CONTROL "获取磁盘信息",IDC_BTNGETDRIVERS,"Button",WS_CHILD|WS_VISIBLE|WS_TABSTOP,90,158,66,13

  CONTROL "计算结果:",IDC_STC1,"Static",WS_CHILD|WS_VISIBLE|WS_GROUP,60,48,40,9

  CONTROL "",IDC_EDTMYDRIVERS,"Edit",WS_CHILD|WS_VISIBLE|WS_TABSTOP|ES_READONLY,126,118,86,13,WS_EX_CLIENTEDGE

  CONTROL "",IDC_EDTYESNOD,"Edit",WS_CHILD|WS_VISIBLE|WS_TABSTOP|ES_READONLY,126,99,86,13,WS_EX_CLIENTEDGE

  CONTROL "",IDC_EDTDRIVERSNUM,"Edit",WS_CHILD|WS_VISIBLE|WS_TABSTOP|ES_READONLY,126,136,86,13,WS_EX_CLIENTEDGE

  CONTROL "系统中是否存在有 D 盘:",IDC_STC2,"Static",WS_CHILD|WS_VISIBLE|WS_GROUP,30,101,90,9

  CONTROL "系统中存在的所有磁盘:",IDC_STC3,"Static",WS_CHILD|WS_VISIBLE|WS_GROUP,30,120,88,9

  CONTROL "系统中逻辑驱动器个数:",IDC_STC4,"Static",WS_CHILD|WS_VISIBLE|WS_GROUP,30,138,88,9

  CONTROL "请输入圆的半径 r=",IDC_TEXT,"Static",WS_CHILD|WS_VISIBLE|WS_GROUP,34,31,68,9

  CONTROL "计算圆面积",IDC_GRPCALC,"Button",WS_CHILD|WS_VISIBLE|BS_GROUPBOX,22,14,202,59

  CONTROL "本地磁盘信息",IDC_GRP1,"Button",WS_CHILD|WS_VISIBLE|BS_GROUPBOX,22,84,202,94

  CONTROL "请输入半径",IDC_STC,"Static",WS_CHILD|WS_VISIBLE,104,48,54,9

END

杨中科 at 2008-10-28 21:52:55

diamondboy的答案非常好,值得鼓励!!!

你想一下GetLogicalDriver相关的那几道作业题有没有更高效的解法呢?提示一下使用位运算。你可以参考咱们以前的C#的课的内容:

http://www.jsj321.com/forum/view ... &extra=page%3D1

位运算是语言无关的,而且C#的位运算和C一模一样,完全可以参考。还是那句话“一通百通”。

咱们的论坛已经可以上传附件了,可以把你的作业上传上来。

再次为diamondboy喝彩。

diamondboy at 2008-10-29 21:05:44

晕了呢,我就搞不懂所谓的位运算是怎么回事呢,二进制的东东看了就懵了。。  现在就去看老大给的视频去。。。

calc.rar

(2008-10-29 21:05:44, Size: 19.9 KB, Downloads: 119)

shiren1118 at 2008-10-31 21:37:09

老杨辛苦,呵呵

willweiwei at 2008-12-09 00:37:10

由itoa是英文integer to array得atoi是英文 array to integer

百度得到的

貌似不是 C语言里边字符串转换为数字:atoi:ascii to int

哈哈

鸡蛋里面挑骨头

Joshua at 2008-12-12 21:31:13

void Main_OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)

{

  int i=0;

  DWORD x,t=1;

  char sz[26],p='A';

  if(id==IDC_BTNGET)

    {

         x=GetLogicalDrives();

         while(x>0)

                         {

                                     if((t&x)==1)  { sz=p;++i; }

                                     x>>=1;++p;

                               }

             SetDlgItemText(hwnd,IDC_EDTOUT,sz);

     }

              

}

获取本机磁盘列表。

Joshua at 2008-12-12 22:10:57

更正下:在         SetDlgItemText(hwnd,IDC_EDTOUT,sz); 上还要加一句   sz='\0';

635260101 at 2009-4-04 14:15:59

先做了计算器...

caclv1.rar

(2009-04-04 14:15:59, Size: 31 KB, Downloads: 63)

makesue at 2009-4-16 01:31:51

本人位运算显示磁盘信息的代码

void Main_OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)

{

    switch(id)

    {

        case IDC_BTN1:

         DWORD d=GetLogicalDrives();

         if((d&8)>0)

          {

           TCHAR c[256];

           SetDlgItemText(hwnd,IDC_EDT1,TEXT("存在"));

          }

         else

         {

          SetDlgItemText(hwnd ,IDC_EDT1,TEXT("不存在"));

         }

        break;

        case IDC_BTN2:

   hanshu(hwnd);

        break;

        case IDC_BTN3:        

         hanshu1(hwnd);

   break;

        

        default:break;

    }

}

void hanshu(HWND hwnd)

{

   DWORD d1=GetLogicalDrives();

         int i=0;

         while (d1>0)

         {

          if((d1&1)>0)

          {

           i++;

          }

          d1=d1>>1;

       

         }

         TCHAR j[256];

         itoa(i,j,10);

         lstrcat(j,"个");

         SetDlgItemText(hwnd,IDC_EDT2,j);

}

void hanshu1(HWND hwnd)

{

         DWORD e=GetLogicalDrives();

         int k=0;

         int p=0;

         TCHAR b[]="ABCDEFGHIJKLMNOPQRSTUVWXYZ";

         TCHAR f[256];

         while (e>0)

         {

          if((e&1)==1)

           {

           

            f[p]=b[k];

            f[++p]='\0';

            SetDlgItemText(hwnd,IDC_EDT3,f);

        

                        

           }

    e=e>>1;

    k++;

         }

        

}

磁盘.rar

(2009-04-16 01:31:51, Size: 21.5 KB, Downloads: 33)

命苦 at 2009-4-17 22:30:20

我的作业终于交上来啦!不过只有计算园面积的一道题。

初次奇遇 at 2009-4-20 13:32:51

哈哈 代码越来复杂了啊

slsky at 2009-4-29 19:46:17

^不是“非”

George at 2009-5-16 22:37:07

太难了,即使能用控制台写出来也不知道在这里怎么实现。

在VC里面写,可以把字符存入数组的,可这里怎么弄都不行。

一个头3个大!!!

未命名.JPG

CalcV1.rar

(2009-05-16 22:37:07, Size: 13.1 KB, Downloads: 30)

星尘幻 at 2009-5-18 21:58:11

未命名.jpg

除法的浮点数还不会弄..显示的还是整数。还想改成点击按钮输入数字,好像有点困难,唉继续想..

angustao at 2009-5-19 03:03:44

本以为一天可以看2节,结果看到第五节就耗费了我2天时间,并且还有位运算的练习没有做。非常感谢杨老师送给我的一句话:只有蚂蚁啃骨头慢慢来。看来也是没有办法的事,这个东西急步得。重看C++/mfc再看c跳来跳去终于还是选择了C。我把作业补交下吧

功能:四则运算(整型,双精度型) 圆面积计算

BUG:0字符显示没有控制好,没有处理键盘输入。

源代码带有文字说明,呵呵~~代码没有整理不要扔砖。

counter2.0.rar

(2009-05-19 03:03:44, Size: 24.5 KB, Downloads: 21)

星尘幻 at 2009-5-19 23:43:20

参考之前同学的代码修改了一下弄好了,用的与运算不知对不对..

明天听第六课了,希望暑假前能赶上进度..我要努力,在如鹏能学到东西我很喜欢这里

jisuanqiv1.rar

(2009-05-19 23:43:20, Size: 21.2 KB, Downloads: 23)

我是老实人 at 2009-5-28 09:17:37

用函数返回值来传递动态内存这种方法虽然好用,但是常常有人把return语句用错了。这里强调不要用return语句返回指向“栈内存”的指针,因为该内存在函数结束时自动消亡,见示例7-4-4。

  char *GetString(void)

  {

 

char p[] = "hello world";

 

return p;

// 编译器将提出警告

  }

  

  void Test4(void)

  {

  char *str = NULL;

  str = GetString();

// str 的内容是垃圾

  cout<< str <<  endl;

  }

  

示例7-4-4 return语句返回指向“栈内存”的指针

用调试器逐步跟踪Test4,发现执行str = GetString语句后str不再是NULL指针,但是str的内容不是“helloworld”而是垃圾。

(你上课讲的那个指针返回值问题)

引自--林锐《高质量c/c++编程指南》

zhiping1989 at 2009-6-07 11:37:31

这边要注意if((a&8)>0)     a&8要用括号括起来 我刚开始没注意这个问题 一直出错

zhiping1989 at 2009-6-07 13:51:42

我的作业也写好了 虽然有点晚 发出来大家指教指教 我的变量名起的有点乱 大家要谅解 因为我英文水平非常烂

CODE:

switch(id)

{

case IDC_BTN2: //计算器

{

TCHAR aname[256],bname[256],cname[256];

double a_1,a_2,a_3;

GetDlgItemText(hwnd,IDC_EDT1,aname,sizeof(aname)/sizeof(int));

GetDlgItemText(hwnd,IDC_EDT2,bname,sizeof(bname)/sizeof(int));

a_1=atof(aname);

a_2=atof(bname);

a_3=a_1+a_2;

gcvt(a_3,10,cname);

SetDlgItemText(hwnd,IDC_EDT3,cname);

}

break;

case IDC_BTN6: //面积

{

TCHAR dname[256],ename[256],qname[256];

double a_4,a_5,a_9;

GetDlgItemText(hwnd,IDC_EDT4,dname,sizeof(dname)/sizeof(TCHAR));

GetDlgItemText(hwnd,IDC_EDT5,qname,sizeof(qname)/sizeof(TCHAR));

a_4=atoi(dname);

a_9=atoi(qname);

a_5=a_4*3.1415;

gcvt(a_5,10,ename);

SetDlgItemText(hwnd,IDC_EDT6,ename);

}

break;

case IDC_BTN1:  //是否存在d盘

{

int a_6;

char fname[256];

a_6=GetLogicalDrives();

itoa(a_6,fname,2) ;

if((a_6&8)>0)

{

SetDlgItemText(hwnd,IDC_EDT7,TEXT("存在D盘"));

}

else

{

SetDlgItemText(hwnd,IDC_EDT7,TEXT("不存在D盘"));

}

}

break;

case IDC_BTN3:  //响应显示所有盘符

{

int a_7,a_8=0,a_9=0;

char hname[256];

a_7=GetLogicalDrives();

while(a_7!=0)

{

if((a_7&1)==1)

{

hname[a_8]='A'+a_9;

a_8++;

}

a_7=a_7>>1;

a_9++;

}

hname[a_8]='\0';

SetDlgItemText(hwnd,IDC_EDT7,hname);

}

break;

case IDC_BTN4:  // 响应显示磁盘个数

{

int j=0;

int i=GetLogicalDrives();

while(i!=0)

{

if((i&1)==1)

{

j++;

}

i=i>>1;

}

TCHAR k[256];

itoa(j,k,10);

SetDlgItemText(hwnd,IDC_EDT7,k);

}

break;

case IDM_A:  //退出

{

EndDialog(hwnd, 0);

}

break;

case IDM_B:   //打开如鹏网站

{

ShellExecute(hwnd,TEXT("OPEN"),TEXT("www.rupeng.com"),NULL,NULL,SW_NORMAL);

}

break;

default:break;

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

猫一样的女子245

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值