SemanticKITTI点云地图拼接 SemanticKITTI语义地图

news2025/1/15 6:26:08

 自己用点云分割的预测结果,和里程计的结果拼接出整个轨迹,某种程度上也算是语义语义地图

只需要pcl c++就可以,参了以下博主

kitti点云地图拼接_kitti点云拼接-CSDN博客

SemanticKITTI点云拼接+PCL可视化_可视化semantickitti .bin-CSDN博客

代码

#include <pcl/io/pcd_io.h>  // pcl::io::savePCDFileBinary
#include <pcl/common/transforms.h>  // pcl::transformPointCloud
#include <pcl/registration/ndt.h>  // voxelGridFilter
#include <pcl/filters/statistical_outlier_removal.h>
#include <unordered_map>
// #include <tf/tf.h>  // tf::Matrix3x3, tf::createQuaternionMsgFromRollPitchYaw, tf::Quaternion
// #include <ros/ros.h>

#include <iostream>
#include <string>
#include <vector>
#include <fstream>
#include <sstream>
#include <boost/filesystem.hpp>


typedef pcl::PointXYZL pointType;

static Eigen::Matrix4f tform;
std::vector<Eigen::Matrix4f> tforms;
std::string save_map_name = "output_map.pcd";
std::string filter_flag = "filter";
std::string label_path;
std::string bin_path, poses_file;
std::vector<std::string> bin_names;
std::vector<std::string> label_names;

pcl::PointCloud<pointType>::Ptr source_cloud (new pcl::PointCloud<pointType> ());
pcl::PointCloud<pointType>::Ptr transformed_cloud (new pcl::PointCloud<pointType> ());
pcl::PointCloud<pointType>::Ptr map_cloud (new pcl::PointCloud<pointType> ());

std::unordered_map<int,int> label_map={
                                        {	0	,	0	}	,
                                        {	1	,	0	}	,
                                        {	10	,	1	}	,
                                        {	11	,	2	}	,
                                        {	13	,	5	}	,
                                        {	15	,	3	}	,
                                        {	16	,	5	}	,
                                        {	18	,	4	}	,
                                        {	20	,	5	}	,
                                        {	30	,	6	}	,
                                        {	31	,	7	}	,
                                        {	32	,	8	}	,
                                        {	40	,	9	}	,
                                        {	44	,	10	}	,
                                        {	48	,	11	}	,
                                        {	49	,	12	}	,
                                        {	50	,	13	}	,
                                        {	51	,	14	}	,
                                        {	52	,	0	}	,
                                        {	60	,	9	}	,
                                        {	70	,	15	}	,
                                        {	71	,	16	}	,
                                        {	72	,	17	}	,
                                        {	80	,	18	}	,
                                        {	81	,	19	}	,
                                        {	99	,	0	}	,
                                        {	252	,	1	}	,
                                        {	253	,	7	}	,
                                        {	254	,	6	}	,
                                        {	255	,	8	}	,
                                        {	256	,	5	}	,
                                        {	257	,	5	}	,
                                        {	258	,	4	}	,
                                        {	259	,	5	}	
                                        };
std::unordered_map<int,int> learning_map_inv={
                                              {	0	,	0	}	,
                                              {	1	,	10	}	,
                                              {	2	,	11	}	,
                                              {	3	,	15	}	,
                                              {	4	,	18	}	,
                                              {	5	,	20	}	,
                                              {	6	,	30	}	,
                                              {	7	,	31	}	,
                                              {	8	,	32	}	,
                                              {	9	,	40	}	,
                                              {	10	,	44	}	,
                                              {	11	,	48	}	,
                                              {	12	,	49	}	,
                                              {	13	,	50	}	,
                                              {	14	,	51	}	,
                                              {	15	,	70	}	,
                                              {	16	,	71	}	,
                                              {	17	,	72	}	,
                                              {	18	,	80	}	,
                                              {	19	,	81	}	
                                              };
bool vstring_compare(const std::string &x,const std::string &y)  //&符号不能少
{
  return x < y;
}
//find bin list
void get_bin_names(const std::string& root_path, std::vector<std::string> &names)
{
    names.clear();
    boost::filesystem::path full_path(root_path);
    boost::filesystem::recursive_directory_iterator end_iter;
    for(boost::filesystem::recursive_directory_iterator iter(full_path); iter!=end_iter; ++iter)
    {
        try
        {
            if ( !boost::filesystem::is_directory( *iter ) )
            {
                std::string file = iter->path().string();
                names.push_back(iter->path().string());   // get the golbal full path name.  获取该文件的全局路径
                // boost::filesystem::path file_path(file);
                // names.push_back(file_path.stem().string());   // get the pure name(no suffix) 获取无后缀的文件名称,即000000, 000001, ...
            }
        }
        catch ( const std::exception & ex )
        {
            std::cerr << ex.what() << std::endl;
            continue;
        }
    }   
}
//find label list
void get_label_names(const std::string& root_path, std::vector<std::string> &names)
{
    names.clear();
    boost::filesystem::path full_path(root_path);
    boost::filesystem::recursive_directory_iterator end_iter;
    for(boost::filesystem::recursive_directory_iterator iter(full_path); iter!=end_iter; ++iter)
    {
        try
        {
            if ( !boost::filesystem::is_directory( *iter ) )
            {
                std::string file = iter->path().string();
                names.push_back(iter->path().string());   // get the golbal full path name.  获取该文件的全局路径
                // boost::filesystem::path file_path(file);
                // names.push_back(file_path.stem().string());   // get the pure name(no suffix) 获取无后缀的文件名称,即000000, 000001, ...
            }
        }
        catch ( const std::exception & ex )
        {
            std::cerr << ex.what() << std::endl;
            continue;
        }
    }   
}
// 解析得到对应的变换矩阵,并将方向转换到lidar所在坐标系下:
void get_transforms(std::string pose_file, std::vector<Eigen::Matrix4f>& tforms)
{
  std::string line;
  std::ifstream ifs;
  ifs.open(pose_file, std::ios::in);
  if (!ifs)
  {
    std::cout << "cannot open file: " << pose_file << std::endl;
    return ;
  }
  while (std::getline(ifs, line) && ifs.good())
  {
    if (line.empty()) return;
    std::stringstream lineStream(line);
    std::string cell;
    std::vector<double> vdata;
    while (std::getline(lineStream, cell, ' '))
    {
      vdata.push_back(std::stod(cell));
    }
    

    Eigen::Matrix4f tform;
    tform(0, 0) = vdata[0]; tform(0, 1) = vdata[1]; tform(0, 2) = vdata[2]; tform(0, 3) = vdata[3];
    tform(1, 0) = vdata[4]; tform(1, 1) = vdata[5]; tform(1, 2) = vdata[6]; tform(1, 3) = vdata[7];
    tform(2, 0) = vdata[8]; tform(2, 1) = vdata[9]; tform(2, 2) = vdata[10]; tform(2, 3) = vdata[11];
    tform(3, 0) = 0;        tform(3, 1) = 0;        tform(3, 2) = 0;        tform(3, 3) = 1;
    Eigen::Matrix4f t;
    t<<-1.857739385241e-03, -9.999659513510e-01, -8.039975204516e-03,-4.784029760483e-03,\
       -6.481465826011e-03,  8.051860151134e-03, -9.999466081774e-01, -7.337429464231e-02,\
        9.999773098287e-01, -1.805528627661e-03, -6.496203536139e-03, -3.339968064433e-01,\
        0                 , 0                  ,                   0,                   1;
    Eigen::Matrix4f t_inverse = t.inverse();
    tform = t_inverse*tform*t;
    
    
    // Eigen::Matrix4f tform = Eigen::Matrix4f::Identity();
    // Eigen::Matrix3f tf_mat;     // camera旋转矩阵
    // tf_mat << vdata[0], vdata[1], vdata[2], vdata[4], vdata[5], vdata[6], vdata[8], vdata[9], vdata[10];
    
    // // Eigen::Vector3f trans_camera(vdata[3], vdata[7], vdata[11]);
    // Eigen::Vector3f trans_camera(vdata[11], -vdata[3], -vdata[7]);
    // Eigen::Quaternionf q_camera(tf_mat);
    // Eigen::Quaternionf q_lidar(q_camera.w(), q_camera.z(), -q_camera.x(), -q_camera.y());
    // Eigen::Matrix3f R_lidar = q_lidar.toRotationMatrix();

    // tform.block<3,3>(0,0) = R_lidar;
    // tform.block<3,1>(0,3) = trans_camera;

    tforms.push_back(tform);
    // static int count = 0;
    // std::cout << count++ << "transform: \n" << tform << std::endl;
  }
}
// 解析.bin格式的点云文件,转换到标准的pcl::PointXYZI的格式: 
/*
void parse_bin_cloud(const std::string& bin_file, pcl::PointCloud<pointType>& points)
{
  points.points.clear();
  std::fstream input(bin_file.c_str(), std::ios::in | std::ios::binary);
  if(!input.good())
  {
    std::cerr << "Could not read file: " << bin_file << std::endl;
    exit(EXIT_FAILURE);
  }
  // bin2points:
  input.seekg(0, std::ios::beg);

  for (int i=0; input.good() && !input.eof(); i++)
  {
    pointType point;
    input.read((char *) &point.x, 3*sizeof(float));
    input.read((char *) &point.intensity, sizeof(float));
    points.points.push_back(point);
  }
  input.close();
}
*/
// 解析.bin格式的点云文件,转换到标准的pcl::PointXYZL 
int loadBin_with_Label(
    const std::string& binName, const std::string& labelName,
    pcl::PointCloud<pcl::PointXYZL>& p_cloud_l) {
  p_cloud_l.points.clear();
  int32_t num = 1000000;
  float* data = (float*)malloc(num * sizeof(float));
  uint32_t* data_label = (uint32_t*)malloc(num * sizeof(uint32_t));

  // 点
  float* px = data + 0;
  float* py = data + 1;
  float* pz = data + 2;
  float* pr = data + 3;           //反射强度
  uint32_t* pl = data_label + 0;  // label

  FILE* stream;
  FILE* label_stream;
  const char* filenameInput = binName.c_str();
  const char* labelnameInput = labelName.c_str();

  // fopen_s(&stream, filenameInput, "rb");      // fopen_s是微软的版本
  stream = fopen(filenameInput, "rb");
  label_stream = fopen(labelnameInput, "rb");

  num = fread(data, sizeof(float), num, stream) /4;  //读入点云数据,大概10万+个点
  auto r = fread(data_label, sizeof(uint32_t), num, label_stream);

  // std::cout << *data_label << std::endl;
  // std::cout << "num: " << num << ", label num: " << r << std::endl;
  // 创建点云
  // pcl::PointCloud<pcl::PointXYZI>::Ptr p_cloud_i(new
  // pcl::PointCloud<pcl::PointXYZI>());
  for (size_t i = 0; i < num; i++) {
    pcl::PointXYZL tmp;
    tmp.x = *px;
    tmp.y = *py;
    tmp.z = *pz;
    uint32_t upper_half = *pl >> 16;
    uint32_t lower_half = (*pl) & 0xffff;
    uint32_t label = upper_half << 16 + lower_half;
    lower_half = lower_half & 0x0000ffff;
    tmp.label = learning_map_inv[label_map[lower_half]];
    //todo
    
    // p_cloud_l->push_back(tmp);
    p_cloud_l.points.push_back(tmp);
    px += 4;
    py += 4;
    pz += 4;
    pr += 4;
    pl += 1;
  }
  fclose(stream);
  fclose(label_stream);

  return 1;
}
// 对点云进行将采用,减小对系统资源的占用,加快程序运行:
void voxel_grid_filter(const pcl::PointCloud<pointType>::Ptr& source_cloud, pcl::PointCloud<pointType>::Ptr& filtered_cloud, const double& voxel_leaf_size)
{
  pcl::VoxelGrid<pointType> voxel_filter;
  voxel_filter.setLeafSize(voxel_leaf_size, voxel_leaf_size, voxel_leaf_size);
  voxel_filter.setInputCloud(source_cloud);
  voxel_filter.filter(*filtered_cloud);
  return;
}
// 拼接点云地图
void joint_map(const std::vector<std::string>& bin_names, const std::vector<Eigen::Matrix4f>& tforms,const std::vector<std::string>& label_names)
{
  long long int cnt=0;
  for (int i = 0; i < bin_names.size(); ++i)
  {

    // convert kitti lidar data *.bin to pcl pointcloud type:
    // parse_bin_cloud(bin_names[i], *source_cloud);
    loadBin_with_Label(bin_names[i], label_names[i],*source_cloud);  //load_seg
    // std::cout <<i<<"get points: " << source_cloud->points.size() << std::endl;
    cnt += source_cloud->points.size();

    // transform the point cloud:
    pcl::transformPointCloud(*source_cloud, *transformed_cloud, tforms[i]);

    

    // //统计滤波
    if(filter_flag == "filter")
    {
      // std::cout<<i<<" before:"<<source_cloud->points.size()<<std::endl;
      pcl::StatisticalOutlierRemoval<pointType> sor;
      // 设置滤波器参数
      sor.setInputCloud(source_cloud);
      sor.setMeanK(50);  // 设置用于计算统计信息的临近点数
      sor.setStddevMulThresh(1.0);  // 设置标准差的倍数阈值
      sor.filter(*source_cloud);
      // std::cout<<i<<" after:"<<source_cloud->points.size()<<std::endl;
    }
    
    // voxel grid filter the cloud:
    if(filter_flag == "filter")
      voxel_grid_filter(transformed_cloud, transformed_cloud, 0.2);
    
    // 删除指定label的点云 0 未知
    // for (size_t i = 0; i < transformed_cloud->points.size(); ++i) {
    //     // 如果点的标签为10,则删除该点
    //     if (transformed_cloud->points[i].label == 10) {
    //         transformed_cloud->points.erase(transformed_cloud->points.begin() + i);
    //         // 删除点后,需要将索引回退一步
    //         --i;
    //     }
    // }

    if (i == 0)
    {
      *map_cloud = *transformed_cloud;
      continue;
    }
    *map_cloud += *transformed_cloud;
  }

  // voxel grid filter the output map:
  if(filter_flag == "filter")
    voxel_grid_filter(map_cloud, map_cloud, 0.9);
    //统计滤波
  if(filter_flag == "filter")
  {
    // std::cout<<" before:"<<map_cloud->points.size()<<std::endl;
    pcl::StatisticalOutlierRemoval<pointType> sor;
    // 设置滤波器参数
    sor.setInputCloud(map_cloud);
    sor.setMeanK(50);  // 设置用于计算统计信息的临近点数
    sor.setStddevMulThresh(1.5);  // 设置标准差的倍数阈值
    sor.filter(*map_cloud);
    // std::cout<<" after:"<<map_cloud->points.size()<<std::endl;
  }
  // save pointcloud to pcd:
  pcl::io::savePCDFileBinary(save_map_name, *map_cloud);
  // show save map info:
  std::cout << "the map " << save_map_name << " is saved with " <<  map_cloud->points.size() << " points" <<" // "<<cnt<< std::endl;
}


int main(int argc, char** argv) 
{

  if (argc < 3)
  {
    std::cout << "Usage: bin_name bin_path pose_name output_map_name(optional). " << std::endl;
    return 0;
  }
  bin_path = argv[1];
  poses_file = argv[2];
  save_map_name = argv[3];
  label_path = argv[4];
  filter_flag = argv[5];

  // ./concat 点云数据文件 位姿文件 输出文件名字 label文件

  get_bin_names(bin_path, bin_names);   // get the kitti lidar bin file names.
  sort(bin_names.begin(), bin_names.end(), vstring_compare);  // sort the names.(由于get_bin_names函数得到的文件路径是无序的,这里将它按照000000.bin, 000001.bin的顺序排列好)

  get_label_names(label_path, label_names);   // get the kitti lidar bin file names.
  sort(label_names.begin(), label_names.end(), vstring_compare);  // sort the names.(由于get_bin_names函数得到的文件路径是无序的,这里将它按照000000.bin, 000001.bin的顺序排列好)

  get_transforms(poses_file, tforms);  // get the kitti global poses and convert it to transform: Eigen::Matrix4f.
  // for (size_t i = 0; i < bin_names.size(); ++i)  // print the bin name and its affine matrix.
  // {
  //   std::cout << bin_names[i] << std::endl;
  //   std::cout << tforms[i] << std::endl;
  // }
  joint_map(bin_names, tforms, label_names);   // joint the map.

	return 0;
}

可视化结果

PCDViewer Release Page-CSDN博客

也可以用https://www.cloudcompare.org/

两个软件设置颜色略有不同,不过按照官方的颜色直接设置就行了,最终效果如下。

整体上看差不多,和原始论文差不多,由于自己建图的时候采用的前视角的点云和里程计的数据,因此有一点不准,如果直接使用原始的标签和位姿结果应该和原论文差不多。

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

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

相关文章

《人工智能怎么学》荣获2023年吴文俊人工智能科学技术奖及赠书活动

中国人工智能学会官网&#xff08;www.caai.cn&#xff09;近日正式公布了2023年吴文俊科学技术奖获奖名单&#xff0c;图书《人工智能怎么学》项目被授予2023年吴文俊人工智能科学技术奖科技进步奖&#xff08;科普项目&#xff09;。2023年吴文俊科学技术奖完整获奖名单见htt…

实现大华摄像头的抓图-使用HTTP方式

实现抓图&#xff0c;网上大部分都是使用SDK二次开发的&#xff0c;HTTP接口实现的基本没有介绍&#xff0c;好像官方叫CUI接口&#xff0c;但是找官方要文档&#xff0c;基本要不到&#xff0c;我自己下载了一份以前的文档&#xff0c;可以做大部分操作&#xff0c;这里免费分…

基于MPPT的太阳能光伏电池simulink性能仿真,对比扰动观察法,增量电导法,恒定电压法

目录 1.课题概述 2.系统仿真结果 3.核心程序与模型 4.系统原理简介 4.1 扰动观察法 (Perturb and Observe Method) 4.2 增量电导法 (Incremental Conductance Method) 4.3 恒定电压法 (Constant Voltage Method) 5.完整工程文件 1.课题概述 在simulink中&#xff0c;实…

RNN实现退位减法

文章目录 前言RNNRNN架构图前向传播公式反向传播算法 用RNN实现退位减法代码变量的对应关系 总结 前言 最近深入学习了一下RNN&#xff0c;即循环神经网络。RNN是一类比较基础的神经网络&#xff0c;本文使用的是最基础、最简单的循环神经网络的形式。LSTM也是一种常见的循环神…

中大型工厂人员定位系统源码,实现人、车、物的实时位置监控

UWB高精度定位系统源码&#xff0c;中大型工厂人员定位系统&#xff0c;实现人、车、物的实时位置监控 UWB高精度定位系统源码&#xff0c;智慧工厂人员定位系统源码&#xff0c;基于VueSpring boot前后端分离架构开发的一套UWB高精度定位系统源码。有演示。 随着经济的高速发展…

Flink实时数仓同步:切片表实战详解

一、背景 在大数据领域&#xff0c;初始阶段业务数据通常被存储于关系型数据库&#xff0c;如MySQL。然而&#xff0c;为满足日常分析和报表等需求&#xff0c;大数据平台采用多种同步方式&#xff0c;以适应这些业务数据的不同存储需求。 一项常见需求是&#xff0c;业务使用…

潜水耳机哪个牌子好?认准这几个游泳耳机品牌就对了!

在科技日益发达的今天&#xff0c;人们对于运动设备的需求也在不断提升。作为一项独特的水上运动&#xff0c;潜水爱好者们对耳机的要求也越来越高。一款优秀的潜水耳机不仅能够提供卓越的防水性能和舒适度&#xff0c;还必须具备出色的音质。那么&#xff0c;在众多品牌中&…

C语言进阶—表达式求值

隐式类型转换&#xff1a; C 的整型算术运算总是至少以缺省(默认)整型类型的精度来进行的。 为了获得这个精度&#xff0c;表达式中的字符和短整型操作数在使用之前被转换为普通整型&#xff0c;这种转换称为整型提升。 #include <stdio.h>int main() {char c 1;printf(…

鸿蒙Harmony应用开发—ArkTS声明式开发(自定义手势判定)

为组件提供自定义手势判定能力。开发者可根据需要&#xff0c;在手势识别期间&#xff0c;决定是否响应手势。 说明&#xff1a; 从API Version 11开始支持。后续版本如有新增内容&#xff0c;则采用上角标单独标记该内容的起始版本。 onGestureJudgeBegin onGestureJudgeBegi…

C#与python交互(flask发送Get/Post请求)

先运行python&#xff0c;再运行C# **ps: 注意修改端口号**python发送Get/Post请求 # -*- coding: utf-8 -*- # Time : 2024/1/25 15:52 # Author : YY # File : post_test.py # Content&#xff1a;提交数据给客户端 from flask import Flask, request, jsonify, redirect…

Vue系列-环境快速搭建

vue环境快速搭建 演示视频 快速搭建Vue开发环境pnpm和yarn 1. 基本信息 作者: GMCY系列: Vue仓库: GitHub | Gitee话题(GitHub): tools \ vue创建时间: 2024/03/02 2. 介绍 功能 批处理文件vue 环境的快速搭建nodejs, npm, pnpm, yarn 自动 下载安装npm, pnpm, yarn 自动 …

计网:HTTPS协议详解

1、HTTP 与 HTTPS 有哪些区别&#xff1f;​​​ HTTP以明文方式传输数据&#xff0c;不提供任何加密。如果攻击者截取了传输报文&#xff0c;就可以直接读取其中的信息。HTTPS利用SSL/TLS加密数据包&#xff0c;报文以密文方式传输。 HTTP 连接建立相对简单&#xff0c; TCP …

【办公类-22-08】周计划系列(3-3)“信息窗+主题知识(上传+打印)” (2024年调整版本)

作品展示 背景需求&#xff1a; 前文将信息窗主题知识的内容提取并优化结构 【办公类-22-07】周计划系列&#xff08;3-1&#xff09;“信息窗主题知识&#xff08;提取&#xff09;” &#xff08;2024年调整版本&#xff09;-CSDN博客文章浏览阅读803次&#xff0c;点赞7次…

新一代WLAN解决方案与WLAN的配置实现

案例背景为二层旁挂式组网&#xff0c;转发方式为直接转发&#xff0c;管理Vlan为100&#xff0c;业务Vlan为101。 基本配置&#xff1a; SW1&#xff1a; [SW1]VLAN batch 100 101 [SW1-GigabitEthernet0/0/1]port link-type trunk [SW1-GigabitEthernet0/0/1]port trunk a…

SQL 初级

SQL 初级 SQL 简介 SQL (Structured Query Language:结构化查询语言) 是用于管理关系数据库管理系统&#xff08;RDBMS&#xff09;。 SQL 的范围包括数据插入、查询、更新和删除&#xff0c;数据库模式创建和修改&#xff0c;以及数据访问控制。 SQL 是什么&#xff1f; SQL…

【Linux C | 网络编程】多播的概念、多播地址、UDP实现多播的C语言例子

&#x1f601;博客主页&#x1f601;&#xff1a;&#x1f680;https://blog.csdn.net/wkd_007&#x1f680; &#x1f911;博客内容&#x1f911;&#xff1a;&#x1f36d;嵌入式开发、Linux、C语言、C、数据结构、音视频&#x1f36d; &#x1f923;本文内容&#x1f923;&a…

提高生产效率!虹科MSR165快速检测机器故障,实现精准优化

如今&#xff0c;各种生产机器、机床和运输机都采用了复杂的驱动技术&#xff0c;以便在三个轴上准确生成线性运动或者高效旋转运动。所有机械运动中都会出现特征性的振动或震动模式&#xff0c;可以在该背景下利用这些模式来监测和优化整个驱动技术的机电参数。在这个过程中&a…

太阳能光伏电池的simulink建模与仿真

目录 1.课题概述 2.系统仿真结果 3.核心程序与模型 4.系统原理简介 4.1 光伏电池的基本结构 4.2 光伏电池的工作原理 5.完整工程文件 1.课题概述 太阳能光伏电池的simulink建模与仿真.分析不同光照温度&#xff0c;光照强度下的光伏电池的U-I特性曲线以及P-V特性曲线。 …

备战蓝桥(模板篇)

扩展欧德里几算法 质数筛 分解质因数 LCA BFS floyd Dijkstra prime 日期是否合法 Tire异或 模拟散列表 字符哈希 Tire字符串统计

官方教程 | 在 OpenBayes 平台进行组织协作

想和好 homie 共享账户余额、存储、数据集、模型、容器等资源&#xff0c;又不想共享自己的账户密码&#xff1f; 想跟团队成员分工协作、高效 Coding、加速炼丹&#xff0c;又想隔离权限、差异化管理&#xff1f; 经过为期半年的内测和完善&#xff0c;OpenBayes贝式计算的组织…