java-操作excel

news2024/9/27 12:08:29

文章目录

  • java操作Excel数据
    • 使用场景
    • excel 03 和 07的区别
    • POI
    • easyExcel
    • 解析excel表中的对象
    • POI使用步骤
    • POI 写数据
    • POI 读数据
    • 计算公式
    • easyExcel读写数据
      • ·写数据
      • ·读数据

java操作Excel数据

在 平时 可以使用IO流对Excle进行操作

但是现在使用更加方便的第三方组件来实现


使用场景

1、将用户信息导出为Excel表格,导入数据

2、将Excel表中的数据录入到网站数据库 (习题上传) 减轻网站的录入量

3、开发中经常会设计到Excel的处理,导入Excel到数据库中

目前最流行的是 Apache POI以及阿里巴巴easyExcel


excel 03 和 07的区别

HSSF 对应excel中的03版本 该版本要求excel中最多只能写65536行

​ 后缀名为 03.xls

XSSF 对应excel中的07版本 该版本对于行数没有要求

​ 后缀名为 07.xlsx


POI

Apache提供的,会比较麻烦,比较原生

开放源码函式库,POI提供API给java程序对Office格式档案读和写的功能

但是存在内存问题 => POI将数据会先写入内存中,一旦写入的内容过多时会产生OOM,也叫做内存溢出


easyExcel

https://github.com/alibaba/easyexcel

对POI进行了一些优化,可以使开发者更加简单,读和写代码只需要1行

存在时间的问题 => easyExcel在写数据时是一行一行往磁盘中写,所以解决了POI的内存问题,但是带来了时间问题

在这里插入图片描述


解析excel表中的对象

由于java中万物皆对象,所以需要先观察一张excel表中有哪些对象~

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-k6tlr2ha-1673247094594)(C:\Users\86186\AppData\Roaming\Typora\typora-user-images\1671933499863.png)]

1、工作簿

2、工作表

3、行

4、列 => 单元格


POI使用步骤

第一步:创建Maven项目

第二步:导入依赖

<dependencies>
        <!-- xls 03-->
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi</artifactId>
            <version>3.9</version>
        </dependency>
        <!-- xlsx 07-->
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml</artifactId>
            <version>3.9</version>
        </dependency>
        <!-- 日期格式化工具-->
        <dependency>
            <groupId>joda-time</groupId>
            <artifactId>joda-time</artifactId>
            <version>2.10.1</version>
        </dependency>
        <!--测试-->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
            <scope>test</scope>
        </dependency>
</dependencies>

POI 写数据

第一步:基本文件的写入

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-pWAZ7tGi-1673247094595)(C:\Users\86186\AppData\Roaming\Typora\typora-user-images\1671934401128.png)]

private String PATH = "E:\\JavaCode\\Maven\\excel-demo\\src";

03版本测试

@Test
    public void test03() throws Exception {
        // 1 创建工作簿
        Workbook workbook = new HSSFWorkbook();
        // 2 创建工作表
        Sheet sheet1 = workbook.createSheet("表1");
        // 3 创建行  下标从0开始  第一行
        Row row1 = sheet1.createRow(0);

        // 4 创建单元格 (1,1)
        Cell cell1 = row1.createCell(0);
        // 5 往第一个单元格填入数据
        cell1.setCellValue("今日新加入");
        // 6 创建第二个单元格(1,2)
        Cell cell2 = row1.createCell(1);
        cell2.setCellValue("统计时间");

        // 创建第二行
        Row row2 = sheet1.createRow(1);
        // (2,1)
        Cell cell3 = row2.createCell(0);
        cell3.setCellValue("11000");
        // (2,2)
        Cell cell4 = row2.createCell(1);
        cell4.setCellValue(new DateTime().toString("yyyy-MM-dd hh:mm:ss"));

        // 生成表的IO流 03
        FileOutputStream fos = new FileOutputStream(PATH + "03版本excel.xls");
        // 将工作簿写入
        workbook.write(fos);

        // 释放流
        fos.close();
        System.out.println("创建成功");

    }

07版本测试

 @Test
    public void test07() throws Exception {
        // 1 创建工作簿
        Workbook workbook = new SXSSFWorkbook();    //todo
        // 2 创建工作表
        Sheet sheet1 = workbook.createSheet("表1");
        // 3 创建行  下标从0开始  第一行
        Row row1 = sheet1.createRow(0);

        // 4 创建单元格 (1,1)
        Cell cell1 = row1.createCell(0);
        // 5 往第一个单元格填入数据
        cell1.setCellValue("今日新加入");
        // 6 创建第二个单元格(1,2)
        Cell cell2 = row1.createCell(1);
        cell2.setCellValue("统计时间");

        // 创建第二行
        Row row2 = sheet1.createRow(1);
        // (2,1)
        Cell cell3 = row2.createCell(0);
        cell3.setCellValue("11000");
        // (2,2)
        Cell cell4 = row2.createCell(1);
        cell4.setCellValue(new DateTime().toString("yyyy-MM-dd hh:mm:ss"));

        // 生成表的IO流 03
        FileOutputStream fos = new FileOutputStream(PATH + "07版本excel.xlsx");
        // 将工作簿写入
        workbook.write(fos);

        // 释放流
        fos.close();
        System.out.println("创建成功");

    }

第二步:大数据写入

HSSF 写入

缺点:最多只能处理65536行数据,否则会抛出异常 java.lang.IllegalArgumentException

优点:过程中写入缓存,不操作磁盘,最后一次性写入磁盘,速度快

//    03 版本  多数据写入
    @Test
    public void test03BigData() throws Exception {
        long start = System.currentTimeMillis();

        Workbook workbook = new HSSFWorkbook();
        Sheet s1 = workbook.createSheet("表1");

        for (int row = 0; row < 65536; row++) {
            Row rows = s1.createRow(row);
            for (int cell = 0; cell < 10; cell++) {
                Cell cells = rows.createCell(cell);
                cells.setCellValue(cell);
            }
        }
        FileOutputStream fos = new FileOutputStream(PATH + "03BigData.xls");
        workbook.write(fos);
        fos.close();

        long end = System.currentTimeMillis();
        System.out.println("总耗时:" + (double) (end - start) / 1000 + "秒");
    }

XSSF 写入

缺点:写数据使速度非常慢,非常耗费内存,也会发生内存溢出 ,如写100万条数据

优点:可以写比HSSF大的数据量,如20万条数据

//    07 版本 低性能  XSSF
    @Test
    public void test07BigData() throws Exception {
        long start = System.currentTimeMillis();

        Workbook workbook = new XSSFWorkbook();
        Sheet s1 = workbook.createSheet("表1");

        for (int row = 0; row < 65536; row++) {
            Row rows = s1.createRow(row);
            for (int cell = 0; cell < 10; cell++) {
                Cell cells = rows.createCell(cell);
                cells.setCellValue(cell);
            }
        }
        FileOutputStream fos = new FileOutputStream(PATH + "07BigData-XSSF.xlsx");
        workbook.write(fos);
        fos.close();

        long end = System.currentTimeMillis();
        System.out.println("总耗时:" + (double) (end - start) / 1000 + "秒");
    }

SXSSF 写入

优点:可以写非常大的数据量,如100万条,写的速度也非常快,占用更少的内存

SXSSFWorkbook-来至官方的解释∶实现"BigGridDemo"策略的流式XSSFWorkbook版本。这允许写入非常大的文件而不会耗尽内存,因为任何时候只有可配置的行部分被保存在内存中。

请注意,仍然可能会消耗大量内存,这些内存基于您正在使用的功能,例如合并区域,注.…….然只存储在内存中,因此如果广泛使用,可能需要大量内存。 可以使用jprofile来监控

注意

过程中会产生临时文件,需要清理临时文件

默认由100条记录被保存到内存中,如果超过这个数量,则最前面的数据就被写入临时文件,如果想自定义内存中数据的数量,可以使用 new SXSSFWorkbook(数量)

 //    07 版本 高性能  SXSS   性能优化
    @Test
    public void test07BigData2() throws Exception {
        long start = System.currentTimeMillis();

        Workbook workbook = new SXSSFWorkbook();
        Sheet s1 = workbook.createSheet("表1");

        for (int row = 0; row < 65536; row++) {
            Row rows = s1.createRow(row);
            for (int cell = 0; cell < 10; cell++) {
                Cell cells = rows.createCell(cell);
                cells.setCellValue(cell);
            }
        }
        FileOutputStream fos = new FileOutputStream(PATH + "07BigData-SXSS.xlsx");
        workbook.write(fos);
        fos.close();

        // todo 清除临时文件 需要强转类型
        ((SXSSFWorkbook) workbook).dispose();

        long end = System.currentTimeMillis();
        System.out.println("总耗时:" + (double) (end - start) / 1000 + "秒");
    }

POI 读数据

在读取excel表格中单元格中的数据的时候,需要注意一点的是:单元格中的数据可以有String类型、Number类型、Date类型等,所以需要通过Switch-case来进行判断获取,否则会报错

private String PATH = "E:\\JavaCode\\Maven\\excel-demo\\src";

不用清除临时文件


HSSF 读数据

@Test
    public void Read03() throws Exception {
        // 0、得到文件输入流
        FileInputStream fis = new FileInputStream(PATH + "03版本excel.xls");
        // 1、工作簿
        Workbook workbook = new HSSFWorkbook(fis);
        // 2.得到表   可以根据索引也可以根据表的名称
        Sheet sheet = workbook.getSheetAt(0);
        // 3、得到行
        Row row = sheet.getRow(1);
        // 4、得到列   锁定单元格
        Cell cell = row.getCell(0);
        // 5、根据类型得到单元格中的内容
        String value = cell.getStringCellValue();

        System.out.println(value);
    }

XSSF 读数据

@Test
    public void Read07() throws Exception {
        // 0、得到文件输入流
        FileInputStream fis = new FileInputStream(PATH + "07版本excel.xls");
        // 1、工作簿
        Workbook workbook = new XSSFWorkbook(fis);
        // 2.得到表   可以根据索引也可以根据表的名称
        Sheet sheet = workbook.getSheetAt(0);
        // 3、得到行
        Row row = sheet.getRow(1);
        // 4、得到列   锁定单元格
        Cell cell = row.getCell(0);
        // 5、根据类型得到单元格中的内容
        String value = cell.getStringCellValue();

        System.out.println(value);
    }


循环读取多个不同类型的数据

 // 读取多个
    @Test
    public void Read03teset02() throws Exception {
       String path = PATH + "人员.xlsx";
       getData(path);
    }
    public static void getData(String path){
        // 0、得到文件输入流
        FileInputStream fis = null;
        try {
            fis = new FileInputStream(path);
            Workbook workbook = new XSSFWorkbook(fis);
            Sheet sheet = workbook.getSheetAt(0);

            // 1.拿到第一行所有为String类型的数据
            Row row = sheet.getRow(0);
            if (row != null) {
                int cellCount = row.getPhysicalNumberOfCells();  // 该行总共的单元格数
                for (int cellNum = 0; cellNum < cellCount; cellNum++) {
                    Cell cell = row.getCell(cellNum);
                    if (cell != null) {
                        String cellTitle = cell.getStringCellValue();
                        System.out.print(cellTitle + "|");
                    }

                }
                System.out.println("");
            }

            // 2.拿到剩下行数中的数据
            int rowCount = sheet.getPhysicalNumberOfRows();  // 总行数
            for (int rowNum = 1; rowNum < rowCount; rowNum++) {
                Row rowData = sheet.getRow(rowNum);
                int cellCount = rowData.getPhysicalNumberOfCells();  //总单元格数

                String cellValue = "";

                for (int cellNum = 0; cellNum < cellCount; cellNum++) {
                    Cell cell = rowData.getCell(cellNum);
                    if (cell != null) {
                        // 判断类型
                        int cellType = cell.getCellType();

                        switch (cellType) {

                            case Cell.CELL_TYPE_STRING:   //  todo 字符串
                                cellValue = cell.getStringCellValue();
                                break;

                            case Cell.CELL_TYPE_NUMERIC:   // todo 数值 || 日期
                                if (HSSFDateUtil.isCellDateFormatted(cell)){
                                    // 日期
                                    Date date= cell.getDateCellValue();
                                    cellValue = new DateTime().toString("yyyy-MM-dd hh:mm:ss");
                                }else{
                                    // 数字   防止数字过长
                                    cell.setCellType(XSSFCell.CELL_TYPE_STRING);
                                    cellValue = cell.toString();
                                }
                                break;

                            case Cell.CELL_TYPE_BOOLEAN:  // todo 布尔
                                boolean boolean_value = cell.getBooleanCellValue();
                                cellValue = String.valueOf(boolean_value);
                                break;

                            case Cell.CELL_TYPE_BLANK:   //  todo 为空
                                System.out.print("[BLANK]");
                                break;

                            case Cell.CELL_TYPE_ERROR:   //  todo 数据类型错误
                                System.out.print("[数据类型错误]");
                                break;
                        }

                        System.out.print(cellValue+"|");
                    }
                }
                System.out.println("");
            }


        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            // 3 释放资源
            try {
                fis.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }

    }


计算公式

在excel表格中存在着很多个公式 如:sum、排序、求平均值等。这个时候就需要进行判断。了解即可,需要时可以再看

 @Test
    public void test1() throws Exception {
        FileInputStream fis = new FileInputStream("E:\\JavaCode\\Maven\\excel-demo\\公式.xls");
        Workbook workbook = new HSSFWorkbook(fis);
        Sheet sheet = workbook.getSheetAt(0);
        Row row = sheet.getRow(2);
        Cell cell = row.getCell(0);

        // 得到表中的计算公式
        FormulaEvaluator FormulaEvaluator = new HSSFFormulaEvaluator((HSSFWorkbook) workbook);

        // 得到单元格的内容
        int cellType = cell.getCellType();
        switch (cellType){
            case Cell.CELL_TYPE_FORMULA:  // 公式
                // 拿到公式
                String cellFormula = cell.getCellFormula();
                System.out.println(cellFormula);
                //计算
                CellValue evaluate = FormulaEvaluator.evaluate(cell);
                System.out.println("evaluate---"+evaluate); // org.apache.poi.ss.usermodel.CellValue [700.0]
                String value = evaluate.formatAsString(); // 将数据格式化为字符串
                System.out.println(value);
                break;
        }
        
    }


easyExcel读写数据

根据实体类自动生成表

第一步:导入依赖

该依赖中自带了很多种依赖,如lombok、spring-boot等,需要我们在引入依赖时将自己已经导入的依赖删除,不然会报依赖冲突的错误

<dependency>
	<groupId>com.alibaba</groupId>
	<artifactId>easyexcel</artifactId>
	<version>3.1.4</version>
</dependency>
<dependency>
	<groupId>org.projectlombok</groupId>
	<artifactId>lombok</artifactId>
	<version>1.18.8</version>
</dependency>
<dependency>
	<groupId>com.alibaba</groupId>
	<artifactId>fastjson</artifactId>
	<version>1.2.75</version>
</dependency>

·写数据

第二步:创建excel表对应的实体类

@Data
public class DemoData {
    @ExcelProperty("字符串标题")
    private String string;
    @ExcelProperty("日期标题")
    private Date date;
    @ExcelProperty("数字标题")
    private Double aDouble;
    // 忽略该字段
    @ExcelIgnore
    private String ignore;
}


第三步:编写设置数据的方法

使用集合list来写入excel

 public List easyTest1(){
       List<DemoData> list = new ArrayList<DemoData>();
        for (int i = 0; i < 10; i++) {
            DemoData data = new DemoData();
            data.setString("字符串"+i);
            data.setDate(new Date());
            data.setADouble(0.12);
            list.add(data);
        }
        return list;
    }

第四步:编写测试类

使用链式编写的方式

write(文件路径,excel表对应的java类)

sheet(设置表名)

doWrite(数据)

 @Test
    public void test1(){
        String fileName ="E:\\JavaCode\\Maven\\excel-demo\\easyEasyData.xlsx";
        EasyExcel.write(fileName,DemoData.class).sheet("表1").doWrite(data());
    }

·读数据

1、每执行一条excel表中的数据都会执行一次监听文件中的invoke方法,所以如果需要修改可以修改invoke方法中的内容

2、DemoDataListener 不能被spring管理,要每次读取excel都要new,然后里面用到spring可以构造方法传进去

第二步:准备一个对应excel表中字段的类

与写操作中使用同一个类

@Data
public class DemoData {
    @ExcelProperty("字符串标题")
    private String string;
    @ExcelProperty("日期标题")
    private Date date;
    @ExcelProperty("数字标题")
    private Double aDouble;
    // 忽略该字段
    @ExcelIgnore
    private String ignore;
}


第三步:创建数据层 Mapper || Dao

public class DemoDAO {
    public void save(List<DemoData> list) {
        // 如果是mybatis,尽量别直接调用多次insert,自己写一个mapper里面新增一个方法batchInsert,所有数据一次性插入
    }
}

第四步:创建监听器

package excel.readEasy;

import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.metadata.CellExtra;
import com.alibaba.excel.metadata.data.ReadCellData;
import com.alibaba.excel.read.listener.ReadListener;
import com.alibaba.excel.util.ListUtils;
import com.alibaba.fastjson.JSON;
import excel.easy.DemoData;
import lombok.extern.slf4j.Slf4j;

import java.util.List;
import java.util.Map;

// 有个很重要的点 DemoDataListener 不能被spring管理,要每次读取excel都要new,然后里面用到spring可以构造方法传进去
@Slf4j
public class DemoDataListener implements ReadListener<DemoData> {
    //每隔5条存储数据库,实际使用中可以100条,然后清理list ,方便内存回收
    private static final int BATCH_COUNT = 100;
    //缓存的数据
    private List<DemoData> cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);


    //假设这个是一个DAO,当然有业务逻辑这个也可以是一个service。当然如果不用存储这个对象没用
    private DemoDAO demoDAO;

    // 这里是demo,所以随便new一个。实际使用如果到了spring,请使用下面的有参构造函数
    public DemoDataListener() {
        demoDAO = new DemoDAO();
    }

    //如果使用了spring,请使用这个构造方法。每次创建Listener的时候需要把spring管理的类传进来
    public DemoDataListener(DemoDAO demoDAO) {
        this.demoDAO = demoDAO;
    }


    @Override
    public void onException(Exception exception, AnalysisContext context) throws Exception {

    }

    @Override
    public void invokeHead(Map<Integer, ReadCellData<?>> headMap, AnalysisContext context) {
        System.out.println("111");
    }

    //这个每一条数据解析都会来调用
    @Override
    public void invoke(DemoData data, AnalysisContext analysisContext) {
        System.out.println("2222");
        System.out.println(JSON.toJSONString(data));
        //  log.info("解析到一条数据:{}", JSON.toJSONString(data));
        cachedDataList.add(data);
        // 达到BATCH_COUNT了,需要去存储一次数据库,防止数据几万条数据在内存,容易OOM
        if (cachedDataList.size() >= BATCH_COUNT) {
            saveData();
            // 存储完成清理 list
            cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
        }
    }

    @Override
    public void extra(CellExtra extra, AnalysisContext context) {

    }

    //所有数据解析完成了 都会来调用
    @Override
    public void doAfterAllAnalysed(AnalysisContext analysisContext) {
        saveData();
        log.info("所有数据解析完成!");
    }

    @Override
    public boolean hasNext(AnalysisContext context) {
        return false;
    }

    /**
     * 加上存储数据库
     */
    private void saveData() {
        log.info("{}条数据,开始存储数据库!", cachedDataList.size());
        demoDAO.save(cachedDataList);
        log.info("存储数据库成功!");
    }
}



第五步:测试

@Test
    public void test3() {
        String fileName = "E:\\JavaCode\\Maven\\excel-demo\\easyEasyData.xlsx";
        EasyExcel.read(fileName, DemoData.class, new PageReadListener<DemoData>(dataList -> {
            for (DemoData demoData : dataList) {
                System.out.println(JSON.toJSONString(demoData));
            }
        })).sheet().doRead();
    }

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

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

相关文章

在rhel7系统使用Mariadb

文章目录一 联系和区别二 需求三 部署安装3.1 环境准备3.2 安装软件包3.3 启动服务3.4 设置防火墙策略四 创建用户和库表4.1 登录数据库4.2 创建用户4.3 创建数据库和表五 备份和恢复5.1 备份 com 数据库5.2 模拟误删除操作5.3 恢复表一 联系和区别 Mariadb是由社区开发的一个…

4.4 集成运放的性能指标及低频等效电路

一、集成运放的性能指标 在考察集成运放的性能时&#xff0c;常用下列参数来描述&#xff1a; 1、开环差模增益 AodA_{od}Aod​ 在集成运放无外加反馈时的差模放大倍数称为差模开环增益&#xff0c;记作 AodA_{od}Aod​。AodΔuO/(uP−uN)A_{od}\Delta u_O/(u_P-u_N)Aod​Δ…

【Spring Cloud GateWay】ServerHttpResponseDecorator不生效

文章目录1. BUG描述2. BUG解决3. BUG分析1. BUG描述 在Spring Cloud Gateway使用编码的方式实现一个全局拦截器&#xff0c;在全局拦截器中想要打印响应日志。 于是自己装饰了一个具有打印日志功能的ServerHttpResponseDecorator&#xff0c;但是在转发后的服务返回响应的时候…

在线浏览PDF:Grapecity Documents for PDF Viewer 6.0.2

Grapecity Documents for PDF Viewer跨平台 JavaScript PDF 查看器---备注:必须配合.NET版本才能编辑PDF 使用我们的 JavaScript PDF 查看器在网络上阅读和编辑 PDF。跨浏览器和框架打开和打印。 Grapecity Documents for PDF Viewer全功能的 JavaScript PDF 查看器和 PDF 编辑…

move语言之基础学习(基本类型+表达式+变量)例子

一、基本类型 Move 的基本数据类型包括: 整型 (u8, u64, u128)、布尔型 boolean 和地址 address。 Move 不支持字符串和浮点数。 整型 整型包括 u8、u64 和 u128&#xff0c;我们通过下面的例子来理解整型&#xff1a; script { fun main() { // define empty variable, set…

python(0)计算机基础知识

文章目录计算机是什么计算机的组成计算机的使用方式windows的命令行文本文件和字符集乱码计算机是什么 在现实生活中&#xff0c;越来越无法离开计算机了 电脑、笔记本、手机、游戏机、汽车导航、智能电视。。。 计算机就是一个用来计算的机器 目前来讲&#xff0c;计算机只…

C++模板进阶+继承详解

耕耘和收获不是连贯的&#xff0c;中间还隔着很长一段时间&#xff0c;那就是坚持&#xff01;一&#xff1a;模板进阶1.1&#xff1a;非类型模板参数template<class T,size_t N> class arr { private:T _a[N]; };这里的N就跟define一样&#xff0c;属于非类型模板参数。…

MongoDB常用操作

官网地址&#xff1a;https://www.mongodb.com/docs/manual/reference/method/Date/ 实例&#xff1a;系统上运行的进程及节点集&#xff0c;一个实例可以有多个库&#xff0c;默认端口 27017。如果要在一台机器上启动多个实例&#xff0c;需要设置不同端口和不同的dbpath。库&…

第四章web服务器之httpd

文章目录第四章 web服务器1.1 www简介1.1.1 网址及HTTP简介1.1.2 HTTP协议请求的工作流程1.2 www服务器的类型1.2.1 仅提供用户浏览的单向静态网页1.2.2 提供用户互动接口的动态网站1.3 www服务器的基本配置1.4 实验1.4.1 搭建静态网站——基于http协议的静态网站1.4.2 搭建静态…

Acwing---1211.蚂蚁感冒

蚂蚁感冒1.题目2.基本思想3.代码实现1.题目 长 100 厘米的细长直杆子上有 nnn 只蚂蚁。 它们的头有的朝左&#xff0c;有的朝右。 每只蚂蚁都只能沿着杆子向前爬&#xff0c;速度是 1 厘米/秒。 当两只蚂蚁碰面时&#xff0c;它们会同时掉头往相反的方向爬行。 这些蚂蚁中…

C语言基本数据类型(一)

文章目录 前言 一、int类型 二、八进制和十六进制 三、其他整数类型 四、char 类型 五、_Bool 类型 六、 可移植类型&#xff1a;stdint.h和unttypes.h 前言 C语言基本数据类型包括声明变量、如何表示字面值常量&#xff0c;以及经典的用法。 一、int类型 C语言中包括许…

【openGauss】在openEuler(ARM架构)上安装openGauss(一主一备)

一、系统版本介绍 当前案例中的openGauss安装&#xff0c;底层操作系统为openEuler-20.03-LTS版本&#xff0c;当前openGauss对Python版本兼容性最好的是Python 3.6版本与Python 3.7版本&#xff0c;该实验使用的openEuler版本自带Python 3.7.4&#xff0c;不需要再自行安装 二…

零基础如何入门网络安全?2023年最新,建议收藏!

前言 最近收到不少关注朋友的私信和留言&#xff0c;大多数都是零基础小友入门网络安全&#xff0c;需要相关资源学习。其实看过的铁粉都知道&#xff0c;之前的文里是有过推荐过的。新来的小友可能不太清楚&#xff0c;这里就系统地叙述一遍。 01.简单了解一下网络安全 说白…

前端必会手写面试题合集

实现Event(event bus) event bus既是node中各个模块的基石&#xff0c;又是前端组件通信的依赖手段之一&#xff0c;同时涉及了订阅-发布设计模式&#xff0c;是非常重要的基础。 简单版&#xff1a; class EventEmeitter {constructor() {this._events this._events || ne…

电力系统IEEE33节点Simulink仿真研究(Matlab实现)

&#x1f4a5;&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️❤️&#x1f4a5;&#x1f4a5;&#x1f4a5; &#x1f389;作者研究&#xff1a;&#x1f3c5;&#x1f3c5;&#x1f3c5;主要研究方向是电力系统和智能算法、机器学…

arduino基本知识认识和学习资源

个人对ardunio的感觉 **像是一个模块化功能的单片机&#xff0c;编程时在单片机中就像python在计算机语言的感觉。**硬件方面的功能比较单一依赖于传感器和硬件电路&#xff1b;编程比较简单&#xff0c;所有执行的函数都已经被封装&#xff0c;所以想要成为第一个用这个库吃瓜…

【C语言刷题】猜名次、猜凶手、杨辉三角、杨氏矩阵、字符串左旋、判断是否为左旋子串

目录 一、猜名次 二、猜凶手 三、杨辉三角 解法一&#xff1a; 解法二 四、杨氏矩阵 解法一 解法二 五、字符串左旋 解法一 解法二 六、判断是否为字符串左旋字串 解法一 解法二 总结 一、猜名次 5位运动员参加了10米台跳水比赛&#xff0c;有人让他们预测比赛结果…

基于轻量级CNN开发构建学生课堂行为识别系统

其实早在之前&#xff0c;我的一些文章里面就有做过关于学生课堂行为检测识别的项目&#xff0c;感兴趣的话可以自行移步阅读&#xff1a;《yolov4-tiny目标检测模型实战——学生姿势行为检测》《基于yolov5轻量级的学生上课姿势检测识别分析系统》这些主要是偏目标检测类的项目…

Raft论文阅读

Raft 论文阅读 参考&#xff1a; Raft Paper 一文搞懂Raft算法 - xybaby Raft Demo Raft 实现汇总 Raft 为什么是更易理解的分布式一致性算法 空挡) Raft协议实现之etcd(一)&#xff1a;基本架构 Raft 协议 - buttercup 【raft】学习二&#xff1a;etcd/raft raft数据结构学…

第四十八讲:神州路由器IPv6静态路由配置

实验拓扑图如下所示 配置步骤&#xff1a; 步骤1&#xff1a;配置R1接口 R1_config#ipv6 unicast-routing //启用IPv6路由 R1_config#in g0/4 R1_config-_g0/4#ipv6 address 2001:10::1/64 //手工配置IPv6地址 R1_config-…