React实现文本框输入文字内容动态给图片添加文字信息(多个)并生成新的图片

news2025/1/20 13:34:04

文章目录

      • 思路一
      • 思路二
        • 1. 下载html2canvas依赖包
        • 2. 搭建页面,并且创建新增节点的区域
        • 3. 初始化新增第一个节点到页面中的某个指定模块
        • 4. 当文本框发生变动,修改节点信息
        • 5. 实现节点删除
        • 6. 利用html2canvas将模块生成canvas,然后转化成图片
      • 完整代码

收到这个需求的时候,我的内心是崩溃的,脑子里已经跑过一万匹草泥马,内心想这种事为啥不交给ps去做,哪怕是手机里图片编辑也可以做到吧,专业的事交给专业的工具去干不就好了,何必出这种XX需求。后来想想就释然了,反正拿钱干活,干啥不是干,只要给钱,再XX的需求我也给你写出来。废话不多说,请看下方示例

实现效果:
在这里插入图片描述
本文案例只负责下图所示组件,上图中的图片列表只是为了方便展示
在这里插入图片描述

注:本组件使用的是React + AntD,这里样式就不过多描写了

思路一

首先,先说一开始错误的思路一,给图片添加文字信息,我一开始想到的是用canvas。先创建画布,获取要写入文字的图片宽高->再创建一个新的图片->然后再画布中画出刚才创建的新的图片->输入框文字发生改变则,获取输入框中输入的文字相关信息->在画布中绘制文字->生成新的图片渲染到页面。
但是最后该思路以失败告终,这条思路只适用于只添加一条文字信息的情况,添加第二条后就会出现添加的文字错乱的情况,因为新增代码后我无法保存上条图片的情况,无法知道上条图片的文字新增后的状态,如果是要删除情况又当如何,失败代码如下,如果只添加一条文字信息,可以用来参考,各位慢慢研究

import React from 'react'
import { Button } from 'antd'
import ParamsOptionComp from './ParamsOptionComp'

export default class MaterialEdit extends React.Component {
  canvas = document.createElement("canvas");
  image = React.createRef();

  state = {
    imgSrc: this.props.src,
    paramAttribute: [{
      content: '文字内容',
      color: '#000000',
      size: 24,
      left: 130,
      top: 50
    }],
    srcList: {
      0: this.props.src
    }
  }

  componentDidMount() {
    this.setState({
      paramAttribute: [{
        content: '文字内容',
        color: '#000000',
        size: 24,
        left: (this.image.current.clientWidth / 2) - 48,
        top: (this.image.current.clientHeight / 2) + 12
      }]
    }, () => {
      this.drawImage()
    })
    this.drawImage()
  }

  // 绘制图片
  drawImage = (params, index) => {
    // 创建画布
    const ctx = this.canvas.getContext("2d");
    // 获取图片大小
    const imageRect = this.image.current.getBoundingClientRect();

    const { width, height } = imageRect;
    this.canvas.width = width;
    this.canvas.height = height;

    // 创建图片
    const image = new Image();

    // if (index) {
    //   console.log(this.state.imgSrc, 123)
    //   image.src = this.state.imgSrc;
    // } else {
      image.src = this.props.src;
    // }

    image.onload = () => {
      ctx.drawImage(image, 0, 0, width, height);
      // 绘制文字
      this.drawText(params ? params : this.state.paramAttribute[0]);
    }

  }

  // 绘制文字
  drawText = ({ content, color, size, left, top }) => {
    // console.log(content, color, size, left, top)
    const ctx = this.canvas.getContext("2d");
    ctx.font = `${size}px Arial`;
    ctx.fillStyle = color;
    ctx.fillText(content, left, top);
    this.saveImage()
  };

  // 当文字发生变化
  onValuesChange = (changedValues, allValues) => {
    // console.log(changedValues, allValues);

    allValues.paramAttribute.forEach((item, index) => {
      // if (item && Object.keys(item).length > 0) {
        this.drawImage(item)
      // } else if (index >= 1) {
      //   this.drawImage(item, 'index')
      // }
    })
  }

  setSrcList = (index) => {
    console.log(index)
  }

  // 保存图片
  saveImage = () => {
    const image = this.canvas.toDataURL('image/png');
    // console.log(image);
    this.setState({ imgSrc: image });
    // 根据需要,将生成的图片显示给用户或保存为新的图片文件
    // 如:window.open(image);
  };

  // 将图片转成二进制格式
  base64ToBlob = (code) => {
    let parts = code.split(';base64,')
    let contentType = parts[0].split(':')[1]
    let raw = window.atob(parts[1])
    let rawLength = raw.length
    let uint8Array = new Uint8Array(rawLength)
    for (let i = 0; i < rawLength; i++) {
      uint8Array[i] = raw.charCodeAt(i)
    }
    return new Blob([uint8Array], { type: contentType })
  }

  // 下载图片
  download = (dataUrl, fileName) => {
    let aLink = document.createElement('a')
    let blob = this.base64ToBlob(dataUrl)
    let event = document.createEvent('HTMLEvents')
    event.initEvent('click', true, true)
    aLink.download = fileName + '.' + blob.type.split('/')[1]
    aLink.href = URL.createObjectURL(blob)
    aLink.click()
  }

  render() {
    const { imgSrc } = this.state
    const { src } = this.props
    return <>
      <div style={{ textAlign: 'center', marginBottom: '30px' }}>
        <img src={imgSrc} width={300} ref={this.image} />
      </div>
      <ParamsOptionComp onValuesChange={this.onValuesChange} paramAttribute={this.state.paramAttribute} setSrcList={this.setSrcList}/>
      <Button type="primary" htmlType="submit">
        生成新的图片
      </Button>
    </>
  }
}

思路二

同样也是与canvas相关,不过这次想到的是,将输入的文字信息生成节点添加到页面中的某个模块,然后再将这个模块输出成canvas然后再转化成图片下载下来,这里会用到html2canvas这个库,这个库可以让html页面和canvas相互转换,大致思路如下:

下载html2canvas依赖包->搭建页面,并创建新增节点的区域->初始化新增第一个节点到页面中的某个指定模块->当文本框发生变动,修改节点信息->实现节点删除->利用html2canvas将模块生成canvas,然后转化成图片

1. 下载html2canvas依赖包
npm i html2canvas
2. 搭建页面,并且创建新增节点的区域

MaterialEditComp.jsx

import React, { useEffect, useRef, useState } from 'react'
import { Button } from 'antd'
import ParamsOptionComp from './ParamsOptionComp'
import html2canvas from 'html2canvas';
import './index.less'

export default function MaterialEditComp(props) {
  const { src, getNewImage } = props;
  const [imgSrc, setImgSrc] = React.useState(src);
  const contentRef = useRef(null);
  const [imageSize, setImageSize] = useState({ width: 0, height: 0 });

  const onValuesChange = (changedValues, allValues) => {}

  // 新增节点
  const AddNode = ({content, color, size, left, top }, index) => {}

  //  删除节点
  const removeNode = (index) => {}
  
  // 将图片转换成二进制形式
  const base64ToBlob = (code) => {}

  // 保存图片
  const saveImage = async () => {}


  useEffect(() => {
  	// 坑一
    const img = new Image();
    img.src = imgSrc

    img.onload = () => {
      setImageSize({ width: img.width, height: img.height });
    };
  }, []);


  return (
    <>
      // 新增节点区域
        <div ref={contentRef} style={{
        background: `url(${imgSrc}) no-repeat`,
        backgroundSize: 'cover',
        backgroundPosition: 'center',
        width: imageSize.width + 'px',
        height: imageSize.height + 'px',
        marginBottom: '30px'
      }} className="content">
        
      </div>
      // 输入框组件
      <ParamsOptionComp onValuesChange={onValuesChange} imageSize={imageSize} removeNode={removeNode} />
      <Button type="primary" htmlType="submit" onClick={saveImage} className='btn'>
        生成新的图片
      </Button>
    </>
  )
}

ParamsOptionComp.jsx

import React, { useEffect, useState } from 'react'
import { Input, Form, Space, Button, InputNumber } from 'antd'
import { PlusOutlined, MinusCircleOutlined, DragOutlined } from '@ant-design/icons'

export default function ParamsOptionComp(props) {
  const { onValuesChange, imageSize, removeNode } = props
  const [count, setCount] = useState(0)
  const [form] = Form.useForm();

  // 坑一
  useEffect(() => {
    form.resetFields()
  }, [imageSize])


  return <Form form={form} name="dynamic_form_nest_item"
    // 坑一
    initialValues={{
      paramAttribute: [{
        left: imageSize.width / 2 - 48,
        top: imageSize.height / 2 - 24,
        content: '文字内容'
      }]
    }}
    onValuesChange={onValuesChange} >

    <Form.List name="paramAttribute">
      {(fields, { add, remove, move }) => (
        <>
          {fields.map(({ key, name, ...restField }, index) => (
            <Space key={key} style={{ display: 'flex', marginBottom: 0 }} align="baseline">
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'content']}
                label="文字内容"
                rules={[{ required: true, message: '请输入文字内容' }]}
              >
                <Input placeholder="文字内容" maxLength={50} />
              </Form.Item>
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'color']}
                label="文字颜色"
                rules={[{ required: true, message: '请输入文字颜色' }]}
                initialValue="#000000"
              >
                <Input placeholder="文字颜色" type="color" style={{ width: '80px' }} />
              </Form.Item>
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'size']}
                label="文字大小"
                rules={[{ required: true, message: '请输入文字大小' }]}
                initialValue="24"
              >
                <InputNumber placeholder="文字大小" min={12} value={13} />
              </Form.Item>
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'top']}
                label="上边距"
                rules={[{ required: true, message: '请输入上边距' }]}
                initialValue={imageSize.height / 2 - 24}
              >
                <InputNumber placeholder="上边距" value={13} />
              </Form.Item>
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'left']}
                label="左边距"
                rules={[{ required: true, message: '请输入左边距' }]}
                initialValue={imageSize.width / 2 - 48}
              >
                <InputNumber placeholder="左边距" value={13} />
              </Form.Item>
              <MinusCircleOutlined onClick={() => {
                if (count === 0) {
                  return
                }
                remove(name)
                removeNode(index)
                setCount(count => count - 1);
              }} />
            </Space>
          ))}
          <Form.Item>
            <Button type="dashed" onClick={async () => {
              try {
                const values = await form.validateFields()
                add();
                setCount(count => count + 1);
              } catch (errorInfo) {
                return;
              }
            }} block icon={<PlusOutlined />}>添加选项</Button>
          </Form.Item>
        </>
      )}
    </Form.List>
  </Form>
}

此时页面如下
在这里插入图片描述
上述代码中有个坑

  • 因为产品要求,需要给个初始化数据,并且文字要在图片内水平垂直居中但是初始化数据时是无法获取到图片的宽高的,并且初始化时在useEffect中也无法通过ref获取图片的大小,此时只能将接收到的图片生成新的图片,然后读取新图片的宽高赋值给ImageSize,此时方可获取到图片真正的宽高,然后再传递给ParamsOptionComp组件

    MaterialEditComp.jsx

    const [imageSize, setImageSize] = useState({ width: 0, height: 0 });
    
    useEffect(() => {
        const img = new Image();
        img.src = imgSrc
    
        img.onload = () => {
          setImageSize({ width: img.width, height: img.height });
    	};
    }, []);
    <ParamsOptionComp onValuesChange={onValuesChange} imageSize={imageSize} removeNode=		 {removeNode} />
    

    ParamsOptionComp.jsx

    下方初始化第一条数据时,因为initialValue就是所谓的defaultValue,只会在第一次赋值的时候改变,无法直接设置initialValue的值来改变,所以获取到的也是第一次初始化的宽高都为0,但是我们 可以通过Form的resetFields()方法来解决这个问题,当监听到imageSize发生变化时我们可以调用resetFields()来重新设置initialValue的值。
    在这里插入图片描述

    useEffect(() => {
      form.resetFields()
    }, [imageSize])
    
3. 初始化新增第一个节点到页面中的某个指定模块

此时输入框中已经有值了,但是此处图片中还没有初始化值,此时,需要在useEffect中调用AddNode初始化第一个节点的值。

const AddNode = ({ content, color, size, left, top }, index) => {
    const contentNode = contentRef.current;
    let newNode = document.createElement('div');
    newNode.className = 'node' + index;
    // 此处判断节点是否已经存在
    const bool = contentNode?.childNodes[index]
    if (bool) {
      newNode = contentNode.childNodes[index]
    }

    newNode.textContent = content
    newNode.style.color = color;
    newNode.style.fontSize = size + 'px';
    newNode.style.top = top + 'px';
    newNode.style.left = left + 'px';
    newNode.style.position = 'absolute';

    // 节点不存在新增阶段
    if (!bool) {
      contentNode.appendChild(newNode);
    } else {
      // 节点存在则替换原来的节点
      contentNode.replaceChild(newNode, contentNode.childNodes[index])
    }
  }
  
   useEffect(() => {
    const img = new Image();
    img.src = imgSrc

    img.onload = () => {
      setImageSize({ width: img.width, height: img.height });
      AddNode({
        content: '文字内容',
        color: '#000000',
        size: 24,
        left: img.width / 2 - 48,
        top: img.height / 2 - 24
      }, 0);
    };
  }, []);
4. 当文本框发生变动,修改节点信息

当文本框发生变动通过表单的onValuesChange 进行监听,遍历表单中的数据新增节点信息

const onValuesChange = (changedValues, allValues) => {
    // index标记当前是第几个节点
    allValues.paramAttribute.forEach((item, index) => {
      item && AddNode(item, index)
    })
  }
5. 实现节点删除

当进行节点删除时,调用传递给ParamsOptionComp组件的removeNode方法获取到删除的节点

  //  删除节点
  const removeNode = (index) => {
    const contentNode = contentRef.current;
    const bool = contentNode?.childNodes[index]
    if (bool) {
      contentNode.removeChild(contentNode.childNodes[index])
    }
  }
6. 利用html2canvas将模块生成canvas,然后转化成图片

此时需要利用html2canvas将模块生成canvas,然后转化成图片,如果需要调用接口将图片保存下来,此处还需将图片转换成二进制,如果不需要则直接下载就好

  // 将图片转换成二进制形式
  const base64ToBlob = (code) => {
    let parts = code.split(';base64,')
    let contentType = parts[0].split(':')[1]
    let raw = window.atob(parts[1])
    let rawLength = raw.length
    let uint8Array = new Uint8Array(rawLength)
    for (let i = 0; i < rawLength; i++) {
      uint8Array[i] = raw.charCodeAt(i)
    }
    return new Blob([uint8Array], { type: contentType })
  }

  // 保存图片
  const saveImage = async () => {
    const contentNode = contentRef.current;
    const canvas = await html2canvas(contentNode, {
      useCORS: true,
      allowTaint: true,//允许污染
      backgroundColor: '#ffffff',
      // toDataURL: src
    })

    const imgData = canvas.toDataURL('image/png');
    let blob = base64ToBlob(imgData)
    const link = document.createElement('a');
    link.href = imgData;
    // link.href = URL.createObjectURL(blob);
    getNewImage(link.href)
    // console.log(blob, 11)
    link.download = 'page-image.' + blob.type.split('/')[1];
    link.click();
  }

注意:此处使用html2canvas时, 必须配置以下信息,否则图片信息无法进行转换

{
      useCORS: true,
      allowTaint: true,//允许污染
      backgroundColor: '#ffffff',
      // toDataURL: src
    }

完整代码

MaterialEditComp.jsx

import React, { useEffect, useRef, useState } from 'react'
import { Button } from 'antd'
import ParamsOptionComp from './ParamsOptionComp'
import html2canvas from 'html2canvas';
import './index.less'

export default function MaterialEditComp(props) {
  const { src, getNewImage } = props;
  const [imgSrc, setImgSrc] = React.useState(src);
  const contentRef = useRef(null);
  const [imageSize, setImageSize] = useState({ width: 0, height: 0 });

/* 
  const divStyle = {
    background: `url(${imgSrc}) no-repeat`,
    backgroundSize: 'cover',
    backgroundPosition: 'center',
    width: imageSize.width + 'px',
    height: imageSize.height + 'px',
    marginBottom: '30px'
  }; */

  const onValuesChange = (changedValues, allValues) => {
    // console.log(changedValues, allValues, 11)
    allValues.paramAttribute.forEach((item, index) => {
      item && AddNode(item, index)
    })
  }

  // 新增节点
  const AddNode = ({ content, color, size, left, top }, index) => {
    const contentNode = contentRef.current;
    let newNode = document.createElement('div');
    newNode.className = 'node' + index;
    // 此处判断节点是否已经存在
    const bool = contentNode?.childNodes[index]
    if (bool) {
      newNode = contentNode.childNodes[index]
    }

    newNode.textContent = content
    newNode.style.color = color;
    newNode.style.fontSize = size + 'px';
    newNode.style.top = top + 'px';
    newNode.style.left = left + 'px';
    newNode.style.position = 'absolute';

    // 节点不存在新增阶段
    if (!bool) {
      contentNode.appendChild(newNode);
    } else {
      // 节点存在则替换原来的节点
      contentNode.replaceChild(newNode, contentNode.childNodes[index])
    }
  }

  //  删除节点
  const removeNode = (index) => {
    const contentNode = contentRef.current;
    const bool = contentNode?.childNodes[index]
    if (bool) {
      contentNode.removeChild(contentNode.childNodes[index])
    }
  }
  // 将图片转换成二进制形式
  const base64ToBlob = (code) => {
    let parts = code.split(';base64,')
    let contentType = parts[0].split(':')[1]
    let raw = window.atob(parts[1])
    let rawLength = raw.length
    let uint8Array = new Uint8Array(rawLength)
    for (let i = 0; i < rawLength; i++) {
      uint8Array[i] = raw.charCodeAt(i)
    }
    return new Blob([uint8Array], { type: contentType })
  }

  // 保存图片
  const saveImage = async () => {
    const contentNode = contentRef.current;
    const canvas = await html2canvas(contentNode, {
      useCORS: true,
      allowTaint: true,//允许污染
      backgroundColor: '#ffffff',
      // toDataURL: src
    })

    const imgData = canvas.toDataURL('image/png');
    let blob = base64ToBlob(imgData)
    const link = document.createElement('a');
    link.href = imgData;
    // link.href = URL.createObjectURL(blob);
    getNewImage(link.href)
    // console.log(blob, 11)
    link.download = 'page-image.' + blob.type.split('/')[1];
    link.click();
  }


  useEffect(() => {
    const img = new Image();
    img.src = imgSrc

    img.onload = () => {
      setImageSize({ width: img.width, height: img.height });
      AddNode({
        content: '文字内容',
        color: '#000000',
        size: 24,
        left: img.width / 2 - 48,
        top: img.height / 2 - 24
      }, 0);
    };
  }, []);


  return (
    <>
      <div ref={contentRef} style={{
        background: `url(${imgSrc}) no-repeat`,
        backgroundSize: 'cover',
        backgroundPosition: 'center',
        width: imageSize.width + 'px',
        height: imageSize.height + 'px',
        marginBottom: '30px'
      }} className="content">

      </div>
      <ParamsOptionComp onValuesChange={onValuesChange} imageSize={imageSize} removeNode={removeNode} />
      <Button type="primary" htmlType="submit" onClick={saveImage} className='btn'>
        生成新的图片
      </Button>
    </>
  )
}

ParamsOptionComp.jsx

import React, { useEffect, useState } from 'react'
import { Input, Form, Space, Button, InputNumber } from 'antd'
import { PlusOutlined, MinusCircleOutlined, DragOutlined } from '@ant-design/icons'

export default function ParamsOptionComp(props) {
  const { onValuesChange, imageSize, removeNode } = props
  const [count, setCount] = useState(0)
  const [form] = Form.useForm();

  // 坑
  useEffect(() => {
    form.resetFields()
  }, [imageSize])


  return <Form form={form} name="dynamic_form_nest_item"
    // 坑
    initialValues={{
      paramAttribute: [{
        left: imageSize.width / 2 - 48,
        top: imageSize.height / 2 - 24,
        content: '文字内容'
      }]
    }}
    onValuesChange={onValuesChange} >

    <Form.List name="paramAttribute">
      {(fields, { add, remove, move }) => (
        <>
          {fields.map(({ key, name, ...restField }, index) => (
            <Space key={key} style={{ display: 'flex', marginBottom: 0 }} align="baseline">
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'content']}
                label="文字内容"
                rules={[{ required: true, message: '请输入文字内容' }]}
              >
                <Input placeholder="文字内容" maxLength={50} />
              </Form.Item>
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'color']}
                label="文字颜色"
                rules={[{ required: true, message: '请输入文字颜色' }]}
                initialValue="#000000"
              >
                <Input placeholder="文字颜色" type="color" style={{ width: '80px' }} />
              </Form.Item>
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'size']}
                label="文字大小"
                rules={[{ required: true, message: '请输入文字大小' }]}
                initialValue="24"
              >
                <InputNumber placeholder="文字大小" min={12} value={13} />
              </Form.Item>
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'top']}
                label="上边距"
                rules={[{ required: true, message: '请输入上边距' }]}
                initialValue={imageSize.height / 2 - 24}
              >
                <InputNumber placeholder="上边距" value={13} />
              </Form.Item>
              <Form.Item
                style={{ marginBottom: '10px' }}
                {...restField}
                name={[name, 'left']}
                label="左边距"
                rules={[{ required: true, message: '请输入左边距' }]}
                initialValue={imageSize.width / 2 - 48}
              >
                <InputNumber placeholder="左边距" value={13} />
              </Form.Item>
              <MinusCircleOutlined onClick={() => {
                if (count === 0) {
                  return
                }
                remove(name)
                removeNode(index)
                setCount(count => count - 1);
              }} />
            </Space>
          ))}
          <Form.Item>
            <Button type="dashed" onClick={async () => {
              try {
                const values = await form.validateFields()
                add();
                setCount(count => count + 1);
              } catch (errorInfo) {
                return;
              }
            }} block icon={<PlusOutlined />}>添加选项</Button>
          </Form.Item>
        </>
      )}
    </Form.List>
  </Form>
}

使用:

const src=''
const getNewImage = (image) => {
    console.log(image)
    page.close()
 }
return <MaterialEditComp src={src}  getNewImage={getNewImage} />

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

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

相关文章

Leetcode刷题详解——汉诺塔问题

1. 题目链接&#xff1a;面试题 08.06. 汉诺塔问题 2. 题目描述&#xff1a; 在经典汉诺塔问题中&#xff0c;有 3 根柱子及 N 个不同大小的穿孔圆盘&#xff0c;盘子可以滑入任意一根柱子。一开始&#xff0c;所有盘子自上而下按升序依次套在第一根柱子上(即每一个盘子只能放…

【Web】TCP 和 UCP 的含义和区别

文章目录 一、两者含义二、两者区别 一、两者含义 TCP/IP 协议组为传输层指明了两个协议&#xff1a;TCP 和 UDP&#xff0c;他们都是作为应用程序和网络操作的中介物 TCP &#xff08;传输控制协议&#xff09;&#xff1a;通过三次握手建立可靠的连接&#xff0c;发送端将数据…

Excel自学三部曲_Part3:Excel工作场景实战(二)

文章目录 二、基础概念、表格结构与常用函数1. 业务背景、字段含义2. 筛选、排序、冻结窗格3. 状态栏数据提示、调整数据显示格式4. 公式、引用、溢出5. 连接和提取函数、时间函数、IF和IFS函数、SUMIF和SUMIFS函数&#xff08;1&#xff09;每个业务组的成交额有多少&#xff…

解决使用IDEA启动SpringBoot项目报错 java: 警告: 源发行版 17 需要目标发行版 17 或者 java: 无效的目标发行版: 17

问题描述 今天新建了个SpringBoot项目&#xff0c;在启动的时候报错如下&#xff1a; java: 警告: 源发行版 17 需要目标发行版 17 分析问题 其实错误已经很明显了&#xff0c;由于我本地只有JDK8的环境&#xff0c;但是项目以及编译器的JDK是17&#xff0c;这时候把JDK版本修…

提示3D标题编辑器仍在运行如何解决 3D标题编辑器怎么使用

品牌型号&#xff1a;联想GeekPro 2020 系统&#xff1a;Windows 10 64位专业版 软件版本&#xff1a;会声会影2023旗舰版 3D标题因其独特的表现形式和多变的画面效果&#xff0c;被广泛应用于节目片头、宣传片、开幕式等诸多场景之中。掌握3D标题的使用技巧&#xff0c;能够…

SecureCRT 手动全部Tab窗口重连

在工作了一天后&#xff0c;到第二天上班的时候&#xff0c;前一天连接的服务会断开&#xff0c;因为公司内部网络自动断开了&#xff0c;所以一个个重新连接和登录 &#xff0c;用脚本轻松搞定。 # $language "VBScript" # $interface "1.0"Dim g_objTa…

数组基础知识三

二分查找法&#xff1a;也叫折半查找算法。二分查找针对的是一个有序的数据集合&#xff0c;每次都通过跟区间的中间元素对比&#xff0c;将待查找的区间缩小为之前的一半&#xff0c;直到找到要查找的元素&#xff0c;或者区间被缩小为 0。 #include <stdio.h>int main…

干了3年“点点点”,我废了...

简单概括一下 先说一下自己的情况&#xff0c;普通本科&#xff0c;18年通过校招进入深圳某软件公司&#xff0c;干了3年多的功能测试&#xff0c;21年的那会&#xff0c;因为大环境不好&#xff0c;我整个人心惊胆战的&#xff0c;怕自己卷铺盖走人了&#xff0c;我感觉自己不…

day53【子序列】1143.最长公共子序列 1035.不相交的线 53.最大子序和

文章目录 1143. 最长公共子序列1035.不相交的线53. 最大子序和 1143. 最长公共子序列 题目链接&#xff1a;力扣链接 讲解链接&#xff1a;代码随想录讲解 题意&#xff1a;给定两个字符串 text1 和 text2&#xff0c;返回这两个字符串的最长 公共子序列 的长度。如果不存在 …

C++并发编程实战——05.内存模型与原子操作

文章目录 内存模型与原子操作内存模型原子操作和原子类型标准原子类型std::atomic_flagstd::atomic\<bool>std::atomic<T\*>std::atomic<user_define_type> 类模板非成员函数 同步操作和强制排序同步发生与先行发生内存序**顺序一致性**(memory_order_seq_cs…

Java选择与循环

1.选择 前言&#xff1a;什么是选择呢&#xff1f;在我们的人生中处处面临着选择&#xff0c;比如说在学校你可以选择玩&#xff0c;摆烂&#xff0c;当然也可以选择努力写代码&#xff0c;刷题。什么样的选择就会面临什么样的结果。 其实程序和人生一样&#xff1a;顺序中夹杂…

intellij idea拉取最新的依赖包

intellij idea setting 拉取最新的依赖包 File --> Settings --> Build, Execution, Deployment --> Build Tools --> Maven --> 勾选 Always update snapshots. 一般情况下&#xff0c;设置完就可以拉取到最新的依赖包了。 如下&#xff1a; 安装最新的依赖…

计算机视觉基础——基于yolov5-face算法的车牌检测

文章目录 车牌检测算法检测实现1.环境布置2.数据处理2.1 CCPD数据集介绍2.1.1 ccpd2019及20202.1.2 文件名字解析 2.2数据集处理2.2.1 CCPD数据处理2.2.2 CPRD数据集处理 2.3 检测算法2.3.1 数据配置car_plate.yaml2.3.2 模型配置2.3.3 train.py2.3.4 训练结果 2.4 部署2.4.1 p…

activiti的核心对象与API与对应操作的表与核心字段

【Deployment】 &#xff08;创建并部署一个新的流程定义&#xff09; 获取方式&#xff1a; repositoryService.createDeployment().deploy(); 对应的表&#xff1a;act_re_deployment 用于存储流程部署的相关信息。该表记录了每个流程部署的唯一标识符&#xff08;ID&#x…

【Redis】Redis在Linux与windows上的安装基本操作语法

一、Redis简介 Redis 是完全开源免费的&#xff0c;遵守BSD协议&#xff0c;是一个高性能的key-value数据库。 Redis 与其他 key - value 缓存产品有以下三个特点&#xff1a; Redis支持数据的持久化&#xff0c;可以将内存中的数据保持在磁盘中&#xff0c;重启的时候可以再…

Python条件判断的运用

问题 在生活中&#xff0c;我们可以通过判断条件是否成立&#xff0c;来决定执行哪个分支。选择语句有多种形式&#xff1a;if语句&#xff0c;if-else语句&#xff0c;if-elif-else语句等。 Python使用if条件判断语句来实现条件判断时&#xff0c;可以在多个循环中实现对问题的…

【蓝桥杯 第十四届省赛Java B组】真题训练(A - C)正在更新

目录 A、阶乘求和 - BigInteger B、幸运数字 - 字符串 进制转换 暴力大法 C、数组分割 - A、阶乘求和 - BigInteger 思路&#xff1a; 当时比赛时&#xff0c;拿计算器算的&#xff0c;然后辛辛苦苦也没对 看到这个数肯定很大&#xff0c;而且只求后9位&#xff0c;阶乘越…

三篇文章了解计算机网络(一)

目标 三篇文章&#xff0c;由浅入深理解计算机网络 适宜人群 小小白、无计算机网络基础&#xff0c;非技术人员&#xff0c;网络兴趣爱好者 内容简介 本文不谈技术&#xff0c;不谈理论&#xff0c;那谈什么&#xff1f;--谈生活。。。跑题了&#xff1f; 正文 网络是什…

SECS/GEN HSMS半导体通信协议解析

协议族总体结构 HSMS消息格式&#xff08;网口连接&#xff09; 超时时间设置 T3 回复超时&#xff1a;指发送指令到接收到回复指令的最大时间&#xff1b; T5 连接间隔&#xff1a;指断开连接和重新连接的最小时间&#xff1b; T6 控制指令超时时间&#xff1a;主要指连接选…

本地部署清华大模型 ChatGLM3

ChatGLM 是一个开源的、支持中英双语的对话语言模型&#xff0c;由智谱 AI 和清华大学 KEG 实验室联合发布&#xff0c;基于 General Language Model (GLM) 架构&#xff0c;具有 62 亿参数。ChatGLM3-6B 更是在保留了前两代模型对话流畅、部署门槛低等众多优秀特性的基础上增加…