luffy-(13)

news2025/1/13 11:42:29

内容概览

  • 支付宝支付介绍
  • 支付宝支付二次封装
  • 订单相关表设计
  • 生成订单接口
  • 支付前端
  • 支付宝回调接口

支付宝支付介绍

"""
项目中有需要在线支付功能,可以使用
	支付宝支付(沙箱环境)
	微信支付(需要有备案过的域名)
	云闪付

我们的项目以支付宝支付为例
	支付流程
	使用官方或第三方的API/sdk
		第三方sdk:基于API封装(我们使用这个)
		官方sdk:https://opendocs.alipay.com/open/02no41

支付宝支付介绍
	申请条件很严苛
    沙箱环境:Sandbox:程序的虚拟执行环境,不需要申请各种认证,直接写程序,后期只需要换成只是的秘钥即可
"""

img

使用支付宝支付

1. 使用沙箱环境:https://open.alipay.com/develop/sandbox/app
2. 第三方的sdk:https://github.com/fzlee/alipay
	-基于支付宝的API接口封装的,开源软件,
    -pip install python-alipay-sdk

3. 生成公钥私钥:使用支付宝提供的工具
	-https://opendocs.alipay.com/common/02kipk
	-生成公钥私钥,在本地(私钥好好保管)

4. 把公钥填在支付宝网站上(沙箱环境/正式环境)
	-把我们的公钥填进网站,会生成一个支付宝公钥,以后项目中我们使用支付宝公钥来做

在这里插入图片描述

基本使用

新建两个文件保存应用私钥与支付宝公钥
app_private_key.pem,应用私钥

-----BEGIN RSA PRIVATE KEY-----
base64 encoded content
-----END RSA PRIVATE KEY-----

alipay_public_key.pem,支付宝公钥

-----BEGIN PUBLIC KEY-----
base64 encoded content
-----END PUBLIC KEY-----
from alipay import AliPay
from alipay.utils import AliPayConfig

# 初始化得到对象,传入一堆参数
alipay = AliPay(
    appid="2016092000554611",  # app的id号
    app_notify_url=None,  # 默认回调 url
    app_private_key_string=open('./app_private_key.pem').read(),   # 应用私钥
    alipay_public_key_string=open('./alipay_public_key.pem').read(),  # 支付宝的公钥,验证支付宝回传消息使用,不是你自己的公钥,
    sign_type="RSA2",  # RSA 或者 RSA2
    debug=False,  # 默认 False
    verbose=False,  # 输出调试数据
    config=AliPayConfig(timeout=15)  # 可选,请求超时时间
)

# 生成支付链接
order_string = alipay.api_alipay_trade_page_pay(
    out_trade_no="订单号(uuid)",
    total_amount=99999,  # 支付金额
    subject='精品内衣',  # 订单标题
    return_url="https://example.com",  # get回调地址(前端)
    notify_url="https://example.com/notify"  # post回调地址(后端)
)
# print('https://openapi.alipay.com/gateway.do?'+order_string)  # 真实的地址
print('https://openapi.alipaydev.com/gateway.do?'+order_string)  # 测试的地址

支付宝支付二次封装

"""
目录结构
alipay_common  # 包名
    pem        # 放公钥私钥
    	alipay_public_key.pem
    	app_private_key.pem
    __init__.py 
    pay.py    # 核心文件
    settings.py #配置文件
"""

# 如果保证公钥私钥的安全:
	1. 不在项目中直接写,写在环境变量中
	2. 专门写一个公钥私钥管理的服务,发送请求获取,接口携带很多认证

pay.py

from alipay import AliPay
from alipay.utils import AliPayConfig
from . import settings

# 初始化得到对象,传入一堆参数
alipay = AliPay(
    appid=settings.APPID,  # app的id号
    app_notify_url=None,  # 默认回调 url
    app_private_key_string=settings.APP_PRIVATE_KEY_STRING,
    # 支付宝的公钥,验证支付宝回传消息使用,不是你自己的公钥,
    alipay_public_key_string=settings.ALIPAY_PUBLIC_KEY_STRING,
    sign_type=settings.SIGN,  # RSA 或者 RSA2
    debug=settings.DEBUG,  # 默认 False
    verbose=False,  # 输出调试数据
    config=AliPayConfig(timeout=15)  # 可选,请求超时时间
)

settings.py

import os

# 应用ID
APPID = '2021000121695894'
# 应用私钥
APP_PRIVATE_KEY_STRING = open(
    os.path.join(os.path.dirname(os.path.abspath(__file__)), 'pem', 'app_private_key.pem')).read()
# 支付宝公钥
ALIPAY_PUBLIC_KEY_STRING = open(
    os.path.join(os.path.dirname(os.path.abspath(__file__)), 'pem', 'alipay_public_key.pem')).read()
# 加密方式
SIGN = 'RSA2'

# 是否是支付宝测试环境(沙箱环境),如果采用真是支付宝环境,配置False
DEBUG = True

# 支付网关
GATEWAY = 'https://openapi.alipaydev.com/gateway.do?' if DEBUG else 'https://openapi.alipay.com/gateway.do?'

init

from .pay import alipay
from .settings import GATEWAY

订单相关表设计

1. 创建订单app
...\apps> python ../../manage.py startapp order
# 注册app 

2. 设计表
	-订单表
    -订单详情表 :订单表和订单详情是一对多的关系
from django.db import models
from user.models import UserInfo
from course.models import Course


# Create your models here.
# 订单表
class Order(models.Model):
    """订单模型"""
    status_choices = (
        (0, '未支付'),
        (1, '已支付'),
        (2, '已取消'),
        (3, '超时取消'),
    )
    pay_choices = (
        (1, '支付宝'),
        (2, '微信支付'),
    )
    subject = models.CharField(max_length=150, verbose_name="订单标题")
    total_amount = models.DecimalField(max_digits=10, decimal_places=2, verbose_name="订单总价", default=0)
    # 唯一的,咱们生成的, uuid
    out_trade_no = models.CharField(max_length=64, verbose_name="订单号", unique=True)
    # 支付宝 成功后,会有个流水号
    trade_no = models.CharField(max_length=64, null=True, verbose_name="流水号")
    # 订单状态
    order_status = models.SmallIntegerField(choices=status_choices, default=0, verbose_name="订单状态")
    pay_type = models.SmallIntegerField(choices=pay_choices, default=1, verbose_name="支付方式")
    # 支付时间:支付宝支付完成返回的数据中有这个字段
    pay_time = models.DateTimeField(null=True, verbose_name="支付时间")
    # 订单创建时间,不一定支付
    created_time = models.DateTimeField(auto_now_add=True, verbose_name='创建时间')
    user = models.ForeignKey(UserInfo, related_name='order_user', on_delete=models.DO_NOTHING, db_constraint=False,
                             verbose_name="下单用户")

    class Meta:
        db_table = "luffy_order"
        verbose_name = "订单记录"
        verbose_name_plural = "订单记录"

    def __str__(self):
        return "%s - ¥%s" % (self.subject, self.total_amount)

    @property
    def courses(self):
        data_list = []
        for item in self.order_courses.all():
            data_list.append({
                "id": item.id,
                "course_name": item.course.name,
                "real_price": item.real_price,
            })
        return data_list


# 订单详情表
class OrderDetail(models.Model):
    """订单详情"""
    order = models.ForeignKey(Order, related_name='order_courses', on_delete=models.CASCADE, db_constraint=False,
                              verbose_name="订单")
    course = models.ForeignKey(Course, related_name='course_orders', on_delete=models.CASCADE, db_constraint=False,
                               verbose_name="课程")
    price = models.DecimalField(max_digits=6, decimal_places=2, verbose_name="课程原价")
    real_price = models.DecimalField(max_digits=6, decimal_places=2, verbose_name="课程实价")

    class Meta:
        db_table = "luffy_order_detail"
        verbose_name = "订单详情"
        verbose_name_plural = "订单详情"

    def __str__(self):
        try:
            return "%s的订单:%s" % (self.course.name, self.order.out_trade_no)
        except:
            return super().__str__()

生成订单接口

# 前端:点击购买按钮---->向后端发送请求【需要登录认证】,携带数据{courses:[1,2,3],total_amount:999,subject:订单名,pay_type:1}

# 后端:
	-两个重要的事:生成支付链接,在订单表中插入数据
    -视图类:create;序列化类:数据校验,生成支付链接

视图类

from django.shortcuts import render
from .models import Order
from .serializers import OrderSerializer
from utils.response import APIResponse
from rest_framework.viewsets import GenericViewSet
from rest_framework_jwt.authentication import JSONWebTokenAuthentication
from rest_framework.permissions import IsAuthenticated


# Create your views here.

class OrderView(GenericViewSet):
    queryset = Order.objects.all()
    serializer_class = OrderSerializer
    # 登录后才能够访问
    authentication_classes = [JSONWebTokenAuthentication]
    permission_classes = [IsAuthenticated]

    # 这个新增接口,既要新增,又要返回支付链接;重写create方法
    def create(self, request, *args, **kwargs):
        serializer = self.get_serializer(data=request.data, context={'request': request})
        serializer.is_valid(raise_exception=True)
        serializer.save()
        pay_url = serializer.context.get('pay_url')
        return APIResponse(pay_url=pay_url)

序列化类

import uuid

from rest_framework import serializers
from .models import Order, OrderDetail
from rest_framework.exceptions import APIException
from libs.alipay_common import GATEWAY, alipay
from django.conf import settings
from course.models import Course


class OrderSerializer(serializers.ModelSerializer):
    # courses不是order表的字段,一定要重写
    # 前端传入的是 courses:[1,3,4]----->转换成课程对象[课程对象1,课程对象3,课程对象4]
    courses = serializers.PrimaryKeyRelatedField(queryset=Course.objects.all(), write_only=True, many=True)

    class Meta:
        model = Order
        fields = ['courses', 'total_amount', 'subject', 'pay_type']  # 只用来做数据校验和反序列化

    def _check_price(self, attrs):
        total_amount = attrs.get('total_amount')
        # 循环所有课程,取出价格累加得到总价格
        real_total = 0
        for course in attrs.get('courses'):
            real_total += course.price
        if not total_amount == real_total:  # 不正常,抛异常
            raise APIException('价格不一致')
        else:
            return real_total

    def _get_trade_no(self):
        return str(uuid.uuid4())

    def _get_user(self):
        # 当前登录用户,request.user
        request = self.context.get('request')
        return request.user  # 必须通过了登录认证,才有当前登录用户

    def _get_pay_url(self, total_amount, subject, trade_no):

        pay_str = alipay.api_alipay_trade_page_pay(
            out_trade_no=trade_no,
            total_amount=float(total_amount),
            subject=subject,
            return_url=settings.RETURN_URL,  # get回调地址
            notify_url=settings.NOTIFY_URL  # post回调地址
        )
        return GATEWAY + pay_str

    def _before_create(self, pay_url, attrs, user, trade_no):
        self.context['pay_url'] = pay_url
        attrs['user'] = user
        attrs['out_trade_no'] = trade_no

    def validate(self, attrs):
        # 1 校验价格:计算一下总价格和后端算出来的总价格是否一致
        total_amount = self._check_price(attrs)
        # 2)生成订单号:唯一的,
        trade_no = self._get_trade_no()
        # 3)支付用户:request.user
        user = self._get_user()
        # 4)支付链接生成,放入到self.context中
        pay_url = self._get_pay_url(total_amount, attrs.get('subject'), trade_no)
        # 5)入库(两个表)的信息准备:重写create方法
        self._before_create(pay_url, attrs, user, trade_no)

        return attrs

    def create(self, validated_data):
        # {courses:[对象1,对象2],total_amount:11,subject:xx,pay_type:1,user:user,out_trade_no:3333}
        courses = validated_data.pop('courses')
        order = Order.objects.create(**validated_data)
        # 存订单详情表
        for course in courses:
            OrderDetail.objects.create(order=order, course=course, price=course.price, real_price=course.price)

        return order

路由

from rest_framework.routers import SimpleRouter
from . import views

router = SimpleRouter()
router.register('alipay', views.OrderView, 'alipay')
urlpatterns = [

]
urlpatterns += router.urls

配置文件

# 上线后必须换成公网地址
# 后台基URL
HOST_URL = 'http://127.0.0.1:8000'
# 前台基URL
LUFFY_URL = 'http://127.0.0.1:8080'
# 支付宝同步异步回调接口配置
# 后台异步回调接口 :支付宝post回调地址:后端地址,后端配合一个接口
NOTIFY_URL = HOST_URL + "/order/success/"
# 前台同步回调接口,没有 / 结尾 支付宝get回调地址:前端地址,前端配合一个支付成功的页面组件
RETURN_URL = LUFFY_URL + "/pay/success"

## jwt配置
import datetime
JWT_AUTH = {
    'JWT_EXPIRATION_DELTA': datetime.timedelta(days=7),
}

支付前端

// 在点击支付按钮添加点击事件

go_buy(price, subject) {
  let token = this.$cookies.get('token')
  if (token) {
    // 发送ajax请求
    this.$axios.post(this.$settings.BASE_URL + 'order/alipay/', {
      courses: [this.course_id],
      total_amount: price,
      subject: subject,
      pay_type: 1
    }, {
      headers: {
        'Authorization': 'jwt ' + token
      }
    }).then(res => {
      if (res.data.code == 100) {
        // 打开支付链接
        open(res.data.pay_url, '_self')
      } else {
        this.$message({
          message: res.data.msg,
          type: 'error'
        });
      }

    })
  } else {
    this.$message({
      message: '您没有登录,请先登录',
      type: 'error'
    });
  }
}

支付成功页面PaySuccess.vue

<template>
  <div class="pay-success">
    <!--如果是单独的页面,就没必要展示导航栏(带有登录的用户)-->
    <Header/>
    <div class="main">
      <div class="title">
        <div class="success-tips">
          <p class="tips">您已成功购买 1 门课程!</p>
        </div>
      </div>
      <div class="order-info">
        <p class="info"><b>订单号:</b><span>{{ result.out_trade_no }}</span></p>
        <p class="info"><b>交易号:</b><span>{{ result.trade_no }}</span></p>
        <p class="info"><b>付款时间:</b><span><span>{{ result.timestamp }}</span></span></p>
      </div>
      <div class="study">
        <span>立即学习</span>
      </div>
    </div>
  </div>
</template>

<script>

import Header from "@/components/Header"

export default {
  name: "Success",
  data() {
    return {
      result: {},
    };
  },
  created() {
    // url后拼接的参数:?及后面的所有参数 => ?a=1&b=2
    // console.log(location.search);

    // 解析支付宝回调的url参数
    // location.search 路径中 ? 后的数据
    let params = location.search.substring(1);  // 去除? ------- a=1&b=2
    let items = params.length ? params.split('&') : [];  // ['a=1', 'b=2']
    //逐个将每一项添加到args对象中
    for (let i = 0; i < items.length; i++) {  // 第一次循环a=1,第二次b=2
      let k_v = items[i].split('=');  // ['a', '1']
      //解码操作,因为查询字符串经过编码的
      if (k_v.length >= 2) {
        // url编码反解 https://www.baidu.com/s?ie=utf-8&f=8&rsv_bp=1&rsv_idx=1&tn=baidu&wd=%E7%BE%8E%E5%A5%B3
        let k = decodeURIComponent(k_v[0]);
        this.result[k] = decodeURIComponent(k_v[1]);
        // 没有url编码反解
        // this.result[k_v[0]] = k_v[1];
      }

    }
    // 解析后的结果
    console.log(this.result);


    // 把地址栏上面的支付结果,再get请求转发给后端
    this.$axios.get(this.$settings.BASE_URL + 'order/success/' + location.search,).then(response => {
      if (response.data.code == 100) {
        alert('支付成功')
      } else {
        alert('暂未收到您的付款,请稍后刷新在试')
      }
    }).catch(() => {
      console.log('支付结果同步失败');
    })
  },
  components: {
    Header,
  }
}
</script>

<style scoped>
.main {
  padding: 60px 0;
  margin: 0 auto;
  width: 1200px;
  background: #fff;
}

.main .title {
  display: flex;
  -ms-flex-align: center;
  align-items: center;
  padding: 25px 40px;
  border-bottom: 1px solid #f2f2f2;
}

.main .title .success-tips {
  box-sizing: border-box;
}

.title img {
  vertical-align: middle;
  width: 60px;
  height: 60px;
  margin-right: 40px;
}

.title .success-tips {
  box-sizing: border-box;
}

.title .tips {
  font-size: 26px;
  color: #000;
}


.info span {
  color: #ec6730;
}

.order-info {
  padding: 25px 48px;
  padding-bottom: 15px;
  border-bottom: 1px solid #f2f2f2;
}

.order-info p {
  display: -ms-flexbox;
  display: flex;
  margin-bottom: 10px;
  font-size: 16px;
}

.order-info p b {
  font-weight: 400;
  color: #9d9d9d;
  white-space: nowrap;
}

.study {
  padding: 25px 40px;
}

.study span {
  display: block;
  width: 140px;
  height: 42px;
  text-align: center;
  line-height: 42px;
  cursor: pointer;
  background: #ffc210;
  border-radius: 6px;
  font-size: 16px;
  color: #fff;
}
</style>

支付宝回调接口

"""
支付成功,支付号会有两个回调
	-get 回调,回调给前端
		-为了保证准确性,支付宝回调会前端后,我们自己向后端发送一个请求,查询一下这个订单是否支付成功
	
	-post 回调,回调给后端接口
		-后端接口,接收支付宝的回调,修改订单状态
		-支付宝在24小时内,会有8次回掉,防止没有收到


后端需要写两个接口:
	-post回调,给支付宝用
	-get回调,给前端做二次校验使用
"""
from rest_framework.exceptions import APIException
from utils.common_logger import logger
from utils.response import APIResponse
from rest_framework.response import Response

class PaySuccessView(ViewSet):
    def list(self, request):
        try:
            out_trade_no = request.query_params.get('out_trade_no')
            Order.objects.get(out_trade_no=out_trade_no, order_status=1)
            return APIResponse()
        except Exception as e:
            raise APIException('该订单还没支付成功')

    def create(self, request):  # 给支付宝用的,写完后无法测试-----》外网无法直接访问到内网
        try:
            # from django.http.request import QueryDict
            # print(type(request.data))
            result_data = request.data.dict()  # 回调回来编码格式是urlencoded,QueryDic对象---》.dict--->转成真正的字典对象
            out_trade_no = result_data.get('out_trade_no')
            signature = result_data.pop('sign')  # 如果是QueryDic对象不允许pop
            from libs import alipay_common
            # 验证签名,result_data和signature验证签名,sdk帮咱们写好了,一定要验证签名
            result = alipay_common.alipay.verify(result_data, signature)
            if result and result_data["trade_status"] in ("TRADE_SUCCESS", "TRADE_FINISHED"):
                # 完成订单修改:订单状态、流水号、支付时间
                Order.objects.filter(out_trade_no=out_trade_no).update(order_status=1,
                                                                       pay_time=result_data.get('timestamp'),
                                                                       trade_no=result_data.get('trade_no'))
                # 完成日志记录
                logger.warning('%s订单支付成功' % out_trade_no)
                return Response('success')  # 支付宝要求的格式
            else:
                logger.error('%s订单支付失败' % out_trade_no)
                return Response('failed')
        except:
            return Response('failed')

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

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

相关文章

【动手学深度学习】关于“softmax回归的简单实现”报错的解决办法(含源代码)

目录&#xff1a;关于“softmax回归的简单实现”报错的解决办法一、前言二、实现步骤2.1 导包2.2 初始化模型参数2.3 重新审视Softmax的实现2.4 优化算法2.5 训练2.6 源代码三、问题出现四、问题的解决五、再跑代码六、改正后的函数源代码一、前言 在之前的学习中&#xff0c;…

题库系统(公众号免费调用)

题库系统(公众号免费调用) 本平台优点&#xff1a; 多题库查题、独立后台、响应速度快、全网平台可查、功能最全&#xff01; 1.想要给自己的公众号获得查题接口&#xff0c;只需要两步&#xff01; 2.题库&#xff1a; 查题校园题库&#xff1a;查题校园题库后台&#xff0…

arthas 源码构建

arthas 源码构建 git下载代码 git clone https://github.com/alibaba/arthas.git 若github被墙&#xff0c;可以在gitee搜索下载 maven clean 可以在项目目录执行 mvn clean &#xff0c; ide可以执行界面执行 maven package 可以在项目目录执行mvn package 问题记录 ja…

四肽Suc-AAPD-对硝基苯胺,165174-58-3

粒酶B底物succ - aapd - pna。也被ICE劈开了。 编号: 177581中文名称: 四肽Suc-Ala-Ala-Pro-Asp-对硝基苯胺CAS号: 165174-58-3单字母: Suc-AAPD-pNA三字母: Suc-Ala-Ala-Pro-Asp-pNA氨基酸个数: 4分子式: C25H32O11N6平均分子量: 592.56精确分子量: 592.21等电点(PI): -pH7.0时…

新进场的獴哥健康、至真健康们,讲不出互联网医疗的新故事

文丨智能相对论 作者丨沈浪 曾几何时&#xff0c;互联网医疗风靡一时&#xff0c;现如今潮水退去&#xff0c;当市场回归理性&#xff0c;赛道竞争趋于同质化&#xff0c;一批互联网医疗企业正在试图通过讲好新故事&#xff0c;来拉开品牌与品牌之间的商业差距&#xff0c;寻…

ArcGIS Pro 转换Smart3D生成的倾斜3D模型数据osgb——创建集成网格场景图层包

最近在做Arcgis 批处理的一些工作&#xff0c;然后再学习Python的同时&#xff0c;偶然觉得arcgis Pro是个好东西呢&#xff1f;然后结合近期的Smart3D倾斜3D模型数据&#xff0c;是否可以在arcgis里查看呢&#xff1f;带着这样的疑问和好奇&#xff0c;开始了arcgis Pro的学习…

【408专项篇】C语言笔记-第四章(选择与循环)

第四章&#xff1a;选择、循环 第一节&#xff1a;选择if-else 1. 关系表达式与逻辑表达式 if-else的判断条件结果还是真与假&#xff0c;即1或0&#xff0c;一般是关系表达式或逻辑表达式。 算术运算符的优先级高于关系运算符&#xff0c;关系运算符的优先级高于逻辑与和逻…

Webpack 5 超详细解读(四)

31.proxy 代理设置 为什么开发阶段需要设置代理,在开发阶段,我们需要请求后端接口,但是一般后端接口地址和我们本地的不在同一个服务中提供,这时进行访问就会存在跨域的问题,所以我们需要对我们的请求进行转啊操作。模拟跨域请求代码如下: https://api.github.com/users…

高项 沟通管理论文

3个过程&#xff1a; 1&#xff0c;规划沟通管理&#xff1a;根据干系人的信息需要和要求及组织的可用资产情况&#xff0c;制订合适的项目沟通方式和计划的过程。 2&#xff0c;管理沟通&#xff1a;根据沟通管理计划,生成、收集、分发、储存、检索及最终处置项目信息的过程…

Spring boot 实践Rabbitmq消息防丢失

之前看很多网上大佬的防丢失的文章&#xff0c;文章中理论知识偏多&#xff0c;所以自己想着实践一下&#xff0c;实践过程中也踩了一些坑&#xff0c;因此写出了这篇文章。如果文章有误人子弟的地方&#xff0c;望在评论区指出。 导致消息出现丢失的原因 发送时失败&#xff…

295348-87-7,AF 594 Succinimidyl Ester可用于成像和流式细胞分析

理论分析&#xff1a; 中文名&#xff1a;AF 594活性酯 英文名&#xff1a;AF 594 Succinimidyl Ester&#xff0c;Alexa Fluor 594 NHS Ester&#xff0c;AF 594 NHS Ester CAS号&#xff1a;295348-87-7 化学式&#xff1a;C39H37N3O13S2 分子量&#xff1a;819.85 ex/em : 5…

招投标业务总结

最近接了一个招标投标的项目&#xff0c;开发完成后&#xff0c;整个招投标的流程也就理清楚了&#xff0c;简介一下业务过程。业务主流程可以分为3个阶段: 招标方建立招标项目发布招标公告投标人参与竞标招标方开标评选公示 系统可以划分为两套&#xff0c;一个是给招标方使…

若依框架解读(微服务版)——2.模块间的调用逻辑(ruoyi-api模块)(OpenFeign)

模块之间的关系 我们可以了解到一共有这么多服务&#xff0c;我们先启动这三个服务 其中rouyi–api模块是远程调用也就是提取出来的openfeign的接口 ruoyi–commom是通用工具模块 其他几个都是独立的服务 ruoyi-api模块 api模块当中有几个提取出来的OpenFeign的接口 分别为文件…

华为机试 - ABR 车路协同场景

目录 题目描述 输入描述 输出描述 用例 题目解析 算法源码 题目描述 数轴有两个点的序列 A{A1&#xff0c; A2, …, Am}和 B{B1, B2, ..., Bn}&#xff0c; Ai 和 Bj 均为正整数&#xff0c; A、 B 已经从小到大排好序&#xff0c; A、 B 均肯定不为空&#xff0c; 给定…

大数据培训课程Reduce Join案例实操

Reduce Join案例实操 1&#xff0e;需求 表4-4 订单数据表t_order idpidamount100101110020221003033100401410050251006036 表4-5 商品信息表t_product pidpname01小米02华为03格力将商品信息表中数据根据商品pid合并到订单数据表中。 表4-6 最终数据形式 idpnameamount…

2022我的前端面试总结

Webpack Proxy工作原理&#xff1f;为什么能解决跨域 1. 是什么 webpack proxy&#xff0c;即webpack提供的代理服务 基本行为就是接收客户端发送的请求后转发给其他服务器 其目的是为了便于开发者在开发模式下解决跨域问题&#xff08;浏览器安全策略限制&#xff09; 想…

盘点 | 跨平台桌面应用开发的5大主流框架

受益于开源技术的发展&#xff0c;以及响应快速开发的实际业务需求&#xff0c;跨平台开发不仅限于移动端跨平台&#xff0c;桌面端虽然在市场应用方面场景不像移动端那么丰富&#xff0c;但也有市场的需求。 相对于个人开发者而言&#xff0c;跨平台框架的使用&#xff0c;主…

Vue开发 提交后台,二维码,自定义

1. 修改title和图标 资源可以放在static下面&#xff0c;给一个小的&#xff1a; 直接再index里面改&#xff1a; 不生效&#xff0c;需要在 vue.config.js 中增加&#xff1a; module.exports {pwa: {iconPaths: {favicon32: logo.png,favicon16: logo.png,appleTouchIcon:…

阿里巴巴全新SpringCloud实战笔记(全彩版)GitHub狂揽70000标星

最近小编淘到一份宝贝&#xff01; 先看看目录&#xff1a; 这份手册真的非常全面&#xff0c;涵盖了所有SpringCloud所有的内容&#xff0c;限于文章篇幅原因&#xff0c;只能以截图的形式展示出来&#xff0c;有需要的小伙伴可以文末获取↓↓↓ 直接展示内容&#xff1a; …

react redux 状态管理

1.store store是一个状态管理容器&#xff0c;它通过createStore创建&#xff0c;createStore接收initialState和reducer两个参数。它暴露了4个api分别是&#xff1a; getState() dispatch(action) subscribe(listener) replaceReducer 前三个是比较常用的api&#xff0c;之…