在不同操作系统上自动生成Protocol Buffers的Java语言包的方法

各语言的Protocol Buffers文件都需要通过protoc来生成,这个动作往往需要手动输入命令完成。本文介绍的方法,将借助Maven来实现自动化生成工作。这样开发者只要专注于proto的定义,且不用将生成的文件上传到代码仓库,从而降低开发的复杂度。

Protocol Buffers介绍

Protocol Buffers是一个强大的数据序列化工具,它提供了一种高效、便捷、可读性强且安全性高的方式来处理结构化数据。它能够将复杂的数据结构转换为紧凑的二进制格式,从而方便地进行网络传输或硬盘存储。接收方可以使用相同的数据结构定义来解析这些二进制数据,从而还原成原始的数据。
ProtoBuf的用途广泛,特别适用于需要频繁处理数据的场景,如网络通信和数据存储。在网络通信中,ProtoBuf可以帮助开发者在不同系统和平台之间实现高效、可靠的数据交换和通信。而在数据存储方面,ProtoBuf则提供了一种紧凑、可移植的数据表示方式,使得数据的读写和存储变得更加高效和便捷。
特别是在多语言开发环境下,不同语言可以通过Protocol Buffers描述文件生成各自语言的代码,从而实现:一套定义,多语言便捷使用的目的。
本文我们将介绍如果使用Maven自动生成Java语言包。

环境

首先介绍下测试环境

Windows 10

Java

java.exe -version

openjdk version “21.0.2” 2024-01-16
OpenJDK Runtime Environment (build 21.0.2+13-58)
OpenJDK 64-Bit Server VM (build 21.0.2+13-58, mixed mode, sharing)

Maven

IntelliJ捆绑的Maven3.9.5
在这里插入图片描述

Ubuntu TLS 22

Java

java -version

openjdk version “21.0.2” 2024-01-16
OpenJDK Runtime Environment (build 21.0.2+13-Ubuntu-122.04.1)
OpenJDK 64-Bit Server VM (build 21.0.2+13-Ubuntu-122.04.1, mixed mode, sharing)

Maven

mvn -version

Apache Maven 3.6.3
Maven home: /usr/share/maven
Java version: 21.0.2, vendor: Private Build, runtime: /usr/lib/jvm/java-21-openjdk-amd64
Default locale: en_US, platform encoding: UTF-8
OS name: “linux”, version: “5.15.0-105-generic”, arch: “amd64”, family: “unix”

准备工作

目录结构

├── pom.xml
└── src
    └── main
        ├── java
        │   ├── org
        │   │   └── example
        │   │       ├── AddressBuilder.java
        │   │       ├── Main.java
        │   │       ├── PersonBuilder.java
        │   │       ├── RequestBuilder.java
        │   │       └── ResponseBuilder.java
        └── proto
            ├── person.proto
            ├── request.proto
            └── response.proto

proto目录放置的是我们需要被编码为Java语言的Protocol Buffers文件。

pom.xml的配置

protoc

protoc是将proto文件转译成各种编程语言对应的源码的工具,所以这个工具一定是要使用的。只是我们不希望开发人员自己关注该工具的维护,而是统一在pom.xml中自动维护。protobuf-maven-plugin这个插件就提供了这个功能。

维护protoc的插件

<plugin>
     <groupId>org.xolstice.maven.plugins</groupId>
     <artifactId>protobuf-maven-plugin</artifactId>
     <version>0.6.1</version>
     <extensions>true</extensions>
     <executions>
         <execution>
             <goals>
                 <goal>compile</goal>
                 <goal>test-compile</goal>
             </goals>
         </execution>
     </executions>
</plugin>

指定读取的proto文件位置

additionalProtoPathElements可以指定一系列proto文件路径的位置。本例中我们的proto都在一个文件夹下,所以只用设定一个additionalProtoPathElement就行。借助这个属性,我们可以在复杂的项目中,管理多个proto文件路径。

<configuration>
    <additionalProtoPathElements>
        <additionalProtoPathElement>${project.basedir}/src/main/proto</additionalProtoPathElement>
    </additionalProtoPathElements>
</configuration>

指定生成路径

假如我们希望生成的文件不在target目录下,则可以考虑该指定protoc的产出路径。

 <configuration>
         <outputDirectory>${project.basedir}/src/main/java/protojava</outputDirectory>
</configuration>

这样最终我们的目录会变成如下结构

├── pom.xml
└── src
    └── main
        ├── java
        │   ├── org
        │   │   └── example
        │   │       ├── AddressBuilder.java
        │   │       ├── Main.java
        │   │       ├── PersonBuilder.java
        │   │       ├── RequestBuilder.java
        │   │       └── ResponseBuilder.java
        │   └── protojava
        │       └── com
        │           └── proto
        │               └── message
        │                   └── gen
        │                       └── proto
        │                           ├── AcademicDiplomas.java
        │                           ├── Address.java
        │                           ├── AddressOrBuilder.java
        │                           ├── Person.java
        │                           ├── PersonOrBuilder.java
        │                           ├── PersonOuterClass.java
        │                           ├── RequestOuterClass.java
        │                           └── ResponseOuterClass.java
        └── proto
            ├── person.proto
            ├── request.proto
            └── response.proto

多操作系统支持

现实中,一个项目的开发人员可能因为不同的习惯而需要在不同操作系统上进行开发,比如Windows、Mac或Linux。这些操作系统的可执行程序的文件格式不一样,这样就需要不同protoc来支持。为了完成这个功能,我们需要引入${os.detected.classifier}来识别操作系统。

[INFO] os.detected.classifier: windows-x86_64
[INFO] os.detected.classifier: linux-x86_64

具体的配置如下:

<configuration>
     <protocArtifact>com.google.protobuf:protoc:3.7.6:exe:${os.detected.classifier}</protocArtifact>
</configuration>

要使用${os.detected.classifier}还需要os-maven-plugin

<extensions>
    <extension>
        <groupId>kr.motd.maven</groupId>
        <artifactId>os-maven-plugin</artifactId>
        <version>1.7.1</version>
    </extension>
</extensions>

指定protobuf-java的版本

protoc生成文件只是proto文件的解释,而不会包含更底层的Protocol Buffers代码。
比如本例中

enum AcademicDiplomas {
    POSTGRADUATE = 0;
    BACHELOR = 1;
    JUNIOR_COLLEGE = 2;
    VOCATIONAL = 3;
    SENIOR = 4;
    JUNIOR = 5; 
}

生成的代码如下

package com.proto.message.gen.proto;

/**
 * Protobuf enum {@code AcademicDiplomas}
 */
public enum AcademicDiplomas
    implements com.google.protobuf.ProtocolMessageEnum {
  /**
   * <code>POSTGRADUATE = 0;</code>
   */
  POSTGRADUATE(0),
  /**
   * <code>BACHELOR = 1;</code>
   */
  BACHELOR(1),
  /**
   * <code>JUNIOR_COLLEGE = 2;</code>
   */
  JUNIOR_COLLEGE(2),
  /**
   * <code>VOCATIONAL = 3;</code>
   */
  VOCATIONAL(3),
  /**
   * <code>SENIOR = 4;</code>
   */
  SENIOR(4),
  /**
   * <code>JUNIOR = 5;</code>
   */
  JUNIOR(5),
  UNRECOGNIZED(-1),
  ;

  static {
    com.google.protobuf.RuntimeVersion.validateProtobufGencodeVersion(
      com.google.protobuf.RuntimeVersion.RuntimeDomain.PUBLIC,
      /* major= */ 4,
      /* minor= */ 26,
      /* patch= */ 1,
      /* suffix= */ "",
      AcademicDiplomas.class.getName());
  }
  /**
   * <code>POSTGRADUATE = 0;</code>
   */
  public static final int POSTGRADUATE_VALUE = 0;
  /**
   * <code>BACHELOR = 1;</code>
   */
  public static final int BACHELOR_VALUE = 1;
  /**
   * <code>JUNIOR_COLLEGE = 2;</code>
   */
  public static final int JUNIOR_COLLEGE_VALUE = 2;
  /**
   * <code>VOCATIONAL = 3;</code>
   */
  public static final int VOCATIONAL_VALUE = 3;
  /**
   * <code>SENIOR = 4;</code>
   */
  public static final int SENIOR_VALUE = 4;
  /**
   * <code>JUNIOR = 5;</code>
   */
  public static final int JUNIOR_VALUE = 5;


  public final int getNumber() {
    if (this == UNRECOGNIZED) {
      throw new java.lang.IllegalArgumentException(
          "Can't get the number of an unknown enum value.");
    }
    return value;
  }

  /**
   * @param value The numeric wire value of the corresponding enum entry.
   * @return The enum associated with the given numeric wire value.
   * @deprecated Use {@link #forNumber(int)} instead.
   */
  @java.lang.Deprecated
  public static AcademicDiplomas valueOf(int value) {
    return forNumber(value);
  }

  /**
   * @param value The numeric wire value of the corresponding enum entry.
   * @return The enum associated with the given numeric wire value.
   */
  public static AcademicDiplomas forNumber(int value) {
    switch (value) {
      case 0: return POSTGRADUATE;
      case 1: return BACHELOR;
      case 2: return JUNIOR_COLLEGE;
      case 3: return VOCATIONAL;
      case 4: return SENIOR;
      case 5: return JUNIOR;
      default: return null;
    }
  }

  public static com.google.protobuf.Internal.EnumLiteMap<AcademicDiplomas>
      internalGetValueMap() {
    return internalValueMap;
  }
  private static final com.google.protobuf.Internal.EnumLiteMap<
      AcademicDiplomas> internalValueMap =
        new com.google.protobuf.Internal.EnumLiteMap<AcademicDiplomas>() {
          public AcademicDiplomas findValueByNumber(int number) {
            return AcademicDiplomas.forNumber(number);
          }
        };

  public final com.google.protobuf.Descriptors.EnumValueDescriptor
      getValueDescriptor() {
    if (this == UNRECOGNIZED) {
      throw new java.lang.IllegalStateException(
          "Can't get the descriptor of an unrecognized enum value.");
    }
    return getDescriptor().getValues().get(ordinal());
  }
  public final com.google.protobuf.Descriptors.EnumDescriptor
      getDescriptorForType() {
    return getDescriptor();
  }
  public static final com.google.protobuf.Descriptors.EnumDescriptor
      getDescriptor() {
    return com.proto.message.gen.proto.PersonOuterClass.getDescriptor().getEnumTypes().get(0);
  }

  private static final AcademicDiplomas[] VALUES = values();

  public static AcademicDiplomas valueOf(
      com.google.protobuf.Descriptors.EnumValueDescriptor desc) {
    if (desc.getType() != getDescriptor()) {
      throw new java.lang.IllegalArgumentException(
        "EnumValueDescriptor is not for this type.");
    }
    if (desc.getIndex() == -1) {
      return UNRECOGNIZED;
    }
    return VALUES[desc.getIndex()];
  }

  private final int value;

  private AcademicDiplomas(int value) {
    this.value = value;
  }

  // @@protoc_insertion_point(enum_scope:AcademicDiplomas)
}

生成的AcademicDiplomas要继承于com.google.protobuf.ProtocolMessageEnum,而后者不存在于当前代码中。这就需要引入包含这些底层代码的其他依赖,比如protobuf-java。这些依赖有版本号,也就意味着protoc也要与之适配。这样我们就可以将版本号提出来,作为属性供后面各个依赖以及protoc来使用。

<properties>
     <protobuf-java.version>4.26.1</protobuf-java.version>
</properties>
<configuration>
    <protocArtifact>com.google.protobuf:protoc:${protobuf-java.version}:exe:${os.detected.classifier}</protocArtifact>
</configuration>

引入依赖

这些依赖就是提供Protocol Buffers Java的底层代码,比如com.google.protobuf.ProtocolMessageEnum、com.google.protobuf.GeneratedMessage和com.google.protobuf.MessageOrBuilder之类。

<dependencies>
    <dependency>
        <groupId>com.google.protobuf</groupId>
        <artifactId>protobuf-java</artifactId>
        <version>${protobuf-java.version}</version>
    </dependency>
    <dependency>
        <groupId>com.google.protobuf</groupId>
        <artifactId>protobuf-java-util</artifactId>
        <version>${protobuf-java.version}</version>
    </dependency>
</dependencies>

整个文件

整体来说,dependencies部分提供底层代码依赖;build部分用于自动生成proto各个操作系统上的Java文件包。

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>org.example</groupId>
    <artifactId>proto-message-gen</artifactId>
    <version>1.0-SNAPSHOT</version>

    <properties>
        <maven.compiler.source>21</maven.compiler.source>
        <maven.compiler.target>21</maven.compiler.target>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <protobuf-java.version>4.26.1</protobuf-java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>com.google.protobuf</groupId>
            <artifactId>protobuf-java</artifactId>
            <version>${protobuf-java.version}</version>
        </dependency>
        <dependency>
            <groupId>com.google.protobuf</groupId>
            <artifactId>protobuf-java-util</artifactId>
            <version>${protobuf-java.version}</version>
        </dependency>
    </dependencies>

    <build>
        <extensions>
            <extension>
                <groupId>kr.motd.maven</groupId>
                <artifactId>os-maven-plugin</artifactId>
                <version>1.7.1</version>
            </extension>
        </extensions>
        <plugins>
            <plugin>
                <groupId>org.xolstice.maven.plugins</groupId>
                <artifactId>protobuf-maven-plugin</artifactId>
                <version>0.6.1</version>
                <extensions>true</extensions>
                <executions>
                    <execution>
                        <goals>
                            <goal>compile</goal>
                            <goal>test-compile</goal>
                        </goals>
                    </execution>
                </executions>
                <configuration>
                    <additionalProtoPathElements>
                        <additionalProtoPathElement>${project.basedir}/src/main/proto</additionalProtoPathElement>
                    </additionalProtoPathElements>
                    <outputDirectory>${project.basedir}/src/main/java/protojava</outputDirectory>
                    <protocArtifact>com.google.protobuf:protoc:${protobuf-java.version}:exe:${os.detected.classifier}</protocArtifact>
                </configuration>
            </plugin>
        </plugins>
    </build>

</project>

测试

person.proto

proto

syntax = "proto3";

option java_multiple_files = true;
option java_package = "com.proto.message.gen.proto";

message Address {
    string city = 1;
    string street = 2;
    string door = 3;
}

message Person {
    string name = 1;
    int32 age = 2;
    float weight = 3;
    AcademicDiplomas academicdiplomas = 4;
    repeated Address address = 5;
}

enum AcademicDiplomas {
    POSTGRADUATE = 0;
    BACHELOR = 1;
    JUNIOR_COLLEGE = 2;
    VOCATIONAL = 3;
    SENIOR = 4;
    JUNIOR = 5; 
}

java

// AddressBuilder.java
package org.example;
import com.proto.message.gen.proto.Address;

public class AddressBuilder {
    public Address buildAddress(String city, String street, String door) {
        Address.Builder addressBuilder = Address.newBuilder();
        addressBuilder.setCity(city);
        addressBuilder.setStreet(street);
        addressBuilder.setDoor(door);
        return addressBuilder.build();
    }
}
// PersonBuilder.java
package org.example;
import com.proto.message.gen.proto.Person;
import com.proto.message.gen.proto.Address;

public class PersonBuilder {
    public Person buildPerson(String name, int age) {
        AddressBuilder addressBuilder = new AddressBuilder();
        Person.Builder builder = Person.newBuilder();
        builder.setName(name);
        builder.setAge(age);
        builder.setWeight(70.5F);

        for (int i = 1; i <= 5; i++) {
            Address address = addressBuilder.buildAddress("city" + i, "street" + i, "door" + i);
            builder.addAddress(address);
        }
        return builder.build();
    }
}

request.proto

proto

syntax = "proto3";

option java_package = "com.proto.message.gen.proto";

message Request {
    string name = 1;
    int32 age = 2;
}

java

// RequestBuilder.java
package org.example;
import com.proto.message.gen.proto.RequestOuterClass;

public class RequestBuilder {
    public RequestOuterClass.Request buildRequest() {
        RequestOuterClass.Request.Builder builder = RequestOuterClass.Request.newBuilder();
        builder.setName("Bob");
        builder.setAge(24);
        return builder.build();
    }
}

reponse.proto

proto

syntax = "proto3";

option java_package = "com.proto.message.gen.proto";

import "person.proto";

message Response {
    repeated Person peoples = 1;
}

java

// ResponseBuilder.java
package org.example;

import com.proto.message.gen.proto.ResponseOuterClass;
import com.proto.message.gen.proto.Person;
public class ResponseBuilder {
    public ResponseOuterClass.Response buildResponse() {
        ResponseOuterClass.Response.Builder builder = ResponseOuterClass.Response.newBuilder();
        for (int i = 1; i <= 5; i++) {
            Person person = new PersonBuilder().buildPerson("name" + i, i);
            builder.addPeoples(person);
        }
        return builder.build();
    }
}

代码

https://github.com/f304646673/proto-gen.git

参考资料

  • https://www.xolstice.org/protobuf-maven-plugin/
  • https://www.xolstice.org/protobuf-maven-plugin/compile-mojo.html
  • https://stackoverflow.com/questions/67577800/how-to-use-google-protobuf-compiler-with-maven-compiler-plugin

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

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

相关文章

Tracecat:开源 SOAR

Tracecat 是一个面向安全团队的开源自动化平台。 开发人员认为&#xff0c;每个人都应该可以使用安全自动化&#xff0c;特别是人手不足的中小型团队。 核心功能、用户界面和日常工作流程基于一流安全团队的现有最佳实践。 使用专门的人工智能模型来标记、总结和丰富警报。 …

【软件开发规范篇】JAVA后端开发编码注释规范

作者介绍&#xff1a;本人笔名姑苏老陈&#xff0c;从事JAVA开发工作十多年了&#xff0c;带过大学刚毕业的实习生&#xff0c;也带过技术团队。最近有个朋友的表弟&#xff0c;马上要大学毕业了&#xff0c;想从事JAVA开发工作&#xff0c;但不知道从何处入手。于是&#xff0…

20240502在WIN11下显示桌面

20240502在WIN11下显示桌面 2024/5/2 15:06 百度&#xff1a;WIN11 状态栏 右键 显示桌面 在WIN10下&#xff0c;可以在状态栏点击右键→“显示桌面”来最小化全部窗口&#xff0c;特别是我打开的浏览器的巨多的窗口&#xff01; 但是在WIN11下&#xff0c;这个【显示桌面】怎…

2024五一杯数学建模B题思路代码文章教学-交通需求规划与可达率问题

交通需求规划与可达率问题 问题总结&#xff1a; 问题一&#xff1a;在一个小型交通网络中&#xff0c;给定的起点和终点之间的交通需求需分配到相应路径上。目标是最大化任意一条路段出现突发状况时的交通需求期望可达率。 问题二&#xff1a;在一个较大的交通网络中&#xff…

Linux-进程调度器

1. 前言 在计算机中&#xff0c;进程的数量远多于cpu的数量&#xff0c;所以就存在&#xff0c;多个进程抢占一个cpu的情况&#xff0c;所以就需要一套规则&#xff0c;决定这些进程被处理的顺序&#xff0c;这就叫做进程调度。 在我的简单理解下&#xff0c;其实就是把进程放…

普乐蛙景区vr体验馆VR游乐场设备身历其境体验

小编给大家推荐一款gao坪效产品【暗黑战车】&#xff0c;一次6人同乘&#xff0c;炫酷外观、强大性能和丰富内容适合各个年龄层客群&#xff0c;紧张刺激的VR体验让玩家沉浸在元宇宙的魅力中&#xff0c;无论是节假日还是平日&#xff0c;景区商场助力门店提高客流量和营收~ ◆…

实验三 .Java 语言继承和多态应用练习 (课内实验)

一、实验目的 本次实验的主要目的是通过查看程序的运行结果及实际编写程序&#xff0c;练习使用 Java 语言的继承特性。 二、实验要求 1. 认真阅读实验内容&#xff0c;完成实验内容所设的题目 2. 能够应用多种编辑环境编写 JAVA 语言源程序 3. 认真体会多态与继承的作用…

B+树详解与实现

B树详解与实现 一、引言二、B树的定义三、B树的插入四、B树的删除五、B树的查找效率六、B树与B树的区别和联系 一、引言 B树是一种树数据结构&#xff0c;通常用于数据库和操作系统的文件系统中。它的特点是能够保持数据稳定有序&#xff0c;其插入与修改拥有较稳定的对数时间…

WebGL/Cesium 大空间相机抖动 RTE(Relative to Eye)实现原理简析

在浏览器中渲染大尺寸 3D 模型&#xff1a;Speckle 处理空间抖动的方法 WebGL/Cesium 大空间相机抖动 RTE(Relative to Eye)实现原理简析 注: 相机空间和视图空间 概念等效混用 1、实现的关键代码 const material new THREE.RawShaderMaterial({uniforms: {cameraPostion: {…

【Qt QML】用CMake管理Qt工程

CMake是一个开源、跨平台的工具系列&#xff0c;用于构建、测试和打包软件。CMake使用简单的独立配置文件来控制软件编译过程。与许多跨平台系统不同&#xff0c;CMake被设计为与本地构建环境结合使用。 下面我们在CMake项目中使用Qt的最基本方法。首先&#xff0c;创建一个基本…

如何解决pycharm创建项目报错 Error occurred when installing package ‘requests‘. Details.

&#x1f42f; 如何解决PyCharm创建项目时的包安装错误&#xff1a;‘requests’ &#x1f6e0;️ 文章目录 &#x1f42f; 如何解决PyCharm创建项目时的包安装错误&#xff1a;requests &#x1f6e0;️摘要引言正文&#x1f4d8; **问题分析**&#x1f680; **更换Python版本…

OpenCV 实现重新映射(53)

返回:OpenCV系列文章目录&#xff08;持续更新中......&#xff09; 上一篇&#xff1a;OpenCV 实现霍夫圆变换(52) 下一篇 :OpenCV实现仿射变换(54) 目标 在本教程中&#xff0c;您将学习如何&#xff1a; 一个。使用 OpenCV 函数 cv&#xff1a;&#xff1a;remap 实现简…

Java Web 开发 - 掌握拦截器和监听器

目录 深入了解Java Web的拦截器和监听器 拦截器&#xff08;Interceptor&#xff09; 拦截器的使用场景 拦截器实例 思维导图 ​编辑 监听器&#xff08;Listener&#xff09; 监听器的使用场景 监听器类型 监听器实例 思维导图​编辑 总结 深入了解Java Web的拦截器…

C——双向链表

一.链表的概念及结构 链表是一种物理存储单元上非连续、非顺序的存储结构&#xff0c;数据元素的逻辑顺序是通过链表中的指针链接次序实现的。什么意思呢&#xff1f;意思就是链表在物理结构上不一定是连续的&#xff0c;但在逻辑结构上一定是连续的。链表是由一个一个的节点连…

uniapp0基础编写安卓原生插件和调用第三方jar包和编写语音播报插件之使用jar包插件

前言 如果你不会编写安卓插件,你可以先看看我之前零基础的文章(uniapp0基础编写安卓原生插件和调用第三方jar包和编写语音播报插件之零基础编写安卓插件), 我们使用第三方包,jar包编写安卓插件 开始 把依赖包,放到某个模块的/libs目录(myTestPlug/libs) 还要到build…

java-函数式编程-函数对象

定义 什么是合格的函数&#xff1f;无论多少次执行函数&#xff0c;只要输入一样&#xff0c;输出就不会改变 对象方法的简写 其实在类中&#xff0c;我们很多参数中都有一个this&#xff0c;被隐藏传入了 函数也可以作为对象传递&#xff0c;lambda就是很好的例子 函数式接口中…

ROS实操:通信机制的实现

最近闲来无事&#xff0c;打算重温了一下ROS方面的相关知识。先前的学习都是一带而过&#xff0c;发现差不多都忘了&#xff0c;学习的不够深入。因此&#xff0c;在重温的同时&#xff0c;写下了这篇关于ROS架构的学习博客。 上一篇博客的链接为&#xff1a;ROS架构的学习【No…

如何利用有限的数据发表更多的SCI论文?——利用ArcGIS探究环境和生态因子对水体、土壤和大气污染物的影响

原文链接&#xff1a;如何利用有限的数据发表更多的SCI论文&#xff1f;——利用ArcGIS探究环境和生态因子对水体、土壤和大气污染物的影响https://mp.weixin.qq.com/s?__bizMzUzNTczMDMxMg&mid2247602528&idx6&snc89e862270fe54239aa4f796af07fb71&chksmfa82…

visio画图基本用法

添加图形 点击上面的箭头 添加一些基本的形状 添加连接点 点击这个 X 按住Ctrl&#xff0c;在想要的位置上添加连接点 更改线条样式 选中线条之后&#xff0c;右键 可以选择箭头样式 添加文本框 visio对象复制到word里面&#xff0c;画布存在大量空白问题 https://blog.…

【C语言】深入了解文件:简明指南

&#x1f308;个人主页&#xff1a;是店小二呀 &#x1f308;C语言笔记专栏&#xff1a;C语言笔记 &#x1f308;C笔记专栏&#xff1a; C笔记 &#x1f308;喜欢的诗句:无人扶我青云志 我自踏雪至山巅 文章目录 一、文件的概念1.1 文件名:1.2 程序文件和数据文件 二、数据文…