程序汪10万接的多平台视频分发项目,模拟人工发视频

news2024/11/28 8:31:13

本项目来自程序汪背后的私活小团队,开发了一个多平台分发视频项目,给粉丝分享一下解决方案和具体项目分开情况付款情况等等细节,希望给想接私活的朋友一些经验参考

程序汪10万接的多平台视频分发项目,模拟人工发视频

视频版本 在 B站【我是程序汪】

图片

目录

  • 一、项目构成

  • 二、开发人员

  • 三、项目背景

  • 四、APP端

  • 五、后端系统

项目构成

  • 系统基本构成:uniapp(app)+ java后台+selenium(蛇尼泥嗯)免费的分布式的自动化测试工具

selenium

Selenium 是一套 Web网站 的程序自动化操作 解决方案。

通过它,我们可以写出自动化程序,像真人一样在浏览器里操作web界面。比如点击界面按钮,在文本框中输入文字,在B站上发布视频 等操作。

而且还能从web界面获取信息。比如获取 12306火车、汽车票务信息,某招聘网站职位信息,财经网站股票价格信息 等等,然后用程序进行分析处理。

安装命

开发人员以及费用情况

  • 开发周期60天 

  • 开发人数 2人  1前1后

  • 工作量比例  20%前端 10%设计 70%后端

  • 整体费用是10万(不包含维护费,一期的费用哦)

  • 走的正规公司合同

  • 云服务器3台 配置 8核16G(根据情况可增加)

  • 维护费用:项目总款的10% 一年

  • 付款方式 5+4+1 (预付+验收+尾款)

项目背景

现在自媒体视频平台非常火爆,某营销公司为了推广自己的产品,当然后期也可以自己卖会员,需要在多个视频平台一键分发(抖音,B站,今日头条,小红书,快手等等 十多个平台),这样就不用人工一个个平台登陆去发视频了,效率飞速提高,目前我们集成了16个视频平台,后期可能会继续增加

图片

图片

其中关键流程

图片

APP端

本APP是用uniapp开发的,这种外包小项目肯定要选择性价比高的开发方式(程序汪知道原生的好,但开发成本也高哦),利用uniapp可以让多端开发成本降到最低

图片

图片

图片

优点

uni-app是一套可以适用多端的开源框架,一套代码可以同时生成ios,Android,H5,微信小程序,支付宝小程序,百度小程序等。

缺点

uni-app问世的时间还比较短,有很多地方还不是完善,坑很多,如果不喜欢爬坑的朋友那就少用uni-app吧

B站的核心代码

由于是商业代码,所以放出来的是比较老的版本,最新版本不敢乱发啊

核心思路基本差不多,最新版本单机变集群 还使用到了线程池等性能手段,来提高并发量

 <dependency>
      <groupId>org.seleniumhq.selenium</groupId>
      <artifactId>selenium-java</artifactId>
      <version>3.141.59</version>
  </dependency>


package com.video.browserService;

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.openqa.selenium.By;
import org.openqa.selenium.JavascriptExecutor;
import org.openqa.selenium.OutputType;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebElement;

import com.video.ThreadPool.task.workTask;
import com.video.bean.Task;
import com.video.constants.sys_constants;
import com.video.os.WindowsUtils;
import com.video.util.Base64Util;
import com.video.util.WebDriverUtil;
import com.video.util.codeUtil;

import ch.qos.logback.core.net.SyslogOutputStream;

import org.bytedeco.javacv.FFmpegFrameGrabber;
import org.bytedeco.javacv.Frame;
import org.bytedeco.javacv.Java2DFrameConverter;

import javax.imageio.ImageIO;

import java.awt.Image;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.util.UUID;

public class bibi_browser extends workTask{
 private WebDriver webDriver;
 private String username="";
 private String password="";
 private String path="";
 private String content="";
 private Task task;
 
 public bibi_browser(Task task,String username,
   String password,
   String path,
   String content){
  this.task=task;
  this.username=username;
  this.password=password;
  this.path=path;
  this.content=content;
 }
 @Override
 public void excute() throws IOException {
  try{
   setSerid(task.getSerid());
   webDriver=WebDriverUtil.getInstance(getSerid()).getDriver();
   task.setTask_status(1);
   Map<String,String> result=moblieLogin(username, password);
   if(result!=null
     &&result.get("result")!=null
     &&result.get("result").toString().equals("success")){
    result=UploadVideo(path,content);
    task.setResult(result);
   }else{
    task.setResult(result);
   }
   task.setTask_status(3);
  }catch(Exception e){
   Map<String,String> result=new HashMap<String,String>();
   result.put("result", "error");
   result.put("msg", "上传视频异常");
   task.setResult(result);
   task.setTask_status(3);
  }
  sys_constants.taskMap.put(task.getSessionId(), task);
 }
 @Override
 public String info() {
  // TODO Auto-generated method stub
  return null;
 }
 public Map<String,String> moblieLogin(String username,String password){
  //webDriver.manage().deleteAllCookies();
     Map<String,String> resultMap=new HashMap<String,String>();
     try{
      webDriver.get("https://www.bilibili.com/");
      WebDriverUtil
   .getInstance(getSerid()).WebDriverWaitElement(
     10,
     1,
     By.xpath("//img[contains(@class,'bili-avatar-img bili-avatar-face bili-avatar-img-radius')]"));
   
   WebElement head_img=WebDriverUtil
        .getInstance(getSerid())
        .findElement(By.xpath("//img[contains(@class,'bili-avatar-img bili-avatar-face bili-avatar-img-radius')]"));
   if(head_img!=null){
    resultMap.put("result", "success");
    resultMap.put("msg", "登录成功");
    return resultMap;
   }
   WebElement login_btn=WebDriverUtil
   .getInstance(getSerid()).findElement(
     By.xpath("//*[@id='i_cecream']/div[1]/div[1]/ul[2]/li[8]/li/a/div/span"));
   
   if(login_btn!=null){
    login_btn.click();
   }
   WebDriverUtil
   .getInstance(getSerid()).WebDriverWaitElement(
     10,
     1,
     By.xpath("//div[@id='geetest-wrap']"));
   *[@id="layout"]/div/div[2]/p[1]/a
   WebElement sms_btn=WebDriverUtil
   .getInstance(getSerid()).findElement(
     By.xpath("//*[@id='geetest-wrap']/div/div[1]/span[2]"));
   
   if(sms_btn!=null){
    sms_btn.click();
   }
   
      WebDriverUtil
   .getInstance(getSerid()).WebDriverWaitElement(
     10,
     1,
     By.xpath("//a[contains(@class,'btn btn-login')]"));
   
      //输入手机号码
   WebElement userinput =WebDriverUtil
        .getInstance(getSerid())
        .findElement(By.xpath("//input[@name='tel']"));
   if(userinput!=null){
    userinput.sendKeys(username);
   }
   //点击发送短信按钮
   WebElement sendMobile =WebDriverUtil
        .getInstance(getSerid())
        .findElement(By.xpath("//button[contains(@class,'el-button captcha-buttom el-button--primary')]"));
   if(sendMobile!=null){
    sendMobile.click();
    try{
     Thread.currentThread().sleep(2000);

     WebElement backimg=WebDriverUtil
          .getInstance(getSerid())
          .findElement(By.xpath("//img[@class='geetest_item_img']"));
     WebElement clickimg=WebDriverUtil
          .getInstance(getSerid())
          .findElement(By.xpath("//div[@class='geetest_tip_img']"));
     if(backimg!=null){
      String baseimg="";
      if(clickimg!=null){
       File src =clickimg.getScreenshotAs(OutputType.FILE);
          baseimg=WebDriverUtil
          .getInstance(getSerid())
          .captureElement(src, clickimg);
      }
      File backsrc =backimg.getScreenshotAs(OutputType.FILE);
      String back_img=WebDriverUtil
         .getInstance(getSerid())
         .captureElement(backsrc, backimg);
      //String back_img=Base64Util.imgBase64(backimg.getAttribute("src"));
         String result=codeUtil.loadCodeResult("27",back_img,null, null, baseimg);
         if(result!=null&&!result.trim().equals("")){
          String []xys=result.split("\\|");
          if(xys!=null&&xys.length>0){
           WebDriverUtil
              .getInstance(getSerid()).moveAndClick(backimg,xys);
          }
          WebElement que_btn=WebDriverUtil
               .getInstance(getSerid())
               .findElement(By.xpath("//a[@class='geetest_commit']"));
          if(que_btn!=null){
           que_btn.click();
          }
         }else{
          resultMap.put("result", "error");
          resultMap.put("msg", "验证码识别错误,登录失败");
          return resultMap;
         }
     }
    
    }catch(Exception e){
     e.printStackTrace();
    }
   }
   //输入短信验证码
   WebElement pwdinput =WebDriverUtil
        .getInstance(getSerid())
        .findElement(By.xpath("//input[contains(@class,'el-input__inner') and contains(@type,'text') and contains(@placeholder,'请输入短信验证码')]"));
   if(pwdinput!=null){
    String mobilemsg="";
    long time=new Date().getTime();
    task.setTask_status(2);
    while(sys_constants.mobilemsg.get(task.getSessionId())==null
      &&(new Date().getTime()-time<5*60*1000)){
     Thread.currentThread().sleep(5000);
    }
    mobilemsg=sys_constants.mobilemsg.get(task.getSessionId());
    if(mobilemsg!=null&&!mobilemsg.trim().equals("")){
       pwdinput.sendKeys(mobilemsg);
    }else{
     resultMap.put("result", "error");
     resultMap.put("msg", "请输入短信验证码,登录失败");
     return resultMap;
    }
   }
   WebElement submit_btn=WebDriverUtil
        .getInstance(getSerid())
        .findElement(By.xpath("//a[contains(@class,'btn btn-login') and contains(string(), '登录')]"));
   if(submit_btn!=null){
    submit_btn.click();
    //error-- 为保证账号安全,请使用手机验证码登录
    Thread.currentThread().sleep(1000);
    WebDriverUtil
    .getInstance(getSerid()).WebDriverWaitElement(
      10,
      1,
      By.xpath("//a[@class='avatar el-popover__reference']"));
    
    WebElement img_btn=WebDriverUtil
         .getInstance(getSerid())
         .findElement(By.xpath("//a[@class='avatar el-popover__reference']"));
    
    if(img_btn!=null){
     resultMap.put("result", "success");
        resultMap.put("msg", "登录成功");
    }else{
     resultMap.put("result", "error");
     resultMap.put("msg", "登录失败");
     return resultMap;
    }
   }
     
     }catch(Exception e){
      e.printStackTrace();
      resultMap.put("result", "error");
   resultMap.put("msg", "登录异常,请求页面异常");
     }
     return resultMap;
 }
 /**
  * 浏览器登录B站
  * @return
  */
    public Map<String,String> login(String username,String password){
     //webDriver.manage().deleteAllCookies();
     Map<String,String> resultMap=new HashMap<String,String>();
     return resultMap;
    }
    public Map<String,String> UploadVideo(String path,String content){
     Map<String,String> resultMap=new HashMap<String,String>();
     try{
      webDriver.get("https://member.bilibili.com/platform/upload/video/frame");
      WebDriverUtil
   .getInstance(getSerid()).WebDriverWaitElement(
     2,
     1,
     By.xpath("//div[@class='guide_left__wrap']/div[@class='tips_wrap']/img[@class='jump']"));
      
      WebElement guide_left_btn=WebDriverUtil
        .getInstance(getSerid())
        .findElement(By.xpath("//div[@class='guide_left__wrap']/div[@class='tips_wrap']/img[@class='jump']"));
      if(guide_left_btn!=null&&guide_left_btn.isDisplayed()){
       guide_left_btn.click();
      }
      WebDriverUtil
   .getInstance(getSerid()).WebDriverWaitElement(
     2,
     1,
     By.xpath("//div[@class='guide_right__wrap']/div[@class='tips_wrap']/img[@class='jump']"));
      
      WebElement guide_right_btn=WebDriverUtil
        .getInstance(getSerid())
        .findElement(By.xpath("//div[@class='guide_right__wrap']/div[@class='tips_wrap']/img[@class='jump']"));
      if(guide_right_btn!=null&&guide_right_btn.isDisplayed()){
       guide_right_btn.click();
      }
      
      WebDriverUtil
   .getInstance(getSerid()).WebDriverWaitElement(
     10,
     1,
     By.id("video-up"));
      WebDriverUtil
   .getInstance(getSerid())
   .getDriver()
   .switchTo()
   .frame("videoUpload");
      ((JavascriptExecutor)WebDriverUtil
        .getInstance(getSerid()).getDriver()).executeScript("document.getElementsByTagName('input').style='display:block';");
      WebElement upload_btn=WebDriverUtil
        .getInstance(getSerid())
        .findElement(By.xpath("//*[@id='video-up-app']/div/div[2]/div/div[1]/div/div/input"));
      if(upload_btn!=null){
       //upload_btn.setAttribute("value", "D:\\CRM生成代码\\VID_20211113_111758.mp4");
       upload_btn.sendKeys(path);
       
//       List<String> imglist=videoImage(path, 960, 600);
//       WebElement upload_img_btn=WebDriverUtil
//            .getInstance(getSerid())
//            .findElement(By.xpath("//input[@type='file']"));
//       for(String img:imglist){
//        upload_img_btn.sendKeys(img);
//        Thread.currentThread().sleep(5000);
//       }
       WebDriverUtil
       .getInstance(getSerid()).WebDriverWaitElement(
         10*60,
         5,
         By.xpath("//span[contains(@class, 'success') and contains(string(),'上传完成')]"));
       
       WebElement finish_text=WebDriverUtil
            .getInstance(getSerid())
            .findElement(By.xpath("//span[contains(@class, 'success') and contains(string(),'上传完成')]"));
       if(finish_text.isDisplayed()){//上传完成
        Thread.currentThread().sleep(1000);//等待5秒加载完
        WebElement tag_text=WebDriverUtil
                .getInstance(getSerid())
                .findElement(By.xpath("//div[@class='tag-list']/div[2]"));
        if(tag_text!=null){
         tag_text.click();
        }
        WebElement title_text=WebDriverUtil
                .getInstance(getSerid())
                .findElement(By.xpath("//input[contains(@class, 'input-val') and contains(@placeholder, '请输入稿件标题')]"));
           if(title_text!=null){
            title_text.sendKeys(content);
           }
        WebElement desc_text=WebDriverUtil
                .getInstance(getSerid())
                .findElement(By.xpath("//div[contains(@class, 'ql-editor ql-blank') and contains(@placeholder, '填写更全面的相关信息,让更多的人能找到你的视频吧')]"));
           if(desc_text!=null){
               desc_text.sendKeys(content);
           }
           WebElement submit_btn=WebDriverUtil
                .getInstance(getSerid())
                .findElement(By.xpath("//span[contains(@class, 'submit-add') and contains(string(),'立即投稿')]"));
           if(submit_btn!=null){
            Thread.currentThread().sleep(1000);
            submit_btn.click();
            try{
             WebElement success_text=WebDriverUtil
                  .getInstance(getSerid())
                  .findElement(By.xpath("//div[contains(@class, 'step-des') and contains(string(), '稿件投递成功')]"));
             if(success_text!=null){
              resultMap.put("result", "success");
              resultMap.put("msg", "视频上传成功");
             }
            }catch(Exception e){
             resultMap.put("result", "success");
             resultMap.put("msg", "视频上传成功");
            }
           }
       }
      }
     }catch(Exception e){
      e.printStackTrace();
      resultMap.put("result", "error");
   resultMap.put("msg", "视频上传异常,检测是否视频文件");
     }
     return resultMap;
    }
    /**
     * 截取视频第六帧的图片
     *
     * @param filePath 视频路径
     * @param dir      文件存放的根目录
     * @return 图片的相对路径 例:pic/1.png
     */
    public  List<String> videoImage(String filePath,int width,int height) throws Exception {
        String pngPath = "";
//        FFmpegFrameGrabber ff = FFmpegFrameGrabber.createDefault(filePath);
        File file=new File(filePath);
        String dir=file.getParent()+"/";
        System.out.println(dir);
        FFmpegFrameGrabber ff = FFmpegFrameGrabber.createDefault(file);
        List<String> list=new ArrayList<String>();
        ff.start();
        int ffLength = ff.getLengthInFrames();
        Frame f;
        int i = 0;
        int m=0;
        while (i < ffLength) {
            f = ff.grabFrame();
            //截取第6帧
            if ((i > 5&&i%20==0) && (f.image != null)) {
                //生成图片的相对路径 例如:pic/uuid.png
                pngPath = getPngPath();
                //执行截图并放入指定位置
                System.out.println("存储图片 : " + (dir + pngPath));
                doExecuteFrame(f, dir + pngPath,width,height);
                list.add(dir + pngPath);
                m++;
                if(m>4)break;
            }
            i++;
        }
        ff.stop();

        return list;
    }

    /**
     * 生成图片的相对路径
     *
     * @return 图片的相对路径 例:pic/1.png
     */
    private static String getPngPath() {
        return getUUID() + ".png";
    }


    /**
     * 生成唯一的uuid
     *
     * @return uuid
     */
    private static String getUUID() {
        return UUID.randomUUID().toString().replace("-", "");
    }


    /**
     * 截取缩略图
     *
     * @param f                       Frame
     * @param targerFilePath:封面图片存放路径
     */
    private static void doExecuteFrame(Frame f, String targerFilePath,int width,int height) {
        String imagemat = "png";
        if (null == f || null == f.image) {
            return;
        }
        Java2DFrameConverter converter = new Java2DFrameConverter();
        BufferedImage bi = converter.getBufferedImage(f);
        BufferedImage thumbimg = new BufferedImage(width, height, BufferedImage.TYPE_3BYTE_BGR);
        thumbimg.getGraphics().drawImage(bi.getScaledInstance(width, height, Image.SCALE_SMOOTH), 0, 0, null);
        File output = new File(targerFilePath);
        try {
            ImageIO.write(thumbimg, imagemat, output);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}



难点

  • 适配各种视频平台

后端需要利用selenium模拟各种平台的视频上传操作,这块工作量是最大的,另外如果平台升级了,可能会影响开发的接口功能,需要跟着调整对应平台的功能接口,遇要验证还需要利用打码平台(机器验证码)

AI   训练集

图片

下图是核心后端API,下面是利用Apifox的免费功能,输出的API文档

图片

私活合作加VX:itwang007

参考文档:

  1. 某打码平台

    http://www.kuaishibie.cn/?spm=null

  2. 免费的分布式的自动化测试工具 

    http://www.selenium.org.cn/

  3. uniapp

    https://uniapp.dcloud.net.cn/

  4. APIFOX

    https://www.apifox.cn/

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

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

相关文章

【网站项目】自助购药小程序

&#x1f64a;作者简介&#xff1a;拥有多年开发工作经验&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。&#x1f339;赠送计算机毕业设计600个选题excel文件&#xff0c;帮助大学选题。赠送开题报告模板&#xff…

VSCODE目录树缩进调整

VSCode默认的缩进太小了&#xff0c;简直看不出来&#xff0c;很容易弄混目录。在设置里修改就行了。 修改后效果&#xff1a;

代码随想录算法训练营第48天|198.打家劫舍|213.打家劫舍II| 337.打家劫舍III

代码随想录算法训练营第48天|198.打家劫舍|213.打家劫舍II| 337.打家劫舍III 今天就是打家劫舍的一天&#xff0c;这个系列不算难&#xff0c;大家可以一口气拿下。 198.打家劫舍 视频讲解&#xff1a;https://www.bilibili.com/video/BV1Te411N7SX https://programmercarl.c…

合并主分支到子分支

参考&#xff1a;【Git】合并分支出现 Please enter a commit message to explain why this merge is necessary.-CSDN博客 git 如何将主分支(master)合并到子分支上_git 将主分支合并到子分支-CSDN博客 1、先切换到主分支master git checkout master 2、把主分支代码拉到本地…

T-Mamba:用于牙齿 3D CBCT 分割的频率增强门控长程依赖性

T-Mamba&#xff1a;用于牙齿 3D CBCT 分割的频率增强门控长程依赖性 摘要Introduction方法T-Mamba architectureTim block T-Mamba: Frequency-Enhanced Gated Long-Range Dependendcy for Tooth 3D CBCT Segmentation 摘要 三维成像中的高效牙齿分割对于正畸诊断至关重要&am…

机器学习 —— 数据分析与图表绘制

本文使用工具 Anaconda下载安装与使用 Jupyter Notebook的使用 本文使用数据集 机器学习实验所需内容.zip 以朝阳医院2018年销售数据为例&#xff0c;目的是了解朝阳医院在2018年里的销售情况&#xff0c;这就需要知道几个业务指标&#xff0c;本次的分析…

Netty学习——源码篇13 命中缓存的分配

上一篇分析了DirectArena内存分配大小的大概流程(Netty池化内存管理机制),知道了其先命中缓冲&#xff0c;如果没有命中&#xff0c;再去分配一款连续内存。现在分析命中缓存的相关逻辑。前面说到PoolThreadCache中维护了三个缓存数组(实际上是6个&#xff0c;这里仅以Direct为…

matlab/simulink仿真全合集---电力电子的simulink仿真

simulink仿真新手大礼包&#xff0c;共整理了9份simulink仿真模型&#xff0c;每一份都是完美运行&#xff0c;适合电气工程专业/电力电子专业的新手学习。 1、Boost电路 simulink 仿真&#xff0c;boost 电路模块搭建和用传递函数进行验证&#xff0c; 电流开环控制 、电流闭…

基于Spring boot+Vue的业余排球俱乐部会员管理系统

5 系统功能模块的具体实现 5.1超级会员角色 5.1.1 登录 超级管理员登录通过用户名和密码去数据库查询用户表&#xff0c;该名称是否在用户表中存在&#xff0c;如果存在&#xff0c;则通过用户名和密码查询密码是否正确&#xff0c;然后吧用户的信息存在jwt的负载里&#xf…

C语言 | Leetcode C语言题解之第14题最长公共前缀

题目&#xff1a; 题解&#xff1a; char* longestCommonPrefix(char** strs, int strsSize) {if (strsSize 0) {return ""; } for (int i 0; i < strlen(strs[0]); i) {for (int j 1; j < strsSize; j) {if (strs[0][i] ! strs[j][i]){strs[0][i] \0;ret…

ics-05-攻防世界

题目 点了半天只有设备维护中心能进去 御剑扫一下 找到一个css 没什么用 再点击云平台设备维护中心url发生了变化 设备维护中心http://61.147.171.105:65103/index.php?pageindex试一下php伪协议 php://filter/readconvert.base64-encode/resourceindex.php base64解一下…

【二分查找】Leetcode 山脉数组的峰顶索引

题目解析 852. 山脉数组的峰顶索引 这到题使用暴力枚举的查找方法发现这段数组是有二段性的&#xff0c;峰顶左边的一段区间是一段递增区间&#xff0c;右边的一段区间是一段递减区间 算法讲解 class Solution { public:int peakIndexInMountainArray(vector<int>&am…

vue快速入门(十二)v-key索引标志

注释很详细&#xff0c;直接上代码 新增内容 v-key的使用场景数组筛选器的使用 源码 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-…

漫谈:“标准”是一种幻觉 C++语言标准的意义

初级代码游戏的专栏介绍与文章目录-CSDN博客 我的github&#xff1a;codetoys&#xff0c;所有代码都将会位于ctfc库中。已经放入库中我会指出在库中的位置。 这些代码大部分以Linux为目标但部分代码是纯C的&#xff0c;可以在任何平台上使用。 “标准”这个词很迷惑&#xf…

1688详情API接口:解锁多元化应用场景java php c++

随着互联网的快速发展&#xff0c;数据交换和信息共享已成为企业日常运营不可或缺的一部分。在这样的背景下&#xff0c;API&#xff08;应用程序接口&#xff09;接口作为实现数据互通的重要工具&#xff0c;受到了越来越多企业的青睐。1688详情API接口作为阿里巴巴旗下的重要…

黑盒测试—错误推测法

上一篇文章介绍了取款业务的场景测试法&#xff0c;在这里继续用上次的场景&#xff0c;对银行的ATM机进行存款&#xff0c;错误推测法算是对场景测试法的补充&#xff0c;错误推测法通常是根据经验来推测可能产生的结果&#xff0c;由原因推测结果。 上一篇文章地址&#xff…

分布式 SpringCloudAlibaba、Feign与RabbitMQ实现MySQL到ES数据同步

文章目录 ⛄引言一、思路分析⛅实现方式⚡框架选择 二、实现数据同步⌚需求分析⏰搭建环境⚡核心源码 三、测试四、源码获取⛵小结 ⛄引言 本文参考黑马 分布式Elastic search Elasticsearch是一款非常强大的开源搜索引擎&#xff0c;具备非常多强大功能&#xff0c;可以帮助…

python(使用循环显示四种模式)

代码&#xff1a; # 模式A for i in range(1, 6):for j in range(1, 6):if i j:print(i, end"")else:print(" ", end"")print()# 模式B for i in range(1, 6):for j in range(1, 6):if i j 7:print(j, end"")else:print(" &q…

Java常用API_正则表达式_检验字符串是否满足规则——基础使用方法及综合练习

正则表达式可以校验字符串是否满足一定的规则&#xff0c;并用来校验数据格式的合法性。 简单举例&#xff1a; 校验一个qq号是否符合要求 要求&#xff1a;6位到20位之内&#xff0c;不能以0开头&#xff0c;必须全是数字 代码演示&#xff1a; public class Test1 {public…

FreeRTOS移植到标准库

源码下载 1&#xff1a;从官网获取freeRTOS源码 freeRTOS官网 2&#xff1a;FreeRtos源码文件阐述 3&#xff1a;移植FreeRtos源码 FreeRTOS移植步骤1&#xff1a;添加FreeRTOS源码&#xff0c;将FreeRTOS源码添加到基础工程&#xff0c;头文件等路径2&#xff1a;添加FreeR…