STM32智能物流机器人系统教程

news2024/9/24 23:30:53

目录

  1. 引言
  2. 环境准备
  3. 智能物流机器人系统基础
  4. 代码实现:实现智能物流机器人系统 4.1 数据采集模块 4.2 数据处理与导航算法 4.3 通信与网络系统实现 4.4 用户界面与数据可视化
  5. 应用场景:物流机器人管理与优化
  6. 问题解决方案与优化
  7. 收尾与总结

1. 引言

智能物流机器人系统通过STM32嵌入式系统结合各种传感器、执行器和通信模块,实现对物流机器人的实时监控、路径规划和自动控制。本文将详细介绍如何在STM32系统中实现一个智能物流机器人系统,包括环境准备、系统架构、代码实现、应用场景及问题解决方案和优化方法。

2. 环境准备

硬件准备

  1. 开发板:STM32F4系列或STM32H7系列开发板
  2. 调试器:ST-LINK V2或板载调试器
  3. 传感器:如IMU、超声波传感器、红外传感器、激光雷达等
  4. 执行器:如直流电机、步进电机、舵机等
  5. 通信模块:如以太网模块、Wi-Fi模块、蓝牙模块等
  6. 显示屏:如OLED显示屏
  7. 按键或旋钮:用于用户输入和设置
  8. 电源:电池组或电源适配器

软件准备

  1. 集成开发环境(IDE):STM32CubeIDE或Keil MDK
  2. 调试工具:STM32 ST-LINK Utility或GDB
  3. 库和中间件:STM32 HAL库和FreeRTOS

安装步骤

  1. 下载并安装STM32CubeMX
  2. 下载并安装STM32CubeIDE
  3. 配置STM32CubeMX项目并生成STM32CubeIDE项目
  4. 安装必要的库和驱动程序

3. 智能物流机器人系统基础

控制系统架构

智能物流机器人系统由以下部分组成:

  1. 数据采集模块:用于采集机器人的位置、姿态和环境数据
  2. 数据处理与导航算法模块:对采集的数据进行处理和分析,执行导航和路径规划算法
  3. 通信与网络系统:实现机器人与服务器或其他设备的通信
  4. 显示系统:用于显示系统状态和路径信息
  5. 用户输入系统:通过按键或旋钮进行设置和调整

功能描述

通过各种传感器采集机器人的关键数据,并实时显示在OLED显示屏上。系统通过PID控制算法和网络通信,实现对机器人的自动化控制和路径规划。用户可以通过按键或旋钮进行设置,并通过显示屏查看当前状态。

4. 代码实现:实现智能物流机器人系统

4.1 数据采集模块

配置IMU

使用STM32CubeMX配置I2C接口:

  1. 打开STM32CubeMX,选择您的STM32开发板型号。
  2. 在图形化界面中,找到需要配置的I2C引脚,设置为I2C模式。
  3. 生成代码并导入到STM32CubeIDE中。

代码实现:

#include "stm32f4xx_hal.h"
#include "i2c.h"
#include "mpu6050.h"

I2C_HandleTypeDef hi2c1;

void I2C1_Init(void) {
    hi2c1.Instance = I2C1;
    hi2c1.Init.ClockSpeed = 100000;
    hi2c1.Init.DutyCycle = I2C_DUTYCYCLE_2;
    hi2c1.Init.OwnAddress1 = 0;
    hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT;
    hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE;
    hi2c1.Init.OwnAddress2 = 0;
    hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE;
    hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE;
    HAL_I2C_Init(&hi2c1);
}

void Read_IMU_Data(float* accel, float* gyro) {
    MPU6050_ReadAll(accel, gyro);
}

int main(void) {
    HAL_Init();
    SystemClock_Config();
    I2C1_Init();
    MPU6050_Init();

    float accel[3], gyro[3];

    while (1) {
        Read_IMU_Data(accel, gyro);
        HAL_Delay(100);
    }
}
配置超声波传感器

使用STM32CubeMX配置GPIO接口和定时器:

  1. 打开STM32CubeMX,选择您的STM32开发板型号。
  2. 在图形化界面中,找到需要配置的GPIO引脚,设置为输出模式(触发引脚)和输入模式(回响引脚)。
  3. 配置定时器用于测量超声波传感器的回波时间。
  4. 生成代码并导入到STM32CubeIDE中。

代码实现:

#include "stm32f4xx_hal.h"

TIM_HandleTypeDef htim2;
GPIO_InitTypeDef GPIO_InitStruct = {0};

void GPIO_Init(void) {
    __HAL_RCC_GPIOA_CLK_ENABLE();

    // 配置触发引脚
    GPIO_InitStruct.Pin = GPIO_PIN_0;
    GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

    // 配置回响引脚
    GPIO_InitStruct.Pin = GPIO_PIN_1;
    GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
}

void TIM2_Init(void) {
    __HAL_RCC_TIM2_CLK_ENABLE();

    htim2.Instance = TIM2;
    htim2.Init.Prescaler = 84 - 1;
    htim2.Init.CounterMode = TIM_COUNTERMODE_UP;
    htim2.Init.Period = 0xFFFFFFFF;
    htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
    HAL_TIM_Base_Init(&htim2);
    HAL_TIM_Base_Start(&htim2);
}

uint32_t Read_Ultrasonic_Distance(void) {
    // 发送触发信号
    HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_SET);
    HAL_Delay(10);
    HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_RESET);

    // 等待回响信号
    while (HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_1) == GPIO_PIN_RESET);

    // 计时开始
    uint32_t start_time = __HAL_TIM_GET_COUNTER(&htim2);

    // 等待回响信号结束
    while (HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_1) == GPIO_PIN_SET);

    // 计时结束
    uint32_t end_time = __HAL_TIM_GET_COUNTER(&htim2);

    // 计算距离
    return (end_time - start_time) * 0.034 / 2;
}

int main(void) {
    HAL_Init();
    SystemClock_Config();
    GPIO_Init();
    TIM2_Init();

    uint32_t distance;

    while (1) {
        distance = Read_Ultrasonic_Distance();
        HAL_Delay(1000);
    }
}

4.2 数据处理与导航算法

数据处理模块将传感器数据转换为可用于导航系统的数据,并进行必要的计算和分析。

A*路径规划算法

实现一个简单的A*路径规划算法,用于导航机器人到达目标位置:

#include <stdio.h>
#include <stdlib.h>
#include <math.h>

#define GRID_SIZE 10
#define OBSTACLE -1
#define FREE 0
#define START 1
#define END 2

typedef struct {
    int x, y;
} Point;

typedef struct {
    Point point;
    int g, h, f;
    struct Node* parent;
} Node;

Node* create_node(Point point, int g, int h, Node* parent) {
    Node* node = (Node*)malloc(sizeof(Node));
    node->point = point;
    node->g = g;
    node->h = h;
    node->f = g + h;
    node->parent = parent;
    return node;
}

int heuristic(Point a, Point b) {
    return abs(a.x - b.x) + abs(a.y - b.y);
}

void a_star(int grid[GRID_SIZE][GRID_SIZE], Point start, Point end) {
    Node* open_list[GRID_SIZE * GRID_SIZE];
    int open_list_size = 0;
    Node* closed_list[GRID_SIZE * GRID_SIZE];
    int closed_list_size = 0;

    Node* start_node = create_node(start, 0, heuristic(start, end), NULL);
    open_list[open_list_size++] = start_node;

    while (open_list_size > 0) {
        Node* current = open_list[0];
        int current_index = 0;

        for (int i = 1; i < open_list_size; i++) {
            if (open_list[i]->f < current->f) {
                current = open_list[i];
                current_index = i;
            }
        }

        open_list[current_index] = open_list[--open_list_size];
        closed_list[closed_list_size++] = current;

        if (current->point.x == end.x && current->point.y == end.y) {
            Node* path_node = current;
            while (path_node != NULL) {
                grid[path_node->point.x][path_node->point.y] = PATH;
                path_node = path_node->parent;
            }
            return;
        }

        Point directions[4] = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}};
        for (int i = 0; i < 4; i++) {
            Point neighbor = {current->point.x + directions[i].x, current->point.y + directions[i].y};
            if (neighbor.x < 0 || neighbor.x >= GRID_SIZE || neighbor.y < 0 || neighbor.y >= GRID_SIZE || grid[neighbor.x][neighbor.y] == OBSTACLE) {
                continue;
            }

            int g = current->g + 1;
            int h = heuristic(neighbor, end);
            Node* neighbor_node = create_node(neighbor, g, h, current);

            int in_open_list = 0;
            for (int j = 0; j < open_list_size; j++) {
                if (open_list[j]->point.x == neighbor.x && open_list[j]->point.y == neighbor.y && open_list[j]->g <= g) {
                    in_open_list = 1;
                    break;
                }
            }

            if (!in_open_list) {
                open_list[open_list_size++] = neighbor_node;
            }
        }
    }
}

int main(void) {
    int grid[GRID_SIZE][GRID_SIZE] = {
        {0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
        {0, -1, -1, -1, 0, 0, 0, 0, 0, 0},
        {0, 0, 0, -1, 0, 0, 0, 0, 0, 0},
        {0, 0, 0, -1, 0, 0, 0, 0, 0, 0},
        {0, 0, 0, -1, 0, 0, 0, 0, 0, 0},
        {0, 0, 0, -1, 0, 0, 0, 0, 0, 0},
        {0, 0, 0, -1, 0, 0, 0, 0, 0, 0},
        {0, 0, 0, -1, 0, 0, 0, 0, 0, 0},
        {0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
        {0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
    };

    Point start = {0, 0};
    Point end = {9, 9};

    a_star(grid, start, end);

    for (int i = 0; i < GRID_SIZE; i++) {
        for (int j = 0; j < GRID_SIZE; j++) {
            printf("%d ", grid[i][j]);
        }
        printf("\n");
    }

    return 0;
}

4.3 通信与网络系统实现

配置以太网模块

使用STM32CubeMX配置以太网接口:

  1. 打开STM32CubeMX,选择您的STM32开发板型号。
  2. 在图形化界面中,找到需要配置的以太网引脚,设置为以太网模式。
  3. 生成代码并导入到STM32CubeIDE中。

代码实现:

#include "stm32f4xx_hal.h"
#include "lwip.h"
#include "ethernet.h"

void Ethernet_Init(void) {
    MX_LWIP_Init();
}

void Send_Data_To_Server(const char* data) {
    Ethernet_Transmit(data, strlen(data));
}

int main(void) {
    HAL_Init();
    SystemClock_Config();
    Ethernet_Init();

    char message[] = "Hello, Server!";

    while (1) {
        Send_Data_To_Server(message);
        HAL_Delay(1000);
    }
}
配置Wi-Fi模块

使用STM32CubeMX配置UART接口:

  1. 打打开STM32CubeMX,选择您的STM32开发板型号。
  2. 在图形化界面中,找到需要配置的UART引脚,设置为UART模式。
  3. 生成代码并导入到STM32CubeIDE中。

代码实现:

#include "stm32f4xx_hal.h"
#include "usart.h"
#include "wifi_module.h"

UART_HandleTypeDef huart1;

void UART1_Init(void) {
    huart1.Instance = USART1;
    huart1.Init.BaudRate = 115200;
    huart1.Init.WordLength = UART_WORDLENGTH_8B;
    huart1.Init.StopBits = UART_STOPBITS_1;
    huart1.Init.Parity = UART_PARITY_NONE;
    huart1.Init.Mode = UART_MODE_TX_RX;
    huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;
    huart1.Init.OverSampling = UART_OVERSAMPLING_16;
    HAL_UART_Init(&huart1);
}

void Send_Data_To_Server(const char* data) {
    HAL_UART_Transmit(&huart1, (uint8_t*)data, strlen(data), HAL_MAX_DELAY);
}

int main(void) {
    HAL_Init();
    SystemClock_Config();
    UART1_Init();

    char message[] = "Hello, Server!";

    while (1) {
        Send_Data_To_Server(message);
        HAL_Delay(1000);
    }
}

4.4 用户界面与数据可视化

配置OLED显示屏

使用STM32CubeMX配置I2C接口:

  1. 打打开STM32CubeMX,选择您的STM32开发板型号。
  2. 在图形化界面中,找到需要配置的I2C引脚,设置为I2C模式。
  3. 生成代码并导入到STM32CubeIDE中。

代码实现:

首先,初始化OLED显示屏:

#include "stm32f4xx_hal.h"
#include "i2c.h"
#include "oled.h"

void Display_Init(void) {
    OLED_Init();
}

然后实现数据展示函数,将物流机器人状态和路径信息展示在OLED屏幕上:

void Display_Data(const char* message) {
    OLED_ShowString(0, 0, message);
}

int main(void) {
    HAL_Init();
    SystemClock_Config();
    I2C1_Init();
    Display_Init();

    char message[] = "Hello, Robot!";

    while (1) {
        Display_Data(message);
        HAL_Delay(1000);
    }
}

5. 应用场景:物流机器人管理与优化

仓库管理

智能物流机器人系统可以用于仓库管理,通过实时监测机器人状态和环境数据,提高仓库管理的效率和准确性。

工厂自动化

在工厂自动化中,智能物流机器人系统可以实现对物料的自动化搬运和路径规划,提高生产效率和精度。

物流配送

智能物流机器人系统可以用于物流配送,通过自动化控制和路径规划,提高配送效率和精准度。

智能机器人研究

智能物流机器人系统可以用于智能机器人研究,通过数据采集和分析,为机器人导航和控制提供科学依据。

⬇帮大家整理了单片机的资料

包括stm32的项目合集【源码+开发文档】

点击下方蓝字即可领取,感谢支持!⬇

点击领取更多嵌入式详细资料

问题讨论,stm32的资料领取可以私信!

6. 问题解决方案与优化

常见问题及解决方案

传感器数据不准确

确保传感器与STM32的连接稳定,定期校准传感器以获取准确数据。

解决方案:检查传感器与STM32之间的连接是否牢固,必要时重新焊接或更换连接线。同时,定期对传感器进行校准,确保数据准确。

运动控制不稳定

优化控制算法和硬件配置,减少运动控制的不稳定性,提高系统反应速度。

解决方案:优化PID控制算法,调整PID参数,减少振荡和超调。使用高精度传感器,提高数据采集的精度和稳定性。选择更高效的电机和驱动器,提高运动控制的响应速度。

数据传输失败

确保以太网或Wi-Fi模块与STM32的连接稳定,优化通信协议,提高数据传输的可靠性。

解决方案:检查以太网或Wi-Fi模块与STM32之间的连接是否牢固,必要时重新焊接或更换连接线。优化通信协议,减少数据传输的延迟和丢包率。选择更稳定的通信模块,提升数据传输的可靠性。

显示屏显示异常

检查I2C通信线路,确保显示屏与MCU之间的通信正常,避免由于线路问题导致的显示异常。

解决方案:检查I2C引脚的连接是否正确,确保电源供电稳定。使用示波器检测I2C总线信号,确认通信是否正常。如有必要,更换显示屏或MCU。

优化建议

数据集成与分析

集成更多类型的传感器数据,使用数据分析技术进行环境状态的预测和优化。

建议:增加更多监测传感器,如激光雷达、摄像头等。使用云端平台进行数据分析和存储,提供更全面的环境监测和管理服务。

用户交互优化

改进用户界面设计,提供更直观的数据展示和更简洁的操作界面,增强用户体验。

建议:使用高分辨率彩色显示屏,提供更丰富的视觉体验。设计简洁易懂的用户界面,让用户更容易操作。提供图形化的数据展示,如实时环境参数图表、历史记录等。

智能化控制提升

增加智能决策支持系统,根据历史数据和实时数据自动调整控制策略,实现更高效的环境控制和管理。

建议:使用数据分析技术分析环境数据,提供个性化的环境管理建议。结合历史数据,预测可能的问题和需求,提前优化控制策略。

7. 收尾与总结

本教程详细介绍了如何在STM32嵌入式系统中实现智能物流机器人系统,从硬件选择、软件实现到系统配置和应用场景都进行了全面的阐述。

 

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

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

相关文章

ESP32CAM物联网教学11

ESP32CAM物联网教学11 霍霍webserver 在第八课的时候&#xff0c;小智把乐鑫公司提供的官方示例程序CameraWebServer改成了明码&#xff0c;这样说明这个官方程序也是可以更改的嘛。这个官方程序有四个文件&#xff0c;一共3500行代码&#xff0c;看着都头晕&#xff0c;小智决…

第100+15步 ChatGPT学习:R实现Ababoost分类

基于R 4.2.2版本演示 一、写在前面 有不少大佬问做机器学习分类能不能用R语言&#xff0c;不想学Python咯。 答曰&#xff1a;可&#xff01;用GPT或者Kimi转一下就得了呗。 加上最近也没啥内容写了&#xff0c;就帮各位搬运一下吧。 二、R代码实现Ababoost分类 &#xff…

打造热销爆款:LazadaShopee店铺测评与关键词策略

面对Lazada和Shopee平台上店铺销量难以突破的困境&#xff0c;卖家们往往寻求各种解决方案。其中&#xff0c;店铺测评作为提升店铺信誉、优化产品排名及增加曝光度的有效手段&#xff0c;正逐渐成为卖家关注的焦点。以下将深入探讨店铺测评的好处、实施技巧及自养号的关键要素…

RK3588部署YOLOV8-seg的问题

在使用YOLOV8-seg训练出来的pt模型转为onnx的时候&#xff0c;利用以下仓库地址转。 git clone https://github.com/airockchip/ultralytics_yolov8.git 在修改ultralytics/cfg/default.yaml中的task&#xff0c;mode为model为自己需要的内容后&#xff0c; 执行以下语句 cd …

2024最新修复微信公众号无限回调系统源码下载 免授权开心版

2024最新修复微信公众号无限回调系统源码下载 免授权开心版 微信公众平台回调比较麻烦&#xff0c;还不能多次回调&#xff0c;于是搭建一个多域名回调的源码很有必要。 测试环境&#xff1a;Nginx1.24PHP7.2MySQL5.6 图片&#xff1a;

5G与未来通信技术

随着科技的迅猛发展&#xff0c;通信技术也在不断演进。5G技术作为第五代移动通信技术&#xff0c;已成为现代通信技术的一个重要里程碑。本文将详细介绍5G及其对未来通信技术的影响&#xff0c;重点探讨超高速互联网和边缘网络的应用。 一、超高速互联网 1. 低延迟 5G技术最显…

一个vue页面复用方案

前言 问大家一个问题&#xff0c;曾经的你是否也遇到过&#xff0c;一个项目中有好几个页面长得基本相同&#xff0c;但又差那么一点&#xff0c;想用 vue extends 继承它又不能按需继承html模板部分&#xff0c;恰好 B 页面需要用的 A 页面 80% 的模板&#xff0c;剩下的 20%…

在Anaconda环境中安装TensorFlow+启动jupyter notebook

1.打开cmd&#xff0c;输入C:\Users\xy>conda create -n tensorflow python3.7 这是在环境中创建了一个名为tensorflow的环境&#xff0c;具体会显示以下信息&#xff1a; C:\Users\xy>conda create -n tensorflow python3.7 Retrieving notices: ...working... done Co…

等保从哪些方面进行测评

等保&#xff0c;全名叫做信息安全等级保护&#xff0c;顾名思义就是指根据信息系统在国家安全、社会稳定、经济秩序和公共利益方便的中重要程度以及风险威胁、安全需求、安全成本等因素&#xff0c;将其划分不同的安全保护等级并采取相应等级的安全保护技术、管理措施、以保障…

Python面试宝典第11题:最长连续序列

题目 给定一个未排序的整数数组 nums &#xff0c;找出数字连续的最长序列&#xff08;不要求序列元素在原数组中连续&#xff09;的长度。请你设计并实现时间复杂度为 O(n) 的算法解决此问题。 示例 1&#xff1a; 输入&#xff1a;nums [100,4,200,1,3,2] 输出&#xff1a;…

前端JS特效第30集:jQuery焦点图插件edslider

jQuery焦点图插件edslider&#xff0c;先来看看效果&#xff1a; 部分核心的代码如下(全部代码在文章末尾)&#xff1a; <!DOCTYPE html> <html lang"zh"> <head> <meta charset"UTF-8"> <meta http-equiv"X-UA-Compatib…

台湾精锐APEX伺服行星减速机发热原因及解决方案

在实际运行过程中台湾精锐APEX伺服行星减速机常常会遇到发热的问题&#xff0c;这不仅影响减速机的正常运转&#xff0c;还可能缩短其使用寿命&#xff0c;甚至引发安全事故。因此&#xff0c;了解APEX伺服行星减速机发热的原因及相应的解决方案&#xff0c;对于保障生产线的稳…

C语言——基础框架、变量、运算符

基础框架&#xff1a; #include<stdio.h> //编译预处理指令int main() //程序的入口主函数main { //程序&#xff08;函数、功能&#xff09;结束标志return 0; //程序退出前返回给调用者&#xff08;操作系统&#xff09;的值…

MySQL实战45讲学习笔记(持续更新ing……)

文章目录 一、基础架构&#xff1a;一条SQL查询语句是如何执行的&#xff1f;概览连接器查询缓存分析器优化器执行器 二、日志系统&#xff1a;一条SQL更新语句是如何执行的&#xff1f;redo logbinlog两阶段提交 一、基础架构&#xff1a;一条SQL查询语句是如何执行的&#xf…

深度学习DeepLearning多元线性回归 学习笔记

文章目录 多维特征变量与术语公式多元线性回归正规方程法Mean normalizationZ-score normalization设置合适的学习率Feature engineering 多维特征 变量与术语 列属性xj属性数n x ⃗ \vec{x} x (i)行向量某个值 x ⃗ j i \vec{x}_j^i x ji​上行下列均值μ标准化标准差σsigm…

无线速度传感器

对高中物理实验中的速度测量方法进行改进&#xff0c;利用安装在小车上的无线光电门来测量小车运动过程中的速度&#xff0c;即满足了精度的要求&#xff0c;又可以研究物体的运动过程。无线光电门和数据接收器间采用蓝牙无线传输的方式&#xff0c;电脑端的软件使用Flash来制作…

vant-app中加的custom-class为啥审查元素时看不到自定义类名

如下图&#xff1a; 我们发现在左侧审查元素时确实看不到&#xff0c;但是在右侧是可以看到&#xff0c;而且样式是生效的。 是不是微信开发者工具的bug?

SQL基础-DQL 小结

SQL基础-DQL 小结 学习目标&#xff1a;学习内容&#xff1a;SELECTFROMWHEREGROUP BYHAVINGORDER BY运算符ASC 和 DESC 总结 学习目标&#xff1a; 1.理解DQL&#xff08;Data Query Language&#xff09;的基本概念和作用。 2.掌握SQL查询的基本语法结构&#xff0c;包括SEL…

微软子公司Xandr遭隐私诉讼,或面临巨额罚款

近日&#xff0c;欧洲隐私权倡导组织noyb对微软子公司Xandr提起了诉讼&#xff0c;指控其透明度不足&#xff0c;侵犯了欧盟公民的数据访问权。据指控&#xff0c;Xandr的行为涉嫌违反《通用数据保护条例》&#xff08;GFPR&#xff09;&#xff0c;因其处理信息并创建用于微目…

C#开发:VS2022中配置TFS(Team Foundation Server)和使用

第一步&#xff0c;点出团队资源管理器 第二步&#xff0c;输入服务器地址 第三步&#xff0c;输入配置地址和账密&#xff08;问管理员&#xff09; 输入配置地址&#xff1a;$/xxxx 输入工作区地址&#xff1a;本地随便一个路径 第四步&#xff0c;获取最新代码 第五步&#…