【SpringBoot】SpringBoot集成支付宝支付

news2024/11/22 23:23:51

文章目录

  • 背景
  • 1、预期效果展示
  • 2. 开发流程
    • 2.1 沙盒调试
      • 2.1.1. 创建沙盒应用
      • 2.1.2. SpringBoot代码实现
      • 2.1.3. 前端代码实现
    • 2.2 创建并上线APP

背景

在开始集成支付宝支付之前,我们需要准备一个支付宝商家账户,如果是个人开发者,可以通过注册公司或者让有公司资质的单位进行授权,后续在集成相关API的时候需要提供这些信息。

下面我以电脑网页端在线支付为例,介绍整个从集成、测试到上线的具体流程。

1、预期效果展示

在开始之前我们先看下我们要达到的最后效果,具体如下:

  • 前端点击支付跳转到支付宝界面
  • 支付宝界面展示付款二维码
  • 用户手机端支付
  • 完成支付,支付宝回调开发者指定的url。
    在这里插入图片描述

2. 开发流程

2.1 沙盒调试

支付宝为我们准备了完善的沙盒开发环境,我们可以先在沙盒环境调试好程序,后续新建好应用并成功上线后,把程序中对应的参数替换为线上参数即可。

2.1.1. 创建沙盒应用

直接进入 open.alipay.com/develop/san… 创建沙盒应用即可,
在这里插入图片描述

这里因为是测试环境,我们就选择系统默认密钥就行了,下面选择公钥模式,另外应用网关地址就是用户完成支付之后,支付宝会回调的url。在开发环境中,我们可以采用内网穿透的方式,将我们本机的端口暴露在某个公网地址上,这里推荐 natapp.cn/ ,可以免费注册使用。

2.1.2. SpringBoot代码实现

在创建好沙盒应用,获取到密钥,APPID,商家账户PID等信息之后,就可以在测试环境开发集成对应的API了。这里我以电脑端支付API为例,介绍如何进行集成。

关于电脑网站支付的详细产品介绍和API接入文档可以参考:opendocs.alipay.com/open/repo-0… 和 opendocs.alipay.com/open/270/01…

  • 步骤1, 添加alipay sdk对应的Maven依赖。

    <!-- alipay -->  
    <dependency>  
       <groupId>com.alipay.sdk</groupId>  
       <artifactId>alipay-sdk-java</artifactId>  
       <version>4.35.132.ALL</version>  
    </dependency>
    
  • 步骤2,添加支付宝下单、支付成功后同步调用和异步调用的接口。

    这里需要注意,同步接口是用户完成支付后会自动跳转的地址,因此需要是Get请求。异步接口,是用户完成支付之后,支付宝会回调来通知支付结果的地址,所以是POST请求。

    @RestController  
    @RequestMapping("/alipay")  
    public class AliPayController {  
      
        @Autowired  
        AliPayService aliPayService;  
      
        @PostMapping("/order")  
        public GenericResponse<Object> placeOrderForPCWeb(@RequestBody AliPayRequest aliPayRequest) {  
            try {  
                return aliPayService.placeOrderForPCWeb(aliPayRequest);  
            } catch (IOException e) {  
                throw new RuntimeException(e);  
            }  
        }  
      
        @PostMapping("/callback/async")  
        public String asyncCallback(HttpServletRequest request) {  
            return aliPayService.orderCallbackInAsync(request);  
        }  
      
        @GetMapping("/callback/sync")  
        public void syncCallback(HttpServletRequest request, HttpServletResponse response) {  
            aliPayService.orderCallbackInSync(request, response);  
        }  
    
    }
    
  • 步骤3,实现Service层代码

    这里针对上面controller中的三个接口,分别完成service层对应的方法。下面是整个支付的核心流程,其中有些地方需要根据你自己的实际情况进行保存订单到DB或者检查订单状态的操作,这个可以根据实际业务需求进行设计。

public class AliPayService {  
  
    @Autowired  
    AliPayHelper aliPayHelper;  
  
    @Resource  
    AlipayConfig alipayConfig;  
  
    @Transactional(rollbackFor = Exception.class)  
    public GenericResponse<Object> placeOrderForPCWeb(AliPayRequest aliPayRequest) throws IOException {  
        log.info("【请求开始-在线购买-交易创建】*********统一下单开始*********");  
  
        String tradeNo = aliPayHelper.generateTradeNumber();  
    
        String subject = "购买套餐1";  
        Map<String, Object> map = aliPayHelper.placeOrderAndPayForPCWeb(tradeNo, 100, subject);  
  
        if (Boolean.parseBoolean(String.valueOf(map.get("isSuccess")))) {  
            log.info("【请求开始-在线购买-交易创建】统一下单成功,开始保存订单数据");  
  
            //保存订单信息  
            // 添加你自己的业务逻辑,主要是保存订单数据
  
            log.info("【请求成功-在线购买-交易创建】*********统一下单结束*********");  
            return new GenericResponse<>(ResponseCode.SUCCESS, map.get("body"));  
        }else{  
            log.info("【失败:请求失败-在线购买-交易创建】*********统一下单结束*********");  
            return new GenericResponse<>(ResponseCode.INTERNAL_ERROR, String.valueOf(map.get("subMsg")));  
        }  
    }  
  
    // sync return page  
    public void orderCallbackInSync(HttpServletRequest request, HttpServletResponse response) {  
        try {  
            OutputStream outputStream = response.getOutputStream();  
            //通过设置响应头控制浏览器以UTF-8的编码显示数据,如果不加这句话,那么浏览器显示的将是乱码  
            response.setHeader("content-type", "text/html;charset=UTF-8");  
            String outputData = "支付成功,请返回网站并刷新页面。";  
  
            /**  
             * data.getBytes()是一个将字符转换成字节数组的过程,这个过程中一定会去查码表,  
             * 如果是中文的操作系统环境,默认就是查找查GB2312的码表,  
             */  
            byte[] dataByteArr = outputData.getBytes("UTF-8");//将字符转换成字节数组,指定以UTF-8编码进行转换  
            outputStream.write(dataByteArr);//使用OutputStream流向客户端输出字节数组  
        } catch (IOException e) {  
            throw new RuntimeException(e);  
        }  
    }  
  
    public String orderCallbackInAsync(HttpServletRequest request) {  
        try {  
            Map<String, String> map = aliPayHelper.paramstoMap(request);  
            String tradeNo = map.get("out_trade_no");  
            String sign = map.get("sign");  
            String content = AlipaySignature.getSignCheckContentV1(map);  
            boolean signVerified = aliPayHelper.CheckSignIn(sign, content);  
  
            // check order status  
            // 这里在DB中检查order的状态,如果已经支付成功,无需再次验证。
            if(从DB中拿到order,并且判断order是否支付成功过){  
                log.info("订单:" + tradeNo + " 已经支付成功,无需再次验证。");  
                return "success";  
            }  
  
            //验证业务数据是否一致  
            if(!checkData(map, order)){  
                log.error("返回业务数据验证失败,订单:" + tradeNo );  
                return "返回业务数据验证失败";  
            }  
            //签名验证成功  
            if(signVerified){  
                log.info("支付宝签名验证成功,订单:" + tradeNo);  
                // 验证支付状态  
                String tradeStatus = request.getParameter("trade_status");  
                if(tradeStatus.equals("TRADE_SUCCESS")){  
                    log.info("支付成功,订单:"+tradeNo);  
			        // 更新订单状态,执行一些业务逻辑

                    return "success";  
                }else{  
                    System.out.println("支付失败,订单:" + tradeNo );  
                    return "支付失败";  
                }  
            }else{  
                log.error("签名验证失败,订单:" + tradeNo );  
                return "签名验证失败.";  
            }  
        } catch (IOException e) {  
            log.error("IO exception happened ", e);  
            throw new RuntimeException(ResponseCode.INTERNAL_ERROR, e.getMessage());  
        }  
    }  
  
  
    public boolean checkData(Map<String, String> map, OrderInfo order) {  
        log.info("【请求开始-交易回调-订单确认】*********校验订单确认开始*********");  
  
        //验证订单号是否准确,并且订单状态为待支付  
        if(验证订单号是否准确,并且订单状态为待支付){  
            float amount1 = Float.parseFloat(map.get("total_amount"));  
            float amount2 = (float) order.getOrderAmount();  
            //判断金额是否相等  
            if(amount1 == amount2){  
                //验证收款商户id是否一致  
                if(map.get("seller_id").equals(alipayConfig.getPid())){  
                    //判断appid是否一致  
                    if(map.get("app_id").equals(alipayConfig.getAppid())){  
                        log.info("【成功:请求开始-交易回调-订单确认】*********校验订单确认成功*********");  
                        return true;                    }  
                }  
            }  
        }  
        log.info("【失败:请求开始-交易回调-订单确认】*********校验订单确认失败*********");  
        return false;    }  
}
  • 步骤4,实现alipayHelper类。这个类里面对支付宝的接口进行封装。

    public class AliPayHelper {  
      
        @Resource  
        private AlipayConfig alipayConfig;  
      
        //返回数据格式  
        private static final String FORMAT = "json";  
        //编码类型  
        private static final String CHART_TYPE = "utf-8";  
        //签名类型  
        private static final String SIGN_TYPE = "RSA2";  
      
        /*支付销售产品码,目前支付宝只支持FAST_INSTANT_TRADE_PAY*/  
        public static final String PRODUCT_CODE = "FAST_INSTANT_TRADE_PAY";  
      
        private static AlipayClient alipayClient = null;  
      
        private static final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMddHHmmssSSS");  
        private static final Random random = new Random();  
      
        @PostConstruct  
        public void init(){  
            alipayClient = new DefaultAlipayClient(  
                    alipayConfig.getGateway(),  
                    alipayConfig.getAppid(),  
                    alipayConfig.getPrivateKey(),  
                    FORMAT,  
                    CHART_TYPE,  
                    alipayConfig.getPublicKey(),  
                    SIGN_TYPE);  
        };  
      
        /*================PC网页支付====================*/  
        /**  
         * 统一下单并调用支付页面接口  
         * @param outTradeNo  
         * @param totalAmount  
         * @param subject  
         * @return  
         */  
        public Map<String, Object> placeOrderAndPayForPCWeb(String outTradeNo, float totalAmount, String subject){  
            AlipayTradePagePayRequest request = new AlipayTradePagePayRequest();  
            request.setNotifyUrl(alipayConfig.getNotifyUrl());  
            request.setReturnUrl(alipayConfig.getReturnUrl());  
            JSONObject bizContent = new JSONObject();  
            bizContent.put("out_trade_no", outTradeNo);  
            bizContent.put("total_amount", totalAmount);  
            bizContent.put("subject", subject);  
            bizContent.put("product_code", PRODUCT_CODE);  
      
            request.setBizContent(bizContent.toString());  
            AlipayTradePagePayResponse response = null;  
            try {  
                response = alipayClient.pageExecute(request);  
            } catch (AlipayApiException e) {  
                e.printStackTrace();  
            }  
            Map<String, Object> resultMap = new HashMap<>();  
            resultMap.put("isSuccess", response.isSuccess());  
            if(response.isSuccess()){  
                log.info("调用成功");  
                log.info(JSON.toJSONString(response));  
                resultMap.put("body", response.getBody());  
            } else {  
                log.error("调用失败");  
                log.error(response.getSubMsg());  
                resultMap.put("subMsg", response.getSubMsg());  
            }  
            return resultMap;  
        }  
      
        /**  
         * 交易订单查询  
         * @param out_trade_no  
         * @return  
         */  
        public Map<String, Object> tradeQueryForPCWeb(String out_trade_no){  
            AlipayTradeQueryRequest request = new AlipayTradeQueryRequest();  
            JSONObject bizContent = new JSONObject();  
            bizContent.put("trade_no", out_trade_no);  
            request.setBizContent(bizContent.toString());  
            AlipayTradeQueryResponse response = null;  
            try {  
                response = alipayClient.execute(request);  
            } catch (AlipayApiException e) {  
                e.printStackTrace();  
            }  
            Map<String, Object> resultMap = new HashMap<>();  
            resultMap.put("isSuccess", response.isSuccess());  
            if(response.isSuccess()){  
                System.out.println("调用成功");  
                System.out.println(JSON.toJSONString(response));  
                resultMap.put("status", response.getTradeStatus());  
            } else {  
                System.out.println("调用失败");  
                System.out.println(response.getSubMsg());  
                resultMap.put("subMsg", response.getSubMsg());  
            }  
            return resultMap;  
        }  
      
        /**  
         * 验证签名是否正确  
         * @param sign  
         * @param content  
         * @return  
         */  
        public boolean CheckSignIn(String sign, String content){  
            try {  
                return AlipaySignature.rsaCheck(content, sign, alipayConfig.getPublicKey(), CHART_TYPE, SIGN_TYPE);  
            } catch (AlipayApiException e) {  
                e.printStackTrace();  
            }  
            return false;  
        }  
      
        /**  
         * 将异步通知的参数转化为Map  
         * @return  
         */  
        public Map<String, String> paramstoMap(HttpServletRequest request) throws UnsupportedEncodingException {  
            Map<String, String> params = new HashMap<String, String>();  
            Map<String, String[]> requestParams = request.getParameterMap();  
            for (Iterator<String> iter = requestParams.keySet().iterator(); iter.hasNext();) {  
                String name = (String) iter.next();  
                String[] values = (String[]) requestParams.get(name);  
                String valueStr = "";  
                for (int i = 0; i < values.length; i++) {  
                    valueStr = (i == values.length - 1) ? valueStr + values[i] : valueStr + values[i] + ",";  
                }  
                // 乱码解决,这段代码在出现乱码时使用。  
    //            valueStr = new String(valueStr.getBytes("ISO-8859-1"), "utf-8");  
                params.put(name, valueStr);  
            }  
            return params;  
        }  
    
    }
    
  • 步骤5,封装config类,用于存放所有的配置属性。

    @Data  
    @Component  
    @ConfigurationProperties(prefix = "alipay")  
    public class AlipayConfig {  
      
        private String gateway;  
      
        private String appid;  
      
        private String pid;  
      
        private String privateKey;  
      
        private String publicKey;  
      
        private String returnUrl;  
      
        private String notifyUrl;  
      
    }
    

另外需要在application.properties中,准备好上述对应的属性。

# alipay config  
alipay.gateway=https://openapi.alipaydev.com/gateway.do  
alipay.appid=your_appid
alipay.pid=your_pid  
alipay.privatekey=your_private_key
alipay.publickey=your_public_key
alipay.returnurl=完成支付后的同步跳转地址 
alipay.notifyurl=完成支付后,支付宝会异步回调的地址

2.1.3. 前端代码实现

前端代码只需要完成两个功能,

  1. 根据用户的请求向后端发起支付请求。
  2. 直接提交返回数据完成跳转。

下面的例子中,我用typescript实现了用户点击支付之后的功能,

async function onPositiveClick() {  
   paymentLoading.value = true  
  
   const { data } = await placeAlipayOrder<string>({  
	//你的一些请求参数,例如金额等等
   })  
  
   const div = document.createElement('divform')  
   div.innerHTML = data  
   document.body.appendChild(div)  
   document.forms[0].setAttribute('target', '_blank')  
   document.forms[0].submit()  
  
   showModal.value = false  
   paymentLoading.value = false  
}

2.2 创建并上线APP

完成沙盒调试没问题之后,我们需要创建对应的支付宝网页应用并上线。
登录 open.alipay.com/develop/man… 并选择创建网页应用,
在这里插入图片描述

填写应用相关信息:
在这里插入图片描述
创建好应用之后,首先在开发设置中,设置好接口加签方式以及应用网关。
在这里插入图片描述

注意密钥选择RSA2,其他按照上面的操作指南一步步走即可,注意保管好自己的私钥和公钥。
之后在产品绑定页,绑定对应的API,比如我们这里是PC网页端支付,找到对应的API绑定就可以了。如果第一次绑定,可能需要填写相关的信息进行审核,按需填写即可,一般审核一天就通过了。
在这里插入图片描述

最后如果一切就绪,我们就可以把APP提交上线了,上线成功之后,我们需要把下面SpringBoot中的properties替换为线上APP的信息,然后就可以在生产环境调用支付宝的接口进行支付了。

# alipay config  
alipay.gateway=https://openapi.alipaydev.com/gateway.do  
alipay.appid=your_appid
alipay.pid=your_pid  
alipay.privatekey=your_private_key
alipay.publickey=your_public_key
alipay.returnurl=完成支付后的同步跳转地址 
alipay.notifyurl=完成支付后,支付宝会异步回调的地址

参考:

  • blog.csdn.net/xqnode/arti…
  • blog.51cto.com/u_15754099/…
  • zhuanlan.zhihu.com/p/596771147

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

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

相关文章

硬件速攻-E18-D80NK红外光电传感器

介绍 E18-D80NK光电开关是一款常见的光电传感器&#xff0c;具有高性价比、安装方便等优点。其工作原理是利用发射管发射出的红外线照射目标物&#xff0c;当目标物接近开关时&#xff0c;被照射的红外线反射回接收器&#xff0c;接收器就会向微处理器发出信号&#xff0c;从而…

1.9C++不同数据类型转换

C不同数据类型转换 在 C中&#xff0c;不同类型之间的数据转换可以通过强制类型转换&#xff08;类型转换运算符&#xff09;来实现。 C 中强制类型转换有以下三种&#xff1a; 1、static_cast static_cast 可以用于基本数据类型之间的转换&#xff0c;也可以用于类层次结构…

从实现到原理,总结11种延迟任务的实现方式(下)

7 监听Redis过期key 在Redis中&#xff0c;有个发布订阅的机制 生产者在消息发送时需要到指定发送到哪个channel上&#xff0c;消费者订阅这个channel就能获取到消息。图中channel理解成MQ中的topic。 并且在Redis中&#xff0c;有很多默认的channel&#xff0c;只不过向这些…

小鱼深度产品测评之:阿里云云效代码管理 Codeup,一款数十万企业正在使用,全方位保护企业代码资产的实力产品,。

云效代码管理 Codeup 0、引言1、进入页面2、创建代码库3、资源文件页面4、分支→新建保护分支规则5、分支-基本设置5.1 基本信息 模块5.2 存储空间管理 6、分支→仓库备份6.1 点击 如何启用 按钮6.2 点击 前往企业设置查看 7、合并请求8、度量报表9、动态10、流水线11、总结 0、…

基于html+css的图展示129

准备项目 项目开发工具 Visual Studio Code 1.44.2 版本: 1.44.2 提交: ff915844119ce9485abfe8aa9076ec76b5300ddd 日期: 2020-04-16T16:36:23.138Z Electron: 7.1.11 Chrome: 78.0.3904.130 Node.js: 12.8.1 V8: 7.8.279.23-electron.0 OS: Windows_NT x64 10.0.19044 项目…

24届秋招专场 · 数组如何用双指针解题呢?

你好&#xff0c;我是安然无虞。 文章目录 删除有序数组中的重复项删除排序链表中的重复元素移除元素移除零 大家好&#xff0c;近期主要更新数组相关的解题算法咯&#xff0c;感兴趣的老铁可以一起看过来啦。 今天更新使用双指针解决数组部分题型&#xff0c;注意哦&#xff…

支付宝小程序云亮相!向小程序生态开放全面云服务

前言&#xff1a; 小程序是一种轻量级应用程序&#xff0c;不需要安装即可直接在手机上使用。相较于传统的APP来讲&#xff0c;其无需下载安装&#xff0c;轻便快捷&#xff0c;快速启动&#xff0c;易于推广的良好特性为我们所青睐。 为此&#xff0c;支付宝小程序云&#xff…

DataX在有赞大数据平台的实践

文章目录 一、需求二、选型三、前期设计3.1 运行形态3.2 执行器设计3.3 开发策略 四、Datax-Web五、总结 大家好&#xff0c;我是脚丫先生 (o^^o) 在看技术文章的时候&#xff0c;发现有赞平台采用过Datax。想到指北数据中台&#xff0c;数据汇聚采用的是Datax-web二次开发&am…

chatgpt赋能python:Python的字符串索引操作技巧

Python的字符串索引操作技巧 Python是一个强大而灵活的编程语言&#xff0c;被广泛用于各种领域。在Python中&#xff0c;字符串是一个非常重要的数据类型&#xff0c;它可以包含文本、数字、符号和其他任何字符。在处理字符串时&#xff0c;索引操作是常见的操作之一。本文将…

计算机网络之网络层:数据平面

四.网络层&#xff1a;数据平面 4.1 网络层概述 网络层被分解为两个相互作用的部分&#xff0c;即数据平面和控制平面。 数据平面决定到达路由器输入链路之一的数据报如何转发到该路由器的输出链路之一&#xff0c;转发方式有&#xff1a; 传统的IP转发&#xff1a;转发基于…

【算法与数据结构】707、LeetCode设计链表

文章目录 一、题目二、设计链表三、完整代码 所有的LeetCode题解索引&#xff0c;可以看这篇文章——【算法和数据结构】LeetCode题解。 一、题目 二、设计链表 思路分析&#xff1a;这里我将的成员函数放在类外实现了&#xff0c;这样链表类看起来更加简洁&#xff0c;方便大家…

jenkins用户权限管理

环境准备: 登录jenkins: http://192.168.9.190:8091/ admin asdwhl@0 一、用户权限插件安装 1、Dashboard > Manage Jenkins > Manage Plugins > Available(可选插件) 依次安装: Role-based Authorization Strategy Authorize Project

校园二手交易平台实训报告

目 录 一、设计背景 1. 需求分析 2. 课题研究的目的和意义 二、系统需求分析与开发环境 1. 系统功能需求 2. 系统界面需求 3. 开发环境 三、系统设计 四、系统测试 1. 脑模拟器测试 五、总结与展望 六、重要程序 1. LoginActivity 2. RegisterActiv…

51Proteus仿真数控0~20mA恒流源串口DAC0832数码管显示-0036

51Proteus仿真数控0~20mA恒流源串口DAC0832数码管显示-0036 Proteus仿真小实验&#xff1a; 51Proteus仿真数控0~20mA恒流源串口DAC0832数码管显示-0036 功能&#xff1a; 硬件组成&#xff1a;AT89C51单片机 6位数码管DAC0832电压输出多个按键&#xff08;设置、移动、加、…

INTERSPEECH 2023论文|基于多频带时频注意力的复调音乐旋律提取

论文题目&#xff1a; MTANet: Multi-band Time-frequency Attention Network for Singing Melody Extraction from Polyphonic Music 作者列表&#xff1a; 高虞安&#xff0c;胡英&#xff0c;王柳淞&#xff0c;黄浩&#xff0c;何亮 研究背景 复调音乐是一种具有多个声…

[PyTorch][chapter 41][卷积网络实战-LeNet5]

前言 这里结合前面学过的LeNet5 模型&#xff0c;总结一下卷积网络搭建&#xff0c;训练的整个流程 目录&#xff1a; 1&#xff1a; LeNet-5 2: 卷积网络总体流程 3&#xff1a; 代码 一 LeNet-5 LeNet-5是一个经典的深度卷积神经网络&#xff0c;由Yann LeCun在1998年提…

zabbix-agent安装

1.CentOS release 5 1-1.centos5 32位 [rootLV zabbix]# cat /etc/redhat-release CentOS release 5 (Final) [rootLV zabbix]# uname -a Linux LV 2.6.18-53.el5xen #1 SMP Mon Nov 12 03:26:12 EST 2007 i686 i686 i386 GNU/Linux确定了系统centos5 32位rpm方式安装&#…

Ubuntu18.04离线安装redis

因需要安装redis的服务器无法连接互联网&#xff0c;所以需要离线安装。首先需要下载redis的安装包&#xff0c;之后进行安装&#xff0c;在安装之前需要保证gcc&#xff0c;g&#xff0c;make等依赖包已经安装。 1. 安装gcc等依赖包 依赖包安装请参考&#xff1a; Ubuntu18…

CI570 3BSE001440R1需要电流显示和就地/远传控制

​ CI570 3BSE001440R1需要电流显示和就地/远传控制 CI570 3BSE001440R1需要电流显示和就地/远传控制 如果变频器与通讯方式与DCS系统连接&#xff0c;则只需要计算1个通讯点&#xff0c;不需要计算其他点数。 &#xff08;6&#xff09;如DCS系统外接电磁阀、指示灯、接触器等…

物联网云平台数据存储方案,这次我终于找对了

《高并发系统实战派》-- 你值得拥有 文章目录 物联网云平台存储概述为什么要做存储&#xff1f;存储的意义在哪里&#xff1f;数据存储方案设计存储数据库选型需要考虑的因素数据库选型结构化数据半结构化数据非结构化数据 案例分析第一颗栗子第二颗栗子第三颗栗子第四颗栗子 …