[golang gin框架] 33.Gin 商城项目- 集成支付宝微信支付、生成支付二维码、监听处理异步通知跳转到订单页面

news2024/11/28 4:50:13

一.界面展示

当用户点击'去支付'时,请求支付界面,并 展示对应订单相关数据,以及 支付方式相关操作,点击对应的支付方式,进行支付操作
该界面对应的功能:
1.进入该界面,后台逻辑判断: 是否存在该订单,如果不存在,则跳转到购物车页面;如果存在,则获取对应订单相关数据,并 渲染到页面
2.选择支付方式,根据不同支付方式进行支付操作
支付宝支付: 点击 '支付宝'支付方式,会请求后台,通过支付宝相关配置以及方法,生成一个 支付url,重定向到对应的支付url,用户扫描支付,支付完成后,跳转到服务端设置的 返回页面,然后支付宝服务器会请求配置的服务端 回调地址,进行订单校验以及对应的 订单逻辑处理
微信支付: 点击'微信支付'支付方式,也会请求后台,通过微信相关配置以及方法,生成一个支付 二维码,展示到界面,用户扫描支付完成后,微信服务器会请求配置的服务端 回调地址,进行订单校验以及对应的 订单逻辑处理(修改订单支付状态等),然后前端在这个支付页面会每个几秒(一般方法是定时 ajax轮询,或者 web_socket请求)请求一个接口,判断该订单是否支付,如果支付了,就跳转到支付成功页面

去支付页面

微信支付页面

点击'微信支付'支付方式,也会请求后台,通过微信相关配置以及方法,生成一个支付二维码,展示到界面,用户扫描支付完成后,微信服务器会请求配置的服务端回调地址,进行订单校验以及对应的订单逻辑处理(修改订单支付状态等),然后前端在这个支付页面会每个几秒(一般方法是定时ajax轮询,或者web_socket请求)请求一个接口,判断该订单是否支付,如果支付了,就跳转到支付成功页面

微信支付页面

点击 '支付宝'支付方式,会请求后台,通过支付宝相关配置以及方法,生成一个支付url,重定向到对应的支付url,用户扫描支付,支付完成后,跳转到服务端设置的返回页面,然后支付宝服务器会请求配置的服务端回调地址,进行订单校验以及对应的订单逻辑处理

二.代码展示

  1. 集成支付宝支付控制器代码

要使用 github.com/smartwalle/alipay/v3需要先引入:
直接在import中引入 github.com/smartwalle/alipay/v3,然后go mod tidy就可以了
package frontend

import (
    "fmt"
    "net/http"
    "time"
    "github.com/gin-gonic/gin"
    "github.com/smartwalle/alipay/v3"
)

type AlipayController struct{}

//支付宝支付请求接口
func (con AlipayController) Alipay(c *gin.Context) {
    //1、获取订单号 判断此订单号是否值当前用户的
    //2、获取订单里面的支付信息
    var privateKey = "xxx" // 必须,上一步中使用 RSA签名验签工具 生成的私钥
    var client, err = alipay.New("2021xxx588", privateKey, true)
    client.LoadAppPublicCertFromFile("crt/appCertPublicKey_2021xxx588.crt") // 加载应用公钥证书
    client.LoadAliPayRootCertFromFile("crt/alipayRootCert.crt")   // 加载支付宝根证书
    client.LoadAliPayPublicCertFromFile("crt/alipayCertPublicKey_RSA2.crt")  // 加载支付宝公钥证书

    // 将 key 的验证调整到初始化阶段
    if err != nil {
        fmt.Println(err)
        return
    }

    //支付宝PC扫描方式支付
    var p = alipay.TradePagePay{}
    p.NotifyURL = "http://xxx/v3/alipayNotify"  // 回调地址
    p.ReturnURL = "http://xxx5/v3/alipayReturn"  //支付后跳转地址
    p.Subject = "测试 公钥证书模式-这是一个gin订单"
    template := "2006-01-02 15:04:05"
    p.OutTradeNo = time.Now().Format(template)
    p.TotalAmount = "0.1"  //支付金额(元)
    p.ProductCode = "FAST_INSTANT_TRADE_PAY"  //根据支付方式调整

    var url, err4 = client.TradePagePay(p)
    if err4 != nil {
        fmt.Println(err4)
    }

    //支付url
    var payURL = url.String()

    //重定向到支付url
    c.Redirect(http.StatusFound, payURL)
}

//支付回调
func (con AlipayController) AlipayNotify(c *gin.Context) {
    var privateKey = "xxx" // 必须,上一步中使用 RSA签名验签工具 生成的私钥
    var client, err = alipay.New("202xxx588", privateKey, true)
    client.LoadAppPublicCertFromFile("crt/appCertPublicKey_202xxx588.crt") // 加载应用公钥证书
    client.LoadAliPayRootCertFromFile("crt/alipayRootCert.crt")  // 加载支付宝根证书
    client.LoadAliPayPublicCertFromFile("crt/alipayCertPublicKey_RSA2.crt") // 加载支付宝公钥证书

    if err != nil {
        fmt.Println(err)
        return
    }

    req := c.Request
    req.ParseForm()
    //支付校验
    ok, _ := client.VerifySign(req.Form)

    fmt.Println(ok)

    fmt.Println(req.Form)
    //订单逻辑处理...
    c.String(200, "ok")
}

//支付后跳转页面
func (con AlipayController) AlipayReturn(c *gin.Context) {
    c.String(200, "支付成功")
}
  1. 集成微信支付控制器代码

要使用 github.com/objcoding/wxpay需要先引入:
直接在import中引入github.com/objcoding/wxpay,然后go mod tidy就可以了, qrcode的引入方式也是一样的
package frontend

import (
    "fmt"
    "net/http"
    "time"
    "github.com/gin-gonic/gin"
    "github.com/objcoding/wxpay"
    qrcode "github.com/skip2/go-qrcode"
)

type WxpayController struct{}

//微信支付
func (con WxpayController) Wxpay(c *gin.Context) {
    //1、获取订单号 判断此订单号是否值当前用户的
    //2、获取订单里面的支付信息
    //1、配置基本信息
    account := wxpay.NewAccount(
        "wx7xxx6e4",  // appId
        "150xxx",  //商户id
        "zhongyxxx66",  //密钥
        false,
    )
    client := wxpay.NewClient(account)

    //2、获取ip地址,订单号等信息
    ip := c.ClientIP()
    template := "200601021504"
    tradeNo := time.Now().Format(template)
    //3、调用统一下单
    params := make(wxpay.Params)
    params.SetString("body", "Gin微信支付").
        SetString("out_trade_no", tradeNo).
        SetInt64("total_fee", 1). //1分
        SetString("spbill_create_ip", ip).
        SetString("notify_url", "http://xxx/wxpay/notify"). //必须在商户平台的Native支付回调链接里面配置
        // SetString("trade_type", "APP")
        SetString("trade_type", "NATIVE") //网站支付需要改为NATIVE

    p, err1 := client.UnifiedOrder(params)
    if err1 != nil {
        c.String(http.StatusOK, "生成二维码失败")
        return
    }
    //4、获取code_url 生成支付二维码
    png, err := qrcode.Encode(p["code_url"], qrcode.Medium, 256)
    if err != nil {
        c.String(http.StatusOK, "生成二维码失败")
        return
    }
    c.String(http.StatusOK, string(png))
}

/*
支付回调:
1、发布到服务器测试
2、必须在商户平台的Native支付回调链接里面配置
3、如何接收XML的数据  c.GetRawData()
4、如何获取数据
5、如何验证数据
6、更新数据
*/
func (con WxpayController) WxpayNotify(c *gin.Context) {
    //1、获取表单传过来的xml数据
    xmlByte, _ := c.GetRawData()
    xmlStr := string(xmlByte)
    postParams := wxpay.XmlToMap(xmlStr)
    //2、校验签名
    account := wxpay.NewAccount(
        "wx7xxxe4",
        "150xxx1",
        "zhongxxx66",
        false,
    )
    client := wxpay.NewClient(account)
    isValidate := client.ValidSign(postParams)

    fmt.Println(isValidate)
    fmt.Println("-----更新订单-----")
    fmt.Println(postParams)
    //3、更新订单
    c.String(200, "ok")
}
  1. 查看订单支付状态方法

在controllers/frontend/BuyController.go下增加查看订单支付状态方法:前端支付页面,微信支付会每个几秒请求该接口,根据返回结果,判断是否跳转到订单支付成功页面
//查看订单支付状态
func (con BuyController) OrderPayStatus(c *gin.Context) {
    id, err := models.Int(c.Query("id"))
    if err != nil {
        c.JSON(http.StatusOK, gin.H{
            "success": false,
            "message": "传入参数错误",
        })
        return
    }
    //获取用户信息
    user := models.User{}
    models.Cookie.Get(c, "user", &user)

    //获取主订单信息
    order := models.Order{}
    models.DB.Where("id = ?", id).Find(&order)

    //判断当前数据是否合法
    if user.Id != order.Uid {
        c.JSON(http.StatusOK, gin.H{
            "success": false,
            "message": "非法请求",
        })
        return
    }

    //判断是否支付
    if order.PayStatus == 1 && order.OrderStatus == 1 {
        c.JSON(http.StatusOK, gin.H{
            "success": true,
            "message": "支付成功",
        })
        return
    } else {
        c.JSON(http.StatusOK, gin.H{
            "success": false,
            "message": "支付成功",
        })
        return
    }
}
  1. 路由

把微信支付,支付宝支付相关路由写入routers/frontendRouters.go中
//查看订单支付状态
defaultRouters.GET("/buy/orderPayStatus", middlewares.InitUserAuthMiddleware, frontend.BuyController{}.OrderPayStatus)
//支付宝支付
defaultRouters.GET("/alipay", middlewares.InitUserAuthMiddleware, frontend.AlipayController{}.Alipay)
//支付宝支付回调
defaultRouters.POST("/alipayNotify", frontend.AlipayController{}.AlipayNotify)
//支付宝支付完成后跳转
defaultRouters.GET("/alipayReturn", middlewares.InitUserAuthMiddleware, frontend.AlipayController{}.AlipayReturn)
//微信支付
defaultRouters.GET("/wxpay", middlewares.InitUserAuthMiddleware, frontend.WxpayController{}.Wxpay)
//微信支付回调
defaultRouters.POST("/wxpay/notify", frontend.WxpayController{}.WxpayNotify)
  1. html

微信,支付宝支付操作处理
{{ define "frontend/buy/pay.html" }}

{{ template "frontend/public/page_header.html" .}}
<!--end header -->
<link rel="stylesheet" href="/static/frontend/css/pay.css" />
<script src="/static/frontend/js/bootstrap.js"></script>
<link rel="stylesheet" href="/static/frontend/css/bootstrap.css">
<!-- start banner_x -->
<div class="banner_x center clearfix">
    <a href="/" target="_blank">
        <div class="logo fl"></div>
    </a>
    <div class="wdgwc fl ml40">支付页面 </div>
</div>

<div class="page-main">
    <div class="checkout-box">
        <div class="section section-order">
            <div class="order-info clearfix">
                <div class="fl">
                    <h2 class="title">订单提交成功!去付款咯~</h2>
                    <p class="order-time" id="J_deliverDesc"></p>
                    <p class="order-time">请在<span class="pay-time-tip">47小时59分</span>内完成支付, 超时后将取消订单</p>
                    <p class="post-info" id="J_postInfo">
                        收货信息: {{.order.Name}} &nbsp;&nbsp;{{.order.Phone}} &nbsp;&nbsp; {{.order.Address}} </p>
                </div>
                <div class="fr">
                    <p class="total">
                        应付总额:<span class="money"><em>1000元</em>元</span>
                    </p>
                    <br>
                    <br>
                    <a href="javascript:void(0);" class="show-detail" id="J_showDetail"
                        data-stat-id="db85b2885a2fdc53">订单详情</a>
                </div>
            </div>
            <i class="iconfont icon-right">√</i>
            <div class="order-detail">
                <ul>
                    <li class="clearfix">
                        <div class="label">订单号:</div>
                        <div class="content">
                            <span class="order-num">{{.order.OrderId}}</span>
                        </div>
                    </li>
                    <li class="clearfix">
                        <div class="label">收货信息:</div>
                        <div class="content">
                            收货信息: {{.order.Name}} &nbsp;&nbsp;{{.order.Phone}} &nbsp;&nbsp; {{.order.Address}} </div>
                    </li>
                    <li class="clearfix">
                        <div class="label">商品:</div>
                        <div class="content">
                            {{range $key,$value:=.orderItems}}
                            <p>{{$value.ProductTitle}} {{$value.GoodsVersion}} {{$value.GoodsColor}}
                                数量:{{$value.ProductNum}} 价格:{{$value.ProductPrice}}</p>
                            {{end}}
                        </div>
                    </li>
                    <li class="clearfix hide">
                        <div class="label">配送时间:</div>
                        <div class="content">
                            不限送货时间 </div>
                    </li>
                    <li class="clearfix">
                        <div class="label">发票信息:</div>
                        <div class="content">
                            电子发票 个人 </div>
                    </li>
                </ul>
            </div>
        </div>

        <div class="section section-payment">
            <div class="cash-title" id="J_cashTitle">
                选择以下支付方式付款
            </div>

            <div class="payment-box ">

                <div class="payment-body">
                    <ul class="clearfix payment-list J_paymentList J_linksign-customize">
                        <ul class="clearfix payment-list J_paymentList J_linksign-customize">
                            <li id="weixinPay">
                                <img src="/static/frontend/image/weixinpay0701.png" alt="微信支付" />
                            </li>
                            <li id="alipay">
                                <a href="/alipay?id={{.order.Id}}" target="_blank">
                                    <img src="/static/frontend/image/payOnline_zfb.png" alt="支付宝" /></a>
                            </li>
    
                        </ul>

                    </ul>
                </div>
            </div>

        </div>
    </div>
</div>

<!-- 微信支付 Dialog -->
<div class="modal fade" id="weixinPayModel" tabindex="-1" role="dialog">
    <div class="modal-dialog" role="document">
        <div class="modal-content">
            <div class="modal-header">
                <button type="button" class="close" data-dismiss="modal" aria-label="Close"><span
                        aria-hidden="true">&times;</span></button>
                <h4 class="modal-title" id="myModalLabel">微信支付</h4>
            </div>
            <div class="modal-body">
                <img class="lcode" src="/wxpay?id={{.order.Id}}" />
                <img class="rphone" src="/static/frontend/image/phone.png" />
            </div>
        </div>
    </div>
</div>
<!-- 支付宝支付Dialog -->

<div class="modal fade" id="alipayModel" tabindex="-1" role="dialog">
    <div class="modal-dialog" role="document">
        <div class="modal-content">
            <div class="modal-header">
                <button type="button" class="close" data-dismiss="modal" aria-label="Close"><span
                        aria-hidden="true">&times;</span></button>
                <h4 class="modal-title" id="myModalLabel">在线支付提醒</h4>
            </div>
            <div class="modal-body">

                感谢您的支持,支付完成后,页面会自动跳转到订单页面,若需要重新支付请点击“继续支付”按钮,若已完成支付请点击“已完成支付”
                <div class="modal-footer">

                    <button type="button" class="btn btn-default" data-dismiss="modal">继续支付</button>

                    <button type="button" class="btn btn-primary" id="alipayDone">已完成支付</button>
                </div>
            </div>

        </div>
    </div>
</div>

<script>
    $(function () {
        $(".show-detail").click(function () {
            $(".order-detail").slideToggle();
        })
        $("#weixinPay").click(function () {
            $('#weixinPayModel').modal('show');
        })
        $("#alipay").click(function () {
            $('#alipayModel').modal('show');
        })
        $("#alipayDone").click(function () {
            location.href = "/user/order"
        })
        //微信支付:弹出二维码模态框后,每个3秒定时定期支付状态,根据状态,判断逻辑
        setInterval(function () {
            $.get('/buy/orderPayStatus?id={{.order.Id}}', function (response) {            
                if (response.success) { //跳转到支付成功订单页面
                    location.href = '/user/order'
                }
            })
        }, 3000);
    })
</script>
<!-- footer -->
{{ template "frontend/public/page_footer.html" .}}
</body>
</html>
{{end}}

[上一节][golang gin框架] 33.Gin 商城项目- 微信支付操作相关功能讲解

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/524541.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

0511课后作业(C高级)

1.编写一个名为myfirstshell.sh的脚本&#xff0c;它包括以下内容。 1、包含一段注释&#xff0c;列出您的姓名、脚本的名称和编写这个脚本的目的 2、和当前用户说“hello 用户名” 3、显示您的机器名 hostname 4、显示上一级目录中的所有文件的列表 5、显示变量PATH和HOM…

iOS播放与编辑HDR视频

在iPhone12发布后&#xff0c;支持使用Dolby Vision来录制HDR视频。至此&#xff0c;升级到iOS14.1系统后&#xff0c;已经支持录制、播放、编辑和导出HDR视频。接下来&#xff0c;让我们一起探索HDR视频的各种操作。 一、HDR视频边编辑边预览 1、Profile与Level HDR视频中&…

Java奠基】实现面向对象编程方法

目录 标准的JavaBean类 设计对象并使用 对象封装 this关键字 构造方法 要知道对象是一个又一个能帮助我们解决问题的东西&#xff0c;但是这些东西并不是凭空出现的&#xff0c;需要我们根据设计图来进行制造&#xff0c;而这些一个一个的设计图就是一个一个的类。 标准的…

ChatGPT分销版多开v3.9.1-新增 语音识别和绘画多个引擎-已测试

众所周知ChatGPT在国内是无法正常使用的 而我们模块要做的就是这一点让普通人使用上ChatGPT 或娱乐或作为生产力工具 当前ChatGPT是非常火的 但是国内环境复杂&#xff0c;所以机会来了。。。 ChatGPT分销版多开v3.9.1&#xff1a;公众号H5版本 目前没反编译前端容易封号…

计算机网络-SNMP协议与pysnmp

1.概念 2.典型架构 3.snmp的信息交互 4.MIB 4.1常见MIB节点 5.SNMP管理模型 MIB位于被管理进程 6.SNMP的三个版本 6.1 SNMPv1 6.2 SNMPv2C 6.3 SNMPv3 6.3.1 SNMP3的基本操作 6.3.2 SNMP交互GET 6.3.3 SNMP交互-GETBULK 6.3.4 SNMP交互-SET 6.3.5 SNMP交互-trap 6.3.6 SNMP交…

【开源之夏 2023】欢迎报名 Dragonfly、Kata Containers、Nydus 社区项目!

开源之夏是由“开源软件供应链点亮计划”发起并长期支持的一项暑期开源活动&#xff0c;旨在鼓励在校学生积极参与开源软件的开发维护&#xff0c;促进优秀开源软件社区的蓬勃发展&#xff0c;培养和发掘更多优秀的开发者。 活动联合国内外各大开源社区&#xff0c;针对重要开…

Dubbo消费端源码深入分析(8)

目录 简介 过滤器 &#xff08;Filter&#xff09; 负载均衡接口 &#xff08;LoadBalance&#xff09; 容错接口 &#xff08;Cluster&#xff09; 源码分析 1. 获取Invoker过程 2. 获取动态代理对象proxy 3. 最后调用此动态代理对象的invoke方法 过滤器、容错组件、负…

基础IO(二)

磁盘 1.基础概念2.磁盘线性理解3.文件系统4.inode与文件名5.理解增删查改6.补充细节 &#x1f31f;&#x1f31f;hello&#xff0c;各位读者大大们你们好呀&#x1f31f;&#x1f31f; &#x1f680;&#x1f680;系列专栏&#xff1a;【Linux的学习】 &#x1f4dd;&#x1f4…

Chapter7:非线性控制系统分析(下)

第七章:非线性控制系统分析 Exercise7.11 设非线性系统结构图如下图所示,分析系统运动并计算自振参数。 解: 将 3 3 3个串联非线性环节进行等效合并,由于反馈通道饱和特性与前向通道饱和特性同时进入饱和状态,所以反馈通道的非线性环节相当于不起作用,将前向通道的另两…

【计算机视觉 | 自然语言处理】Hugging Face 超详细介绍和使用教程

文章目录 一、前言二、可以获得什么&#xff1f;三、入门实践3.1 帮助文档3.2 安装3.3 模型的组成3.4 BERT模型的使用3.4.1 导入模型3.4.2 使用模型3.4.2.1 tokenizer 3.5 model3.6 后处理 一、前言 Hugging Face 起初是一家总部位于纽约的聊天机器人初创服务商&#xff0c;他…

URP渲染管线里面的摄像机用法

大家好&#xff0c;我是阿赵&#xff0c;这里继续讲一下URP渲染管线。 这次要讲的是URP渲染管线里面的摄像机用法 之前介绍过&#xff0c;URP摄像机和普通摄像机的属性显示上有比较大的变化&#xff1a; 接下来从用法上来说明一下&#xff1a; 1、多个摄像机的处理变化 多个…

unity UGUI系统梳理 -交互组件

概述 unity 中的交互组件可用于处理交互&#xff0c;例如鼠标或触摸事件以及使用键盘或控制器进行的交互 1、按钮 (Button) Button详解 2、开关 (Toggle) Background&#xff1a;背景图片&#xff0c;控制toggle组件的背景颜色改变&#xff0c;从而展示此物体是否被选中的…

5.View的事件分发机制/事件处理机制原理分析

事件MotionEvent包含了哪几个? ACTION_DOWN 手指触碰到屏幕时触发,只会执行一次ACTION_MOVE 手指在屏幕上滑动出发,会执行多次ACTION_UP 手指抬起离开屏幕出发,只会执行一次ACTION_CANCEL 事件被上层拦截时会触发 父容器ViewGroup需要从子View手中抢夺分发的事件进行处理时,会…

【SCI】综合能源系统中热电联产、电制气和碳捕集系统的建模与优化(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

Java 网络编程 —— 非阻塞式编程

线程阻塞概述 在生活中&#xff0c;最常见的阻塞现象是公路上汽车的堵塞。汽车在公路上快速行驶&#xff0c;如果前方交通受阻&#xff0c;就只好停下来等待&#xff0c;等到公路顺畅&#xff0c;才能恢复行驶。 线程在运行中也会因为某些原因而阻塞。所有处于阻塞状态的线程…

C++数据结构:哈希 -- unordered系列容器、哈希表的结构以及如何通过闭散列的方法解决哈希冲突

目录 一. unordered系列关联式容器 1.1 unrodered_map和unordered_set 综述 1.2 常见的接口函数&#xff08;以unordered_map为例&#xff09; 1.3 unordered系列与map和set的效率对比 二. 哈希表的底层结构 2.1 什么是哈希 2.2 哈希函数 2.3 哈希冲突 三. 通过闭散列的…

JavaEE(系列2) -- 多线程(创建多线程)

讲述下面的内容之前,先来回顾一下一个重要的知识点 进程和线程之间的区别 1.进程包括线程。 2.进程有自己独立的内存空间和文件描述符表。同一个进程中的多个线程之间&#xff0c;共享同一份地址空间和文件描述符表。 3.进程是操作系统资源分配的基本单位&#xff0c;线程是操作…

5.14学习周报

文章目录 前言文献阅读摘要介绍方法模型框架评价指标结果结论 时间序列预测总结 前言 本周阅读文献《A Hybrid Model for Water Quality Prediction Based on an Artificial Neural Network, Wavelet Transform, and Long Short-Term Memory》&#xff0c;文献主要提出了基于人…

iconfont-extract: 一个将iconfont图标转化为React组件的工具

iconfont 提供了海量的图标&#xff0c;同时也方便了前端开发者使用这些图标&#xff0c;只需要添加对应的js、css或者字体文件即可。在我们的项目中使用添加js文件的方式&#xff0c;js文件中都包含了所有的图标&#xff0c;一个项目中通常只会使用其中的一部分&#xff0c;所…

【架构设计】DDD 到底解决了什么问题

文章目录 前言一、架构设计是为了解决系统复杂度1.1 架构设计的误区1.1.1 每个系统都要做架构设计/公司流程要求有架构设计1.1.2 架构设计是为了追求高性能、高可用、可扩展性等单一目标 1.2 架构设计的真正目的1.3 系统复杂度的六个来源及通用解法1.3.1 高性能1.3.1.1 单机复杂…