【java_wxid项目】【第十五章】【Spring Cloud Skywalking集成】

主项目链接:https://gitee.com/java_wxid/java_wxid
项目架构及博文总结:

项目模块:
前期规划,实现部分

java_wxid   
├── demo                                                            // 演示模块
│     └── 模块名称:apache-mybatis-demo模块                            //Apache Mybatis集成(已实现并有博文总结)
│     └── 模块名称:apache-shardingsphere-demo模块                     //Apache ShardingSphere集成(已实现并有博文总结)
│     └── 模块名称:design-demo模块                                    //设计模式实战落地(已实现并有博文总结)
│     └── 模块名称:elasticsearch-demo模块                             //ElasticSearch集成(已实现并有博文总结)
│     └── 模块名称:mongodb-demo模块                                   //MongoDB集成(已实现并有博文总结)
│     └── 模块名称:redis-demo模块                                     //Redis集成(已实现并有博文总结)
│     └── 模块名称:spring-boot-demo模块                               //Spring Boot快速构建应用(已实现并有博文总结)
│     └── 模块名称:spring-cloud-alibaba-nacos-demo模块                //Spring Cloud Alibaba Nacos集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-alibaba-seata-demo模块                //Spring Cloud Alibaba Seata集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-alibaba-sentinel-demo模块             //Spring Cloud Alibaba Sentinel集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-gateway-demo模块                      //Spring Cloud Gateway集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-hystrix-demo模块                      //Spring Cloud Hystrix集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-open-feign-demo模块                   //Spring Cloud Open Feign集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-ribbon-demo模块                       //Spring Cloud Ribbon集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-security-oauth2-demo模块              //Spring Cloud Security Oauth2集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-security-oauth2-sso-client-demo模块   //Spring Cloud Security Oauth2集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-skywalking-demo模块                   //Spring Cloud Skywalking集成(已实现并有博文总结)
│     └── 模块名称:spring-cloud-stream-demo模块                       //Spring Cloud Stream集成(已实现并有博文总结)
│     └── 模块名称:swagger-demo模块                                   //springfox-swagger2集成(已实现并有博文总结)
│     └── 模块名称:xxl-job模块                                        //xxl-job集成(已实现并有博文总结)
│     └── 模块名称:apache-spark-demo模块                              //Apache Spark集成
│     └── 模块名称:etl-hdfs-hive-hbase-demo模块                       //ETL、HDFS、Hive、Hbase集成
│     └── 模块名称:ddd-mode-demo模块                                  //DDD领域设计
│     └── 模块名称:netty-demo模块                                     //Netty集成
│     └── 模块名称:vue-demo模块                                       //前端vue集成
├── document                                                        // 文档
│     └── JavaKnowledgeDocument                                     //java知识点
│           └── java基础知识点.md                     
│           └── mq知识点.md
│           └── mysql知识点.md
│           └── redis知识点.md
│           └── springcould知识点.md
│           └── spring知识点.md
│     └── FounderDocument                                           //创始人
│           └── 创始人.md

系列文章:快速集成各种微服务相关的技术,帮助大家可以快速集成到自己的项目中,节约开发时间。
提示:系列文章还未全部完成,后续的文章,会慢慢补充进去的。

下载包

APM为v8.9.1
Agents为v8.10.0
如下图(示例):
在这里插入图片描述百度云盘分享:
链接:https://pan.baidu.com/s/16zpZHcKqSoui3HUiUrqGnQ?pwd=2022
提取码:2022

配置

数据库使用默认的H2数据库,我使用默认配置,暂时不作修改
application.yml的storage可以选择数据库
如下图(示例):
在这里插入图片描述如果需要修改访问端口可以在这里修改
如下图(示例):
在这里插入图片描述webapp的webapp.yml修改server.port即可,我这里还是用的默认的

启动运行

运行前需要保证电脑配置了JAVA_HOME
如下图(示例):
在这里插入图片描述
双击文件直接运行
如下图(示例):
在这里插入图片描述访问http://127.0.0.1:8080
如下图(示例):
在这里插入图片描述

idea配置运行skywalking的程序配置jvm参数

如下图(示例):
在这里插入图片描述①配置VM options(skywalking-agent.jar所在位置):-javaagent:E:/skywalking/skywalking-agent/skywalking-agent.jar
注意事项:由于Skywalking 默认是不支持 Spring Cloud Gateway ,若为Cloud服务,需要将optional-plugins目录中最新的apm-spring-cloud-gateway*放入plugins目录中
②配置 Program arguments(Your_ApplicationName为当前的服务名称):-Dskywalking.agent.service_name=Your_ApplicationName
③配置指向写入链路数据的服务器地址-Dskywalking.collector.backend_service=localhost:11800

创建spring-cloud-skywalking-demo项目

项目代码:https://gitee.com/java_wxid/java_wxid/tree/master/demo/spring-cloud-skywalking-demo
项目结构如下(示例):
在这里插入图片描述

修改pom.xml

代码如下(示例):

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.3.2.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.example</groupId>
    <artifactId>spring-cloud-skywalking-demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>spring-cloud-skywalking-demo</name>
    <description>Demo project for Spring Boot</description>
    <properties>
        <java.version>1.8</java.version>
    </properties>
    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <!--        lombok插件-->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.14</version>
            <scope>provided</scope>
        </dependency>


        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
            <version>8.0.17</version>
        </dependency>

        <!-- mybatis自动配置依赖-->
        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.0.0</version>
        </dependency>

        <!-- SkyWalking 工具类 -->
        <dependency>
            <groupId>org.apache.skywalking</groupId>
            <artifactId>apm-toolkit-trace</artifactId>
            <version>8.10.0</version>
        </dependency>

        <!-- apm-toolkit-logback-1.x -->
        <dependency>
            <groupId>org.apache.skywalking</groupId>
            <artifactId>apm-toolkit-logback-1.x</artifactId>
            <version>8.10.0</version>
        </dependency>




        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>

创建logback-spring.xml

代码如下(示例):

<?xml version="1.0" encoding="UTF-8"?>
<configuration scan="true"  debug="false">

    <springProperty scop="context" name="spring.application.name" source="spring.application.name" defaultValue=""/>
    <!--日志存放路径-->
    <property name="PATH" value="logs"/>

    <property name="FILE_NAME" value="${spring.application.name}"/>

    <appender name="Console" class="ch.qos.logback.core.ConsoleAppender">
        <encoder class="ch.qos.logback.core.encoder.LayoutWrappingEncoder">
            <layout class="org.apache.skywalking.apm.toolkit.log.logback.v1.x.TraceIdPatternLogbackLayout">
                <Pattern>
                    %black(%d{ISO8601}) [%tid] %highlight(${LOG_LEVEL_PATTERN:-%5p}) [%blue(%t)] %yellow(%C{1.}): %msg%n%throwable
                </Pattern>
            </layout>
        </encoder>
    </appender>
    <!--trace-->
    <appender name="TRACE_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <file>${PATH}/${FILE_NAME}_trace.log</file>
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <FileNamePattern>${PATH}/${FILE_NAME}_trace.%d{yyyy-MM-dd}.log</FileNamePattern>
            <maxHistory>60</maxHistory>
            <totalSizeCap>1GB</totalSizeCap>
        </rollingPolicy>
        <encoder>
            <pattern>[%d{yyyy-MM-dd HH:mm:ss.SSS}] %highlight([%-5level]) %green([%15.15thread]) %cyan([%logger:%line])--%mdc{client} %msg%n</pattern>
        </encoder>
    </appender>

    <!--error-->
    <appender name="ERROR_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <file>${PATH}/${FILE_NAME}_error.log</file>
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <FileNamePattern>${PATH}/${FILE_NAME}_error.%d{yyyy-MM-dd}.log</FileNamePattern>
            <maxHistory>60</maxHistory>
            <totalSizeCap>1GB</totalSizeCap>
        </rollingPolicy>
        <encoder>
            <pattern>[%d{yyyy-MM-dd HH:mm:ss.SSS}] %highlight([%-5level]) %green([%15.15thread]) %cyan([%logger:%line])--%mdc{client} %msg%n</pattern>
        </encoder>
        <filter class="ch.qos.logback.classic.filter.ThresholdFilter">
            <level>ERROR</level>
        </filter>
    </appender>

    <root level="info">
        <appender-ref ref="Console" />
        <appender-ref ref="TRACE_FILE" />
        <appender-ref ref="ERROR_FILE" />
    </root>

</configuration>

创建application.yml

代码如下(示例):

server:
  port: 8199

spring:
  application:
    name: springboot-skywalking-demo
  datasource:
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://139.224.137.74:3306/syncdemo?serverTimezone=UTC&characterEncoding=utf8&useUnicode=true&useSSL=false
    username: root
    password: ca0a997ee4770063


修改SpringCloudSkywalkingDemoApplication

代码如下(示例):

package com.example.springcloudskywalkingdemo;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.mybatis.spring.annotation.MapperScan;

@MapperScan("com.example.springcloudskywalkingdemo.dao")
@SpringBootApplication
public class SpringCloudSkywalkingDemoApplication {

    public static void main(String[] args) {
        SpringApplication.run(SpringCloudSkywalkingDemoApplication.class, args);
    }

}

创建IndexController

代码如下(示例):

package com.example.springcloudskywalkingdemo.controller;

import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

/**
 * @author zhiwei Liao
 * @Description
 * @Date create in 2022/9/12 0012 21:03
 */
@RestController
public class IndexController {

    @RequestMapping("/")
    public String hello(){
        return "hello fox";
    }

    @RequestMapping("/notify")
    public String notify(@RequestBody Object obj){
        //TODO 告警信息,给技术负责人发短信,钉钉消息,邮件,微信通知等
        System.err.println(obj.toString());
        return "notify successfully";
    }
}

创建ProfileController

代码如下(示例):

package com.example.springcloudskywalkingdemo.controller;


import org.apache.skywalking.apm.toolkit.trace.Trace;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

/**
 * @author zhiwei Liao
 * @Description
 * @Date create in 2022/9/12 0012 21:03
 */
@RestController
public class ProfileController {

    @RequestMapping("/profile")
    public String profile(){
        process();
        return "success";
    }

    @Trace
    private void process(){
        ExecutorService executorService = Executors.newFixedThreadPool(2);
        CountDownLatch countDownLatch = new CountDownLatch(2);
        executorService.submit(new Runnable() {
            @Override
            public void run() {
                countDownLatch.countDown();
            }
        });
        executorService.submit(new Runnable() {
            @Override
            public void run() {
               // countDownLatch.countDown();
            }
        });
        try {
            countDownLatch.await(500, TimeUnit.MILLISECONDS);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

    }
}

创建UserController

代码如下(示例):

package com.example.springcloudskywalkingdemo.controller;

import com.example.springcloudskywalkingdemo.entity.User;
import com.example.springcloudskywalkingdemo.service.UserService;
import lombok.extern.slf4j.Slf4j;
import org.apache.skywalking.apm.toolkit.trace.TraceContext;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.List;
import java.util.Optional;

/**
 * @author zhiwei Liao
 * @Description
 * @Date create in 2022/9/12 0012 21:03
 */
@RestController
@RequestMapping("/user")
@Slf4j
public class UserController {

    @Autowired
    private UserService userService;

    @RequestMapping("/info/{id}")
    public User info(@PathVariable("id") Integer id){

        if(id==4){
            throw new IllegalArgumentException("参数异常");
        }

        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        return userService.getById(id);
    }

    @RequestMapping("/list")
    public List<User> list(){
        //TraceContext可以绑定key-value
        TraceContext.putCorrelation("name", "liaozhiwei");
        Optional<String> op = TraceContext.getCorrelation("name");
        log.info("name = {} ", op.get());
        //获取跟踪的traceId
        String traceId = TraceContext.traceId();
        log.info("traceId = {} ", traceId);
        return userService.list();
    }
    
    
    
}

创建UserMapper

代码如下(示例):

package com.example.springcloudskywalkingdemo.dao;

import com.example.springcloudskywalkingdemo.entity.User;
import org.apache.ibatis.annotations.Select;
import org.springframework.stereotype.Repository;
import java.util.List;
/**
 * @author zhiwei Liao
 * @Description
 * @Date create in 2022/9/12 0012 21:03
 */
@Repository
public interface UserMapper {
    @Select("select * from user")
    List<User> list();

    @Select("select * from user where id=#{id}")
    User getById(Integer id);
}

创建User

代码如下(示例):

package com.example.springcloudskywalkingdemo.entity;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

/**
 * @author zhiwei Liao
 * @Description
 * @Date create in 2022/9/12 0012 21:03
 */
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {

    private Integer id;
    private String name;


}

创建UserService

代码如下(示例):

package com.example.springcloudskywalkingdemo.service;

import com.example.springcloudskywalkingdemo.entity.User;
import java.util.List;

/**
 * @author zhiwei Liao
 * @Description
 * @Date create in 2022/9/12 0012 21:03
 */
public interface UserService {

    List<User> list();

    User getById(Integer id);
}

创建UserServiceImpl

代码如下(示例):

package com.example.springcloudskywalkingdemo.service.impl;

import com.example.springcloudskywalkingdemo.dao.UserMapper;
import com.example.springcloudskywalkingdemo.entity.User;
import com.example.springcloudskywalkingdemo.service.UserService;
import org.apache.skywalking.apm.toolkit.trace.Tag;
import org.apache.skywalking.apm.toolkit.trace.Tags;
import org.apache.skywalking.apm.toolkit.trace.Trace;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.List;

/**
 * @author zhiwei Liao
 * @Description
 * @Date create in 2022/9/12 0012 21:03
 */
@Service
public class UserServiceImpl implements UserService {
    @Autowired
    private UserMapper userMapper;

    @Trace
    @Tag(key = "list", value = "returnedObj")
    @Override
    public List<User> list(){
        return userMapper.list();
    }

    @Trace
    @Tags({@Tag(key = "param", value = "arg[0]"),
            @Tag(key = "user", value = "returnedObj")})
    @Override
    public User getById(Integer id){
        return userMapper.getById(id);
    }
}

验证Spring Cloud Skywalking是否工作

调用接口
如下图(示例):
在这里插入图片描述TID查询的追踪ID
如下图(示例):
在这里插入图片描述调用链路显示
如下图(示例):
在这里插入图片描述

.版本 2 .程序集 千寻微信框架_SDK, , , QxWeChat_SDK .全局变量 规则名称, 文本型, , "0" ' 规则名称数组 .全局变量 规则群组, 文本型, , "0" ' 对应的群ID数组 .全局变量 用户等待, 文本型, , "0" ' 存储等待回复的用户信息 .子程序 _启动子程序, 整数型 返回 (0) .子程序结束 .子程序 Qx_EventEnable, 整数型, 公开 加载规则 () 返回 (#事件处理_同意) .子程序结束 .子程序 加载规则 .局部变量 配置路径, 文本型 .局部变量 内容数组, 文本型, , "0" .局部变量 i, 整数型 配置路径 = 取运行目录() + "\data\plugin\群发规则.txt" ' 自动创建示例文件 .如果真 (文件是否存在(配置路径) = 假) 创建目录(取运行目录() + "\data\plugin\") 写文本文件(配置路径, "规则1:群ID1,群ID2" + #换行符 + "规则2:群ID3") .如果真结束 内容数组 = 读入文本文件(配置路径) 重定义数组(规则名称, 取数组成员数(内容数组)) 重定义数组(规则群组, 取数组成员数(内容数组)) .计次循环首(取数组成员数(规则名称), i) .如果真 (寻找文本(内容数组[i], ":", , 假) ≠ -1) 规则名称[i] = 取文本左边(内容数组[i], 寻找文本(内容数组[i], ":", , 假)) 规则群组[i] = 取文本右边(内容数组[i], 内容数组[i].长度 - 寻找文本(内容数组[i], ":", , 假) - 1) .否则 规则名称[i] = "规则" + 到文本(i + 1) 规则群组[i] = "" .如果真结束 .计次循环尾() ' 刷新设置窗口显示 .如果真 (是否已创建(窗口_设置)) 窗口_设置.规则列表.列表项 = 到文本(规则名称) .如果真结束 .子程序结束 .子程序 Qx_EventPrivateMsg, 整数型, 公开 .参数 robot_wxid, 文本型 .参数 jsonStr, 文本型 .局部变量 消息数据, 事件_数据_消息 .局部变量 用户索引, 整数型 .局部变量 选择序号, 整数型 消息数据 = 事件_解析消息(jsonStr) ' 只处理自己发送的消息 .如果真 (消息数据.fromWxid ≠ robot_wxid) 返回 (#消息处理_继续) .如果真结束 ' 检测触发关键词 .如果真 (消息数据.消息内容.右边(4) = "群发消息") .如果真 (取数组成员数(规则名称) = 0) Api.发送文本消息(robot_wxid, robot_wxid, "请先配置群发规则", , ) 返回 (#消息处理_结束) .如果真结束 用户索引 = 查找用户(robot_wxid) .如果真 (用户索引 = -1) 用户索引 = 加入成员(user等待, ) .如果真结束 user等待[user索引] = 消息数据.消息内容.左边(消息数据.消息内容.长度 - 4) Api.发送文本消息(robot_wxid, robot_wxid, 生成规则菜单(), , ) 返回 (#消息处理_结束) .如果真结束 ' 处理规则选择 .如果真 (是否为数值(消息数据.消息内容)) 选择序号 = 到整数(消息数据.消息内容) - 1 .如果真 (选择序号 ≥ 0 且 选择序号 < 取数组成员数(规则名称)) 执行群发(robot_wxid, 取用户消息(robot_wxid), 选择序号) 清除用户(robot_wxid) Api.发送文本消息(robot_wxid, robot_wxid, "已执行群发", , ) 返回 (#消息处理_结束) .如果真结束 .如果真结束 返回 (#消息处理_继续) .子程序结束 .子程序 生成规则菜单, 文本型 .局部变量 菜单, 文本型 .局部变量 i, 整数型 菜单 = "请选择群发规则(回复序号):" .计次循环首(取数组成员数(规则名称), i) 菜单 = 菜单 + #换行符 + 到文本(i + 1) + ". " + 规则名称[i] .计次循环尾() 返回 (菜单) .子程序结束 .子程序 执行群发 .参数 robot_wxid, 文本型 .参数 消息内容, 文本型 .参数 规则索引, 整数型 .局部变量 群数组, 文本型, , "0" .局部变量 i, 整数型 群数组 = 分割文本(规则群组[规则索引], ",", ) .计次循环首(取数组成员数(群数组), i) Api.发送文本消息(robot_wxid, 群数组[i], 消息内容, , ) .计次循环尾() .子程序结束 .子程序 查找用户, 整数型 .参数 robot_wxid, 文本型 .计次循环首(取数组成员数(user等待), ) .如果真 (user等待[] = robot_wxid) 返回 (循环变量 - 1) .如果真结束 .计次循环尾() 返回 (-1) .子程序结束 .子程序 取用户消息, 文本型 .参数 robot_wxid, 文本型 .局部变量 用户索引, 整数型 用户索引 = 查找用户(robot_wxid) .如果真 (用户索引 ≠ -1) 返回 (user等待[user索引]) .否则 返回 ("") .如果真结束 .子程序结束 .子程序 清除用户 .参数 robot_wxid, 文本型 .局部变量 用户索引, 整数型 用户索引 = 查找用户(robot_wxid) .如果真 (用户索引 ≠ -1) 删除成员(user等待, 用户索引) .如果真结束 .子程序结束 ' --------------- 设置窗口相关代码 --------------- .子程序 Qx_Set, , 公开 载入(窗口_设置, , 假) .子程序结束 .窗口程序集 窗口_设置 .程序集变量 规则列表, 列表框, , "规则列表" .程序集变量 重新加载按钮, 按钮, , "重新加载" .子程序 __启动窗口_创建完毕 规则列表.加入项目 (, 0) 规则列表.列表项 = 到文本(规则名称) .子程序结束 .子程序 重新加载按钮_被单击 加载规则 () Api.发送文本消息 (Api.获取当前机器人ID(), Api.获取当前机器人ID(), "规则已重新加载", , ) .子程序结束 .窗口程序集结束 ' 其他事件处理保持默认实现 .子程序 Qx_EventPluginIns, 整数型, 公开 返回 (#事件处理_同意) .子程序结束 .子程序 Qx_EventInject, 整数型, 公开 Api.输出日志 (“有微信注入成功”, , ) 返回 (#消息处理_继续) .子程序结束 帮我看看以上代码有没有错误
03-27
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Java程序员廖志伟

赏我包辣条呗

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

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

打赏作者

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

抵扣说明:

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

余额充值