Java modbus 实现RTU串口作为slave(服务端)读写数据

这里要了解下modbus的RTU和TCP 的几个名称关系:

Modbus/RTU:主站 和从站 关系

Modbus/TCP:客户端和服务端关系

关系

主站主动找从站读写数据
客户端主动找服务端读写数据

所以当使用Modbus/TCP时,主站一般作为客户端,从站一般作为服务端

modbus poll和modbus slave模拟软件

当使用Modbus/TCP时,modbus poll一般模拟客户端,modbus slave一般模拟服务端。

以上这几个关系必须了解,否则容易搞混乱。

modbus协议是工业使用比较多的协议,大部分设备都支持modbus的,也就是说,通过modbus可以获取市场上90%的设备,一般新设备都支持modbus。

modbus有几种协议:

modbusTCP、modbusRTU、modbusRTUOVERTCP(本质tcp,需要单独解析)、ASCII(不常用,可以忽略)

Modbus RTU和Modbus TCP两个协议的本质都是Modbus协议,都是靠Modbus寄存器地址来交换数据;但所用的硬件接口不一样,Modbus RTU一般采用串口RS232C或RS485/422,而Modbus TCP一般采用以太网口。现在市场上有很多协议转换器,可以轻松的将这些不同的协议相互转换。
实际上Modbus协议包括ASCII、RTU、TCP。
Modbus协议需要对数据进行校验,串行协议中除有奇偶校验外,ASCII模式采用LRC校验,RTU模式采用16位CRC校验.
Modbus TCP模式没有额外规定校验,因为TCP协议是一个面向连接的可靠协议。
TCP和RTU协议非常类似,只要把RTU协议的两个字节的校验码去掉,然后在RTU协议的开始加上5个0和一个6并通过TCP/IP网络协议发送出去即可。

以上应该可以理解modbus到底干啥的了,明白后,我们开始做测试,整理流程比较长。

先说工具准备情况:

1、虚拟串口工具 vspd,正常安装即可(目的是模拟串口,如果是tcp 协议用不到它)

2、modbus 仿真工具,这个工具很好用,也经常使用

3、java依赖,该依赖比较强大,强推

       <dependency>
            <groupId>com.intelligt.modbus</groupId>
            <artifactId>jlibmodbus</artifactId>
            <version>1.2.9.11</version>
        </dependency>

模拟工具和仿真工具下载。

1、串口模拟工具正常安装即可,然后把补丁进行覆盖安装目录,否则只有14天试用期,建议覆盖,永久使用(安装过程省略)

安装成功后,添加串口,默认是COM2和COM3 两个端口,就用这两个就可以了。

我这边是COM2个slave用,也就是tcp的服务端哈,COM3给master用,也就是tcp的客户端。

其实这块在写代码之前可以进行用modbus 仿真工具进行测试下,例如:

1、打开modbus slave;选择串口

协议类型选择:串口

串口设置,这块根据你的串口模拟工具,我把COM2 slave了,如果是真实设备,具体连接到真实设备的串口即可。

其他的设置,根据具体情况来即可。

点击ok,

应该可以看到模拟工具在跑数据了哈

可以看到串口模拟工具已经被modbus slave模拟器连上了 9600-N-8-1 就是刚才配置的其他参数信息,了解就行。

可以看到COM3的Port 是closed状态,我们这时开始启动poll模拟器。

和slave设置很像,我这边细节就不说了,有个注意点串口要连COM3,因为COM2已经被slave占用了(这里因为是模拟的设备,COM2和COM3 模拟器帮我们进行连接了)。

看modbus模拟器数据已经连上了,也获取到数据了:

串口模拟器也显示连上了:

如果报错,请检查内存起止地址和内存数量。这里通过工具进行模拟的,整体是没问题的,说明环境都是没问题的,确保在编码前环境一定没问题,否则只用用代码容易出问题;可以进行编码了。

用java进行连接,其实我的理解,哪种语言连接都是差不多的,比较简单,直接上代码了哈:

监听接口:

public interface ModbusEventListener {

    /**
     * 描述
     * 单个线圈写入监听  function=01 、 02
     *
     * @param address 内存地址
     * @param value   值
     * @return void
     * @author
     * @date 2024/5/11 15:30:31
     */
    default void onWriteToSingleCoil(int address, boolean value) {
        System.out.print("onWriteToSingleCoil: address " + address + ", value " + value);
    }

    /**
     * 描述
     * 批量线圈写入监听  function=01 、 02
     *
     * @param address  地址
     * @param quantity 数量
     * @param values   值
     * @return void
     * @author
     * @date 2024/5/11 15:31:20
     */
    default void onWriteToMultipleCoils(int address, int quantity, boolean[] values) {
        System.out.print("onWriteToMultipleCoils: address " + address + ", quantity " + quantity);
    }

    /**
     * 描述
     * 读保持寄存器	  function=03
     *
     * @param address 地址
     * @param value   值
     * @return void
     * @author 
     * @date 2024/5/11 15:32:26
     */
    void onWriteToSingleHoldingRegister(int address, int value);

    /**
     * 描述
     * 读保持寄存器	  function=03
     *
     * @param address  地址
     * @param quantity 数量
     * @param values   多个值
     * @return void
     * @author
     * @date 2024/5/11 15:49:22
     */
    void onWriteToMultipleHoldingRegisters(int address, int quantity, int[] values);

    /**
     * 描述
     * 获取监听类型
     *
     * @param
     * @return com.goldstar.common.utils.constants.GlobalConstants.ModbusListenerType
     * @author 
     * @date 2024/5/13 10:54:14
     */
    String getListenerType();


}

dataholder:

/**
 * 从机的寄存器
 */
public class CustomDataHolder extends DataHolder {

    final List<ModbusEventListener> modbusEventListenerList = new ArrayList();


    public CustomDataHolder() {
        // you can place the initialization code here
            /*
            something like that:
            setHoldingRegisters(new SimpleHoldingRegisters(10));
            setCoils(new Coils(128));
            ...
            etc.
             */
    }

    public void addEventListener(ModbusEventListener listener) {
        modbusEventListenerList.add(listener);
    }

    public boolean removeEventListener(ModbusEventListener listener) {
        return modbusEventListenerList.remove(listener);
    }

    @Override
    public void writeHoldingRegister(int offset, int value) throws IllegalDataAddressException, IllegalDataValueException {
        System.out.println("writeHoldingRegister: offset " + offset + ", value " + value);
        for (ModbusEventListener l : modbusEventListenerList) {
            l.onWriteToSingleHoldingRegister(offset, value);
        }
        super.writeHoldingRegister(offset, value);
    }

    @Override
    public void writeHoldingRegisterRange(int offset, int[] range) throws IllegalDataAddressException, IllegalDataValueException {
        System.out.println("writeHoldingRegisterRange: offset " + offset + ", range " + range);
        for (ModbusEventListener l : modbusEventListenerList) {
            l.onWriteToMultipleHoldingRegisters(offset, range.length, range);
        }
        super.writeHoldingRegisterRange(offset, range);
    }

    @Override
    public void writeCoil(int offset, boolean value) throws IllegalDataAddressException, IllegalDataValueException {
        System.out.println("writeCoil: offset " + offset + ", value " + value);
        for (ModbusEventListener l : modbusEventListenerList) {
            l.onWriteToSingleCoil(offset, value);
        }
        super.writeCoil(offset, value);
    }

    @Override
    public void writeCoilRange(int offset, boolean[] range) throws IllegalDataAddressException, IllegalDataValueException {
        System.out.println("writeCoilRange: offset " + offset + ", range " + range);
        for (ModbusEventListener l : modbusEventListenerList) {
            l.onWriteToMultipleCoils(offset, range.length, range);
        }
        super.writeCoilRange(offset, range);
    }
}

modbus slave服务 

public class ModbusSlaveService {


    private String modbusType = "";

    private ModbusSlave modbusSlave;

    // 创建从机的寄存器
    private CustomDataHolder dataHolder = new CustomDataHolder();

    //禁止无参构造函数创建对象
    private ModbusSlaveService() {
    }

    public ModbusSlaveService(@NonNull TcpParameters tcpParameters, @NonNull Integer serverId) {
        this.modbusSlave = ModbusSlaveFactory.createModbusSlaveTCP(tcpParameters);
        modbusSlave.setReadTimeout(0); // if not set default timeout is 1000ms, I think this mus
        // 为从机寄存器添加监听事件,这里的监听事件主要是主机如果发送写命令修改从机则进行业务处理
        //  dataHolder.addEventListener(new DefaultModbusEventListener());
        modbusSlave.setDataHolder(dataHolder);
        modbusSlave.setServerAddress(serverId);
        modbusType = "协议类型:tcp, ip: " + tcpParameters.getHost() + ":" + tcpParameters.getPort();
    }

    public ModbusSlaveService(@NonNull SerialParameters serialParameters, @NonNull Integer serverId) throws SerialPortException {
        this.modbusSlave = ModbusSlaveFactory.createModbusSlaveRTU(serialParameters);
        modbusSlave.setReadTimeout(0); // if not set default timeout is 1000ms, I think this mus
        // 为从机寄存器添加监听事件,这里的监听事件主要是主机如果发送写命令修改从机则进行业务处理
        //  dataHolder.addEventListener(new DefaultModbusEventListener());
        modbusSlave.setDataHolder(dataHolder);
        modbusSlave.setServerAddress(serverId);

        modbusType = "协议类型:RTU(串口), 串口号: " + serialParameters.getDevice();
    }

    /**
     * 注册相关点位数据;用来服务端写到客户端数据
     *
     * @param modbusHoldingRegisters
     * @return
     */
    public ModbusSlaveService addHoldingRegisters(ModbusHoldingRegisters modbusHoldingRegisters) {
        if (modbusHoldingRegisters == null) {
            return this;
        }
        modbusSlave.getDataHolder().setHoldingRegisters(modbusHoldingRegisters);
        return this;
    }

    /**
     * 注册相关点位数据;用来服务端写到客户端数据
     *
     * @param modbusCoils
     * @return
     */
    public ModbusSlaveService addModbusCoils(ModbusCoils modbusCoils) {
        if (modbusCoils == null) {
            return this;
        }
        modbusSlave.getDataHolder().setCoils(modbusCoils);
        return this;
    }

    public ModbusSlaveService addModbusInputRegisters(ModbusHoldingRegisters inputRegisters) {
        if (inputRegisters == null) {
            return this;
        }
        modbusSlave.getDataHolder().setInputRegisters(inputRegisters);
        return this;
    }

    /**
     * 数据监听器,用来监听 客户端写入到数据,进行相关处理
     *
     * @param listener
     * @return
     */
    public ModbusSlaveService addEventListener(ModbusEventListener listener) {
        dataHolder.addEventListener(listener);
        return this;
    }

    public ModbusSlaveService removeEventListener(ModbusEventListener listener) {
        dataHolder.removeEventListener(listener);
        return this;
    }

    /**
     * 开启监听
     *
     * @return
     * @throws Exception
     */
    public ModbusSlaveService startListen() throws Exception {
        modbusSlave.listen();
        log.info("{} ----当前服务已启动", modbusType);
        closeSlave();
        return this;
    }

    /**
     * 关闭modbus 服务
     *
     * @throws InterruptedException
     * @throws ModbusIOException
     */
    private void closeSlave() throws InterruptedException, ModbusIOException {
        //这部分代码主要是设置Java虚拟机关闭的时候需要做的事情,即本程序关闭的时候需要做的事情,直接使用即可
        if (modbusSlave.isListening()) {
            Runtime.getRuntime().addShutdownHook(new Thread(() -> {
                synchronized (modbusSlave) {
                    modbusSlave.notifyAll();
                }
            }));

            synchronized (modbusSlave) {
                modbusSlave.wait();
            }
            modbusSlave.shutdown();
            log.info("{} ----当前服务已关闭", modbusType);

        }
    }

    /**
     * 手动进行关闭服务
     */
    public void closeModbusSlave() {
        try {
            modbusSlave.shutdown();
            log.info("{} ----当前服务已手动关闭", modbusType);
        } catch (ModbusIOException e) {
            throw new RuntimeException(e);
        }

    }

    public static void main(String[] args) throws Exception {      

       SerialParameters serialParameters = new SerialParameters();
        serialParameters.setDevice("COM2");
        serialParameters.setParity(SerialPort.Parity.NONE);
        //  SerialPort.BaudRate baudrate = new SerialPort.BaudRate(921600);
        // serialParameters.setBaudRate(baudrate);
        serialParameters.setBaudRate(SerialPort.BaudRate.BAUD_RATE_9600);
        serialParameters.setDataBits(8);
        serialParameters.setStopBits(1);

        ModbusSlaveService modbusSlaveService1 = new ModbusSlaveService(serialParameters, 1);
        //modbusSlaveService1.addEventListener(new DefaultModbusEventListener());
        modbusSlaveService1.addEventListener(new ModbusEventListener() {
            @Override
            public void onWriteToSingleCoil(int address, boolean value) {
                log.info("onWriteToSingleCoil({},{})", address, value);
            }

            @Override
            public void onWriteToMultipleCoils(int address, int quantity, boolean[] values) {
                log.info("onWriteToMultipleCoils({},{},{})", address, quantity, values);
            }

            @Override
            public void onWriteToSingleHoldingRegister(int address, int value) {
                log.info("onWriteToSingleHoldingRegister({},{})", address, value);
            }

            @Override
            public void onWriteToMultipleHoldingRegisters(int address, int quantity, int[] values) {
                log.info("onWriteToMultipleHoldingRegisters({},{},{})", address, quantity, values);
            }


            @Override
            public String getListenerType() {
                return "";
            }

        });
        //更新寄存器值
        ModbusHoldingRegisters hr1 = new ModbusHoldingRegisters(10);
        //主机发送数据
        ThreadUtil.execute(() -> {
            while (true) {
                ThreadUtil.sleep(1000);
                try {
                    hr1.set(0, RandomUtil.randomInt(0, 10000));
                    hr1.set(1, RandomUtil.randomInt(0, 10000));
                    hr1.set(2, RandomUtil.randomInt(0, 10000));
                    hr1.set(3, RandomUtil.randomInt(0, 10000));
                    hr1.set(4, RandomUtil.randomInt(0, 10000));
                    log.info("COM数据:{}", JSONUtil.toJsonStr(hr1));
                    modbusSlaveService1.addHoldingRegisters(hr1);
                } catch (IllegalDataAddressException e) {
                    throw new RuntimeException(e);
                } catch (IllegalDataValueException e) {
                    throw new RuntimeException(e);
                }

            }
        });

        modbusSlaveService1.startListen();
        


    }


}

测试代码说明:

 public static void main(String[] args) throws Exception { 
//串口参数实体类
        SerialParameters serialParameters = new SerialParameters();
        //这是设置串口 刚才上面说了,用COM2作为slave
        serialParameters.setDevice("COM2");
        //用模拟器时的其他几个参数设置
        serialParameters.setParity(SerialPort.Parity.NONE);
        //  SerialPort.BaudRate baudrate = new SerialPort.BaudRate(921600);
        // serialParameters.setBaudRate(baudrate);
        serialParameters.setBaudRate(SerialPort.BaudRate.BAUD_RATE_9600);
        serialParameters.setDataBits(8);
        serialParameters.setStopBits(1);
        //new modbus slave 对象
        ModbusSlaveService modbusSlaveService1 = new ModbusSlaveService(serialParameters, 1);
        //modbusSlaveService1.addEventListener(new DefaultModbusEventListener());
        //添加监听;此处监听是为了让 master写数据时,slave能监听到,这样就可以相互通信了,这是扩展项
        modbusSlaveService1.addEventListener(new ModbusEventListener() {
            @Override
            public void onWriteToSingleCoil(int address, boolean value) {
                log.info("onWriteToSingleCoil({},{})", address, value);
            }

            @Override
            public void onWriteToMultipleCoils(int address, int quantity, boolean[] values) {
                log.info("onWriteToMultipleCoils({},{},{})", address, quantity, values);
            }

            @Override
            public void onWriteToSingleHoldingRegister(int address, int value) {
                log.info("onWriteToSingleHoldingRegister({},{})", address, value);
            }

            @Override
            public void onWriteToMultipleHoldingRegisters(int address, int quantity, int[] values) {
                log.info("onWriteToMultipleHoldingRegisters({},{},{})", address, quantity, values);
            }


            @Override
            public String getListenerType() {
                return "";
            }

        });
        //更新寄存器值,每个一秒更新一次数据
        ModbusHoldingRegisters hr1 = new ModbusHoldingRegisters(10);
        //主机发送数据;这里如果是真实场景的业务,就是业务数据了,我这边通过随机生成的数据进行模拟的,大家根据自己的情况进行设置即可
        ThreadUtil.execute(() -> {
            while (true) {
                ThreadUtil.sleep(1000);
                try {
                    hr1.set(0, RandomUtil.randomInt(0, 10000));
                    hr1.set(1, RandomUtil.randomInt(0, 10000));
                    hr1.set(2, RandomUtil.randomInt(0, 10000));
                    hr1.set(3, RandomUtil.randomInt(0, 10000));
                    hr1.set(4, RandomUtil.randomInt(0, 10000));
                    log.info("COM数据:{}", JSONUtil.toJsonStr(hr1));
                    modbusSlaveService1.addHoldingRegisters(hr1);
                } catch (IllegalDataAddressException e) {
                    throw new RuntimeException(e);
                } catch (IllegalDataValueException e) {
                    throw new RuntimeException(e);
                }

            }
        });
        //启动监听
        modbusSlaveService1.startListen();
      } 

上面明白后,咱们运行起来:

1、开发工具日志:

已经启动了,再看看串口模拟器:

这里可以看到,COM2已经换成了java.exe了,这就对了

在看poll工具:

数据实时读取到了,我们测试下poll也就是master 发送数据,slave能否接收到:

Ok,slave 发送master(poll)  和master(poll) 发送slave 都是没问题的,完美解决,后面进行业务集成即可。

总结说明:

其实工业的协议要比TCP简单很多,逻辑不复杂,大家尝试下就明白了,modbusTCP 相互通信更简单了,这里不进行演示了,上面代码其实我已经写了,大家可以写下测试代码就支持tcp了,Ok到此结束。

后续会深入研究modbus协议,争取用java进行实现modbus协议,也就是上面用的库文件,已经进行实现了,进行尝试手动实现,这样后期可以深入定制业务以及各类定制化modbus协议。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【观成科技】加密C2框架Xiebro流量分析

一、工具介绍 Xiebro是由Golang和 .NET编写&#xff0c;提供支持的多人和多服务器 C2/后开发框架。它支持多种通信协议&#xff0c;包括TCP、websocket等&#xff0c;并且在客户端与Xiebro服务器之间的通信通常采用AES加密来保障安全性和隐蔽性。 二、工具原理分析 Xiebro C…

Sectigo SSL证书申请的流程是怎样的?

在当今数字化时代&#xff0c;网络安全成为了一个不可忽视的问题。为了保护网站和用户数据的安全&#xff0c;SSL证书成为了网站运营的重要组成部分。Sectigo作为全球领先的数字证书颁发机构之一&#xff0c;提供了一系列的证书解决方案来满足不同类型网站的需求。以下是对Sect…

一体化水处理设备有哪些

一体化水处理设备是一种集成了多种水处理工艺的紧凑型设备&#xff0c;适用于各种规模的水处理需求&#xff0c;包括生活污水、工业废水、饮用水处理等。这些设备通常设计为模块化&#xff0c;便于安装、运输和扩展。以下是一些常见的一体化水处理设备类型&#xff1a; 一体化生…

针对大型企业网站和内部网的框架/内容管理系统建议

1. 问题背景 在大型企业中&#xff0c;网站和内部网的建设至关重要&#xff0c;但企业在选择框架/内容管理系统&#xff08;CMS&#xff09;时往往面临诸多难题。这些难题包括&#xff1a; 稳定性和可靠性&#xff1a; 企业网站和内部网需要稳定可靠&#xff0c;以确保业务的…

【热门话题】Vue.js:现代前端开发的轻量级框架之旅

&#x1f308;个人主页: 鑫宝Code &#x1f525;热门专栏: 闲话杂谈&#xff5c; 炫酷HTML | JavaScript基础 ​&#x1f4ab;个人格言: "如无必要&#xff0c;勿增实体" 文章目录 Vue.js&#xff1a;现代前端开发的轻量级框架之旅一、Vue.js概览1.1 Vue.js的诞…

数据中心逆变电源的功率容量计算方法

随着信息技术的快速发展&#xff0c;数据中心在现代社会中的地位日益凸显&#xff0c;各种企业和机构对数据中心的依赖程度也越来越高。而电源作为数据中心的核心基础设施&#xff0c;其可靠性和高效性直接影响着数据中心的稳定运行。因此&#xff0c;如何设计一款性能优越、可…

以管理员身份运行设置

在使用非 Administrator 用户操作情况下&#xff1a; 举个例子&#xff0c;因为每次想要以管理员身份运行命令提示符&#xff0c;都要右键选择才行&#xff0c;有点麻烦。 可以设置每次点开就是以管理员身份运行命令提示符&#xff0c;操作如下&#xff1a; 1.Win R 输入 s…

知了汇智副总经理赵懋骏出席“走进阿里”CEO联席会,共话AI大模型新趋势

在智能科技日新月异的今天&#xff0c;汇智知了堂副总经理赵懋骏于3月28日受邀出席了在天府软件园举行的“走进阿里–2024年CEO联席会”&#xff0c;会议聚焦阿里云AI技术的最新进展与行业应用&#xff0c;特别是“AI技术正在加速变革&#xff1a;大模型的历史、现在与趋势”&a…

批量剪辑与转码解析:一键实现MP4到FLV顺畅转换的技巧

在数字化多媒体日益盛行的今天&#xff0c;视频格式转换成为了我们日常生活和工作中不可或缺的一部分。尤其是当需要将MP4格式的视频转换为FLV格式时&#xff0c;批量剪辑与转码功能的出现&#xff0c;极大地提升了我们处理视频的效率。本文将为您详细解析云炫AI智剪如何一键实…

apisix3.9.1 和 dashboard 离线安装

服务器配置 centos7 linux x86 64 前置 需要将离线安装包上传到服务器上 {上传目录 /root/apisix-soft/ } 【建议:优先上传etcd-*.jar \ apisix-*.rpm \ cyrus-*.rpm \ openldap-*.rpm 等安装好apisix后再上传apisix-dashboard-*.rpm】 可以自行网上寻找&#xff0c;或找一台可…

spring cloud alibaba、spring cloud和springboot三者的版本兼容

官方版本说明地址: 版本说明 alibaba/spring-cloud-alibaba Wiki GitHub 组件版本关系 每个 Spring Cloud Alibaba 版本及其自身所适配的各组件对应版本如下表所示(注意,Spring Cloud Dubbo 从 2021.0.1.0 起已被移除出主干,不再随主干演进): Spring Cloud Alibaba Ve…

初识java——javaSE(4)类与对象

文章目录 前言一 类与对象1.1 面向过程与面向对象思想的区别&#xff1a;1.2 类的定义1.3 类的实例化——对象通过创建对象&#xff0c;调用对象中的成员变量与方法 1.4 this关键字this的作用一&#xff1a;this 的作用二构造方法&#xff1a;对象创建的两步方法的重载 this的作…

SGPM02陀螺仪模块通过惯性导航助力AGV小车的发展

之前我们介绍过SGPM01系列陀螺仪模块在智能泳池清洁机器人导航的方案(SGPM01)。这款惯性导航模块收到了许多企业的欢迎。由此&#xff0c;爱普生推出了SGPM02系列陀螺仪模块通过惯性导航&#xff0c;助力AGV小车的发展。 AGV是一种用于运输材料的无人驾驶车辆&#xff0c;并且A…

SHAP分析交互作用的功能,如果你用的模型是xgboost

SHAP分析交互作用的功能&#xff0c;如果你用的模型是xgboost 如果在SHAP分析中使用的是xgoost模型&#xff0c;就可以使用SHAP分析内置的交互作用分析&#xff0c;为分析变量间的相互提供了另外一个观察的视角。关于SHAP交互作用分析&#xff0c;一个参考资料&#xff0c;还是…

word2019 64位 NoteExpress突然无法使用解决方法

之前用的好好的&#xff0c;去除格式化运行过一次。 打开别的文档&#xff0c;突然发现红框中的图标全变灰了 根据教程添加 加载项&#xff0c;然后word以管理员身份重启&#xff0c;NE也以管理员身份运行&#xff0c;又可以了 然后突然又不行了&#xff0c;重启电脑后NE变成…

Java——类和对象第二节——封装

1.什么是封装 封装是面向对象程序的三大特性之一&#xff0c;面向对象程序的三大特性分别是封装&#xff0c;继承&#xff0c;多态 而封装简单来说就是套壳隐藏细节 打个比方&#xff1a; 在一些电脑厂商生产电脑时&#xff0c;仅对用户提供开关机键&#xff0c;键盘输入&a…

怎么防止源代码防泄密

随着数字化时代的到来&#xff0c;源代码作为企业和个人的重要资产之一&#xff0c;承载着无价的知识和创新。然而&#xff0c;源代码泄露已成为当今信息安全领域的重要挑战之一&#xff0c;给企业带来了严重的经济损失和声誉风险。面对这一挑战&#xff0c;我们有责任加强对源…

【Power BI】DAX语言 VS Power Query M语言

DAX&#xff08;Data Analysis Expressions&#xff09;和Power Query M语言是Microsoft Power BI和Excel中的两种强大的数据处理和分析工具。尽管它们在许多方面都有重叠之处&#xff0c;特别是用于数据建模和数据转换&#xff0c;但它们在用途、语法和功能上有显著的区别。本…

干部民主测评:深化管理智慧,凝聚团队力量

在现代化组织管理的广阔舞台上&#xff0c;干部民主测评扮演着举足轻重的角色。它不仅是评价干部工作实绩、能力素质的有力工具&#xff0c;更是推动组织向民主化、科学化、规范化迈进的强大引擎。通过民主测评&#xff0c;我们能够深入洞察每位干部的工作表现、群众基础和领导…