(七)Linux库的串口开发

文章目录

  • 基于官方提供的串口测试
    • 代码部分
    • 解析代码部分
      • 1. `usage` 函数
      • 2. `opt_parsing_err_handle` 函数
      • 3. `sig_handle` 函数
      • 4. `init_serial` 函数
      • 5. `serial_write` 函数
      • 6. `serial_read` 函数
      • 7. `run_read_mode` 函数
      • 8. `run_write_mode` 函数
      • 9. `run_loopback_test` 函数
    • 进行测试
      • 第一步编译
      • 第二步发送到板子
      • 第三步调试
        • 测试前注意
        • 先测回环
        • 再测写
        • 最后测读
  • 基于原生Linux串口开发
    • 代码
    • 测试
  • 优化基于Linux原生串口开发
    • 代码
    • 测试


基于官方提供的串口测试

代码部分

创龙T113-i官方资料包给的代码:

uart_rw.c,功能很高端,支持读取、写入和回环测试三种模式!

/* Copyright 2018 Tronlong Elec. Tech. Co. Ltd. All Rights Reserved. */

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/select.h>
#include <sys/time.h>
#include <fcntl.h>
#include <termios.h>
#include <string.h>
#include <errno.h>
#include <stdbool.h>
#include <libgen.h>
#include <signal.h>
#include <getopt.h>

#define NOPASS_CONDITIONS 3
#define INADEQUATE_CONDITIONS 10

enum Mode { READ, WRITE, LOOPBACK };

/* Exit flag */
volatile bool g_quit = false;

/* Short option names */
static const char g_shortopts [] = ":d:s:rwvhl";

/* Option names */
static const struct option g_longopts [] = {
    { "device",      required_argument,      NULL,        'd' },
    { "read",        no_argument,            NULL,        'r' },
    { "write",       no_argument,            NULL,        'w' },
    { "loopback",    no_argument,            NULL,        'l' },
    { "size",        required_argument,      NULL,        's' },
    { "version",     no_argument,            NULL,        'v' },
    { "help",        no_argument,            NULL,        'h' },
    { 0, 0, 0, 0 }
};

static void usage(FILE *fp, int argc, char **argv) {
    fprintf(fp,
            "Usage: %s [options]\n\n"
            "Options:\n"
            " -d | --device        Device such as '/dev/ttyS0'\n"
            " -r | --read          Read\n"
            " -w | --write         Write\n"
            " -l | --loopback      loopback test\n"
            " -s | --size          Read size\n"
            " -v | --version       Display version information\n"
            " -h | --help          Show help content\n"
            " e.g. %s -d /dev/ttyS1 -r -s 256\n"
            "      %s -d /dev/ttyS1 -w -s 1024\n"
            "      %s -d /dev/ttyS1 -l -s 1024\n\n"
            "", argv[0], argv[0], argv[0], argv[0]);
}

static void opt_parsing_err_handle(int argc, char **argv, int flag) {
    /* Exit if no input parameters are entered  */
    int state = 0;
    if (argc < 2) {
        printf("No input parameters are entered, please check the input.\n");
        state = -1;
    } else {
        /* Feedback Error parameter information then exit */
        if (optind < argc || flag) {
            printf("Error:  Parameter parsing failed\n");
            if (flag)
                printf("\tunrecognized option '%s'\n", argv[optind-1]);

            while (optind < argc) {
                printf("\tunrecognized option '%s'\n", argv[optind++]);
            }

            state = -1;
        }
    }

    if (state == -1) {
        printf("Tips: '-h' or '--help' to get help\n\n");
        exit(2);
    }
}

void sig_handle(int arg) {
    g_quit = true;
}

int init_serial(int *fd, const char *dev) {
    struct termios opt;

    /* open serial device */
    if ((*fd = open(dev, O_RDWR)) < 0) {
        perror("open()");
        return -1;
    }

    /* define termois */
    if (tcgetattr(*fd, &opt) < 0) {
        perror("tcgetattr()");
        return -1;
    }

    opt.c_lflag  &= ~(ICANON | ECHO | ECHOE | ISIG);
    opt.c_oflag  &= ~OPOST;

    /* Character length, make sure to screen out this bit before setting the data bit */
    opt.c_cflag &= ~CSIZE;

    /* No hardware flow control */
    opt.c_cflag &= ~CRTSCTS;

    /* 8-bit data length */
    opt.c_cflag |= CS8;

    /* 1-bit stop bit */
    opt.c_cflag &= ~CSTOPB;

    /* No parity bit */
    opt.c_iflag |= IGNPAR;

    /* Output mode */
    opt.c_oflag = 0;
    
    /* No active terminal mode */
    opt.c_lflag = 0;

    /* Input baud rate */
    if (cfsetispeed(&opt, B115200) < 0)
        return -1;

    /* Output baud rate */
    if (cfsetospeed(&opt, B115200) < 0)
        return -1;

    /* Overflow data can be received, but not read */
    if (tcflush(*fd, TCIFLUSH) < 0)
        return -1;

    if (tcsetattr(*fd, TCSANOW, &opt) < 0)
        return -1;

    return 0;
}

int serial_write(int *fd, const char *data, size_t size) {
    int ret = write(*fd, data, size);
    if ( ret < 0 ) {
        perror("write");
        tcflush(*fd, TCOFLUSH);
    }

    return ret;
}

int serial_read(int *fd, char *data, size_t size) {
    size_t read_left = size;
    size_t read_size = 0;
    char *read_ptr = data;
    struct timeval timeout = {5, 0};

    memset(data, 0, size);

    fd_set rfds;
    while (!g_quit) {
        FD_ZERO(&rfds);
        FD_SET(*fd, &rfds);
        timeout.tv_sec = 5;
        timeout.tv_usec = 0;

        if (read_left == 0)
            break;

        switch (select(*fd+1, &rfds, NULL, NULL, &timeout)) {
        case -1:
            perror("select()");
            break;
        case 0:
            perror("timeout and retry");
            break;
        default:
            if (FD_ISSET(*fd,&rfds)) {
                read_size = read(*fd, read_ptr, read_left);
                if (read_size == 0)
                    break;

                read_ptr += read_size;
                read_left -= read_size;
            }
        }
    }

    return read_size;
}

int run_read_mode(char *dev, size_t size) {
    char *buf = NULL;
    int fd = -1;
    int ret = -1;

    ret = init_serial(&fd, dev);
    if (ret < 0) {
        close(fd);
        return -1;
    }

    printf("Mode : read\n");
    if (size <= 0) {
        printf ("Error : Incorrect size settings\n");
        exit(INADEQUATE_CONDITIONS);
    }
    
    buf = (char*)malloc(size + 1);
    buf[size] = '\0';
    ret = serial_read(&fd, buf, size);
    printf("recv: %s\nsize: %d\n", buf, ret);

    free(buf);
    return 0;
}

int run_write_mode(char *dev, size_t size) {
    int fd = -1;
    int ret = -1;

    ret = init_serial(&fd, dev);
    if (ret < 0) {
        close(fd);
        return -1;
    }

    printf("Mode : write\n");
    if (size <= 0) {
        printf("Error : Incorrect size settings\n");
        exit(INADEQUATE_CONDITIONS);
    }

    int i = 0;
    char context;
    size_t write_size = 0;
    while (!g_quit) {
        if (i > 7)
            i = 0;
        context = (char)('0' + i);
            
        write_size += serial_write(&fd, &context, sizeof(context));
        i ++;

        if (size == write_size)
            break;
    }

    printf("send size: %zd\n", write_size);
    return 0;
}

int run_loopback_test(char *dev, size_t size) {
    int fd;
    int ret;
    size_t buf_size;
    int serial_buf_size;

    ret = init_serial(&fd, dev);
    if (ret < 0)
        return -1;

    printf("Start uart loopback testing.\n");

    /* Serial port buffer size generally defaults to 2k - 4k */
    char *write_buf = (char*)malloc(size);
    char *read_buf = (char*)malloc(size);

    buf_size = size;
    while (buf_size > 0)
    {
        if(buf_size > 1024) {
            serial_buf_size = 1024;
        } else {
            serial_buf_size = buf_size;
        }

        // Generate random data to write.
        memset(write_buf, rand() % 26 + 65, serial_buf_size);
        memset(read_buf, 0, serial_buf_size);

        ret = serial_write(&fd, write_buf, serial_buf_size);
        

        /* delay > 1024 / 115200 * 1000000 */
        usleep(90000);

        ret = serial_read(&fd, read_buf, serial_buf_size);
        

        ret = memcmp(read_buf, write_buf, serial_buf_size);
        if (ret != 0) {
            printf("Result : Test failed\n");
            goto release;
        }

        buf_size -= 1024;
    }
    printf("send size: %zd\n", size);
    printf("recv size: %zd\n", size);
    printf("Result : Test pass\n");

release:
    free(write_buf);
    free(read_buf);
    close (fd);
    if (ret != 0) {
        return NOPASS_CONDITIONS;
    } else {
        return 0;
    }
}


int main(int argc, char *argv[]) {
    int c = 0;
    int flag = 0;
    int mode = -1;
    size_t size = 0;
    char *dev = NULL;
    int ret = -1;

    /* Parsing input parameters */
    while ((c = getopt_long(argc, argv, g_shortopts, g_longopts, NULL)) != -1) {
        switch (c) {
        case 'd':
            dev = optarg;
            break;

        case 'r':
            mode = READ;
            break;

        case 'w':
            mode = WRITE;
            break;
            
        case 'l':
            mode = LOOPBACK;
            break;

        case 's':
            size = atoi(optarg);
            break;

        case 'v':
            /* Display the version */
            printf("version : 1.0\n");
            exit(0);

        case 'h':
            usage(stdout, argc, argv);
            exit(0);
                
        default :
            flag = 1;
            break;
        }
    }

    opt_parsing_err_handle(argc, argv, flag);

    /* Ctrl+c handler */
    signal(SIGINT, sig_handle);

    switch (mode) {
    case READ:
        if(run_read_mode(dev, size) < 0) {
            return INADEQUATE_CONDITIONS;
        }
        break;

    case WRITE:
        if(run_write_mode(dev, size) < 0) {
            return INADEQUATE_CONDITIONS;
        }
        break;

    case LOOPBACK:
        ret = run_loopback_test(dev, size);
        if(ret < 0) {
            return INADEQUATE_CONDITIONS;
        } else if(ret == NOPASS_CONDITIONS) {
            return NOPASS_CONDITIONS;
        }
        break;
    default:
        break;
    }

    return 0;
}

解析代码部分

不深究可以不看,了解函数啥功能即可。

1. usage 函数

  • 作用:打印帮助信息给用户。
  • 参数
    • FILE *fp: 输出流,可以是标准输出(stdout)或标准错误(stderr)。
    • argc, argv[]: 命令行参数的数量和值,用于在示例中显示程序名。
  • 行为:当用户请求帮助(-h--help)时调用,向用户提供如何使用该工具的信息。

2. opt_parsing_err_handle 函数

  • 作用:处理命令行选项解析过程中的错误。
  • 参数
    • argc, argv[]: 命令行参数的数量和值。
    • flag: 标记是否有未知选项被识别。
  • 行为:如果命令行参数为空或者存在无法识别的选项,则打印错误信息并提示用户使用 -h 获取帮助,然后退出程序。

3. sig_handle 函数

  • 作用:信号处理器,用来响应中断信号(如Ctrl+C)。
  • 参数int arg,传递给信号处理器的信号编号。
  • 行为:设置全局变量 g_quit 为真,通知其他部分停止工作。

4. init_serial 函数

  • 作用:初始化串行端口配置。
  • 参数
    • fd: 文件描述符指针,将被设置为打开的串行端口。
    • dev: 设备路径字符串,例如 /dev/ttyS0
  • 行为:根据提供的设备路径打开串行端口,并配置波特率、数据位、停止位等参数。它还会清除终端模式并禁用硬件流控制。

5. serial_write 函数

  • 作用:向串行端口写入数据。
  • 参数
    • fd: 文件描述符指针。
    • data: 指向要发送的数据的指针。
    • size: 要发送的数据大小。
  • 行为:尝试将指定数量的字节写入到串行端口,失败时刷新输出缓冲区并报告错误。

6. serial_read 函数

  • 作用:从串行端口读取数据。
  • 参数
    • fd: 文件描述符指针。
    • data: 指向存储接收到的数据的缓冲区。
    • size: 预期接收的数据量。
  • 行为:使用 select 函数等待数据到达,然后尽可能多地读取数据直到达到预期大小或超时。如果发生错误或超时,会给出相应的错误信息。

7. run_read_mode 函数

  • 作用:执行读取模式操作。
  • 参数
    • dev: 设备路径。
    • size: 期望读取的数据大小。
  • 行为:以非阻塞方式读取指定大小的数据,并将其打印出来。如果指定大小无效,则报错退出。

8. run_write_mode 函数

  • 作用:执行写入模式操作。
  • 参数
    • dev: 设备路径。
    • size: 要写入的数据大小。
  • 行为:循环地写入字符到串行端口,直到写入了指定大小的数据。每次写入后,字符递增,模拟连续的数据流。

9. run_loopback_test 函数

  • 作用:执行回环测试。
  • 参数
    • dev: 设备路径。
    • size: 测试过程中使用的数据大小。
  • 行为:生成随机数据写入串行端口,然后立即尝试读回相同的数据,比较两者是否一致。如果不一致,则认为测试失败;否则,测试通过。

进行测试

第一步编译

用交叉编译工具进行编译。

不会配置交叉编译链的,请参考:(二)编译原生SDK以及配置交叉编译链中配置交叉编译链的部分。

arm-linux-gnueabi-gcc uart_rw.c -o uart_rw

请添加图片描述

查看一下可执行文件类型,避免错误。

请添加图片描述

第二步发送到板子

借助基于ssh的scp工具,不会的可以参考(四)配置有线网口、SSH登陆、文件传输以及运行交叉编译程序测试中SHH登陆以及文件传输部分。

scp ./uart_rw root@192.168.1.101:/root/zhua

出现问题:

请添加图片描述

根据提示,解决问题,并再次发送:

请添加图片描述

请添加图片描述

第三步调试

测试前注意

如何运行可执行文件,查看如何运行

./uart_rw -h

请添加图片描述

当然直接读代码也是可以的,参考这部分:

请添加图片描述

注意一下波特率是115200,可以从代码中了解到:

请添加图片描述

最后查看用的串口4是否开启(可以自行指定),注意创龙的是ttyAS4,跟传统的名称还是有区别的

ls /dev

请添加图片描述

可以在设备文件夹下看到,说明串口的设备是成功加载到了系统中。如果看不到,请进一步修改设备树,书写设备驱动文件(不会的,请参考后面的章节),再次编译镜像,进行烧录。

先测回环
 ./uart_rw -d /dev/ttyAS4 -l -s 1024

记得要把串口4的TX和RX短接。另外我开启的是串口4。

请添加图片描述

再测写

注意这里写的内容就是0-7 来回循环,看代码便知。

请添加图片描述

./uart_rw --device /dev/ttyAS4 --write --size 16

或者简写参数

./uart_rw -d /dev/ttyAS4 -w -s 16

请添加图片描述

请添加图片描述

最后测读

这里我读8个长度,用串口助手发送87654321。

./uart_rw --device /dev/ttyAS4 --read --size 8

或者简写

./uart_rw -d /dev/ttyAS4 -r -s 8

请添加图片描述
中间提示的,timeout and retry: Success应该是一帧数据的帧检测,可以参考蓝桥杯专栏的(十六)串口UART进行学习。

基于原生Linux串口开发

功能简单,借助线程操作,支持读写同步。

步骤:初始化串口,开启父子线程,父线程读操作,子线程写操作。

代码

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/select.h>
#include <sys/time.h>
#include <fcntl.h>
#include <termios.h>
#include <string.h>
#include <errno.h>
#include <stdbool.h>
#include <libgen.h>
#include <signal.h>

int init_serial(int *fd, const char *dev) {
    struct termios opt;

    /* open serial device */
    if ((*fd = open(dev, O_RDWR)) < 0) {
        perror("open()");
        return -1;
    }

    /* define termois */
    if (tcgetattr(*fd, &opt) < 0) {
        perror("tcgetattr()");
        return -1;
    }

    opt.c_lflag  &= ~(ICANON | ECHO | ECHOE | ISIG);
    opt.c_oflag  &= ~OPOST;

    /* Character length, make sure to screen out this bit before setting the data bit */
    opt.c_cflag &= ~CSIZE;

    /* No hardware flow control */
    opt.c_cflag &= ~CRTSCTS;

    /* 8-bit data length */
    opt.c_cflag |= CS8;

    /* 1-bit stop bit */
    opt.c_cflag &= ~CSTOPB;

    /* No parity bit */
    opt.c_iflag |= IGNPAR;

    /* Output mode */
    opt.c_oflag = 0;
    
    /* No active terminal mode */
    opt.c_lflag = 0;

    /* Input baud rate */
    if (cfsetispeed(&opt, B115200) < 0)
        return -1;

    /* Output baud rate */
    if (cfsetospeed(&opt, B115200) < 0)
        return -1;

    /* Overflow data can be received, but not read */
    if (tcflush(*fd, TCIFLUSH) < 0)
        return -1;

    if (tcsetattr(*fd, TCSANOW, &opt) < 0)
        return -1;

    return 0;
}


int serial_write(int *fd, const char *data, size_t size) {
    int ret = write(*fd, data, size);
    if ( ret < 0 ) {
        perror("write");
        tcflush(*fd, TCOFLUSH);
    }

    return ret;
}

int serial_read(int *fd, size_t size) 
{
    int read_size = 0;
    char data[128] = {'\0'};

    while(1){
        read_size = read(*fd, data, size);
        if(read_size!=0){
            printf("read_size = %d,context = %s\n",read_size,data);
            memset(data,'\0',128);
            read_size = 0;
        }

    }

    return 0;
}

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

    __pid_t pid = 0;
 
    char *writedata = "write from root\n";

    int ret= -1;
    int fd = -1;

    if(argc<2){
        printf("tips:./myuart /dev/xxx\n");
        printf("argc = %d,argv[0] = %s,,argv[1] = %s\n",argc,argv[0],argv[1]);
    }
    
    ret = init_serial(&fd, argv[1]);
    if(ret == -1){
        return -1;
    }
    //开启2个线程
    pid = fork();
    if(pid>0){  //父进程 就是父进程的pid号	
       serial_read(&fd,128);

    }else if(pid==0){   //子进程
        while(1){
            serial_write(&fd,writedata,strlen(writedata));
            sleep(3);
        }
    }else{      //fork线程错误错误
        perror("fork faild\n");
        return -1;
    }

    return 0;
}

测试

执行指令:

./myuart /dev/ttyAS4

注意使用的是串口4,波特率115200。

请添加图片描述
请添加图片描述

优化基于Linux原生串口开发

向官方牛x的功能靠近一点点!

实现指定长短的发送,可以双方互不干扰。

这里是开启两个线程,一个是发送线程,一个是接收线程。在原有的Linux原生串口代码的基础上改进,之前使用的fork函数,创建的父子线程有局限性,资源争夺问题,数据共享等问题,所以这里借助pthread_create直接创建两个新的线程。

可以参考(十四)基于Linux的串口开发中基于Linux库的开发,几乎一样。

代码

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/select.h>
#include <sys/time.h>
#include <fcntl.h>
#include <termios.h>
#include <string.h>
#include <errno.h>
#include <stdbool.h>
#include <libgen.h>
#include <signal.h>
#include <pthread.h>

int fd = -1;


int init_serial(int *fd, const char *dev) {
    struct termios opt;

    /* open serial device */
    if ((*fd = open(dev, O_RDWR)) < 0) {
        perror("open()");
        return -1;
    }

    /* define termois */
    if (tcgetattr(*fd, &opt) < 0) {
        perror("tcgetattr()");
        return -1;
    }

    opt.c_lflag  &= ~(ICANON | ECHO | ECHOE | ISIG);
    opt.c_oflag  &= ~OPOST;

    /* Character length, make sure to screen out this bit before setting the data bit */
    opt.c_cflag &= ~CSIZE;

    /* No hardware flow control */
    opt.c_cflag &= ~CRTSCTS;

    /* 8-bit data length */
    opt.c_cflag |= CS8;

    /* 1-bit stop bit */
    opt.c_cflag &= ~CSTOPB;

    /* No parity bit */
    opt.c_iflag |= IGNPAR;

    /* Output mode */
    opt.c_oflag = 0;
    
    /* No active terminal mode */
    opt.c_lflag = 0;

    /* Input baud rate */
    if (cfsetispeed(&opt, B115200) < 0)
        return -1;

    /* Output baud rate */
    if (cfsetospeed(&opt, B115200) < 0)
        return -1;

    /* Overflow data can be received, but not read */
    if (tcflush(*fd, TCIFLUSH) < 0)
        return -1;

    if (tcsetattr(*fd, TCSANOW, &opt) < 0)
        return -1;

    return 0;
}


int serial_write(int *fd, const char *data, size_t size) {
    int ret = write(*fd, data, size);
    if ( ret < 0 ) {
        perror("write");
        tcflush(*fd, TCOFLUSH);
    }

    return ret;
}

int serial_read(int *fd, size_t size) 
{
    int read_size = 0;
    char data[128] = {'\0'};

    while(1){
        read_size = read(*fd, data, size);
        if(read_size!=0){
            printf("read_size = %d,context = %s\n",read_size,data);
            memset(data,'\0',128);
            read_size = 0;
        }

    }

    return 0;
}

void* recive()
{
    printf("recive pthread========ok========\n");
    while(1){
       
        serial_read(&fd,128);
    }

}

char *writedata = NULL;
int nwrite = 0;

void* send()
{
    printf("send pthread========ok========\n");
    while(1){

        printf("please enter n size what you want to write and press Enter to end \n");
        scanf("%d",&nwrite);
        writedata = (char *)malloc(nwrite*sizeof(char));
        scanf("%s",writedata);
        serial_write(&fd,writedata,strlen(writedata));
        free(writedata);
        writedata = NULL;
    }
}

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

    pthread_t recivet;
    pthread_t sendt;
    int ret= -1;

    if(argc<2){
        printf("tips:./myuart /dev/xxx\n");
        printf("argc = %d,argv[0] = %s,,argv[1] = %s\n",argc,argv[0],argv[1]);
    }
    
    ret = init_serial(&fd, argv[1]);
    if(ret == -1){
        return -1;
    }

    pthread_create(&recivet,NULL,recive,NULL);//接收线程
    pthread_create(&sendt,NULL,send,NULL);//发送进程
    while(1){
        //啥事也不干 3秒睡一次
        sleep(3);
    }

    return 0;
}

测试

注意这里因为用到的pthread_create函数是POSIX线程(pthreads)库的一部分,因此在编译和链接程序时需要链接到这个库,可以通过在编译命令中添加 -lpthread 选项来实现。否则会失败。

请添加图片描述

必须链接库编译

arm-linux-gnueabi-gcc myuart.c -o myuart -lpthread

输入数据格式一定要注意,因为使用的scanf这个函数,严格遵守它的规范,用回车或者空格断开数据。

请添加图片描述

请添加图片描述

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

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

相关文章

【Uniapp-Vue3】创建自定义页面模板

大多数情况下我们都使用的是默认模板&#xff0c;但是默认模板是Vue2格式的&#xff0c;如果我们想要定义一个Vue3模板的页面就需要自定义。 一、我们先复制下面的模板代码&#xff08;可根据自身需要进行修改&#xff09;&#xff1a; <template><view class"…

【机器视觉】OpenCV 图像轮廓(查找/绘制轮廓、轮廓面积/周长、多边形逼近与凸包、外接矩形)

文章目录 7. 图像轮廓7.1 什么是图像轮廓7.2 查找轮廓7.3 绘制轮廓7.4 轮廓的面积和周长7.5 多边形逼近与凸包7.6 外接矩形 OpenCV官网 7. 图像轮廓 7.1 什么是图像轮廓 图像轮廓是具有相同颜色或灰度的连续点的曲线. 轮廓在形状分析和物体的检测和识别中很有用。 轮廓的作用…

20250109使用M6000显卡在Ubuntu20.04.6下跑whisper来识别中英文字幕

20250109使用M6000显卡在Ubuntu20.04.6下跑whisper来识别中英文字幕 2025/1/9 20:57 https://blog.csdn.net/wb4916/article/details/144541848 20241217使用M6000显卡在WIN10下跑whisper来识别中英文字幕 步骤&#xff1a; 1、在NVIDIA的官网下载并安装M6000显卡在WIN10下的最…

EtherCAT转CANopen数字油田的高效解决方案

在数字化时代&#xff0c;油田的管理和运作正经历着深刻的变革。传统的油田监测系统受限于通信技术&#xff0c;往往难以满足实时、高效的数据传输需求。面对这一挑战&#xff0c;开疆智能EtherCAT转CANopen网关应运而生&#xff0c;它以其卓越的性能和高度的兼容性&#xff0c…

C语言——文件IO 【文件IO和标准IO区别,操作文件IO】open,write,read,dup2,access,stat

1.思维导图 2.练习 1&#xff1a;使用C语言编写一个简易的界面&#xff0c;界面如下 1&#xff1a;标准输出流 2&#xff1a;标准错误流 3&#xff1a;文件流 要求&#xff1a;按1的时候&#xff0c;通过printf输出数据&#xff0c;按2的时候&#xff0c;通过p…

Android - NDK :JNI实现异步回调

在android代码中&#xff0c;通过JNI调用c层子线程执行耗时任务&#xff0c;在c层子线程中把结果回调到android层&#xff0c; C语言小白&#xff0c;请批评指正&#xff01; android层代码&#xff1a; import androidx.appcompat.app.AppCompatActivity;import android.os.…

Java Web开发进阶——RESTful API设计与开发

随着分布式系统和微服务架构的流行&#xff0c;RESTful API已成为现代Web应用中后端与前端、第三方系统交互的重要方式。本节将深入探讨RESTful API的设计原则、实现方式以及如何使用Spring Boot开发高效、可靠的RESTful服务。 1. 理解RESTful API的设计原则 1.1 什么是RESTfu…

PWR-STM32电源控制

一、原理 睡眠模式不响应其他操作&#xff0c;比如烧写程序&#xff0c;烧写时按住复位键松手即可下载&#xff0c;在禁用JTAG也可如此烧写程序。 对于低功耗模式可以通过RTC唤醒、外部中断唤醒、中断唤醒。 1、电源框图&#xff1a; VDDA主要负责模拟部分的供电、Vref和Vref-…

深兰科技董事长陈海波应邀为华东师大心理学专业师生做AI专题讲座

12月28日&#xff0c;应上海华东师范大学的邀请&#xff0c;上海市科协常委、上海交通大学博士生导师、深兰科技创始人兼董事长陈海波专程到校&#xff0c;为该校心理学专业的全体师生做了一场关于人工智能推动个人数字化未来的专题讲座。 他在演讲中&#xff0c;首先详细讲述了…

ssh2-sftp-client uploadDir Upload error: getLocalStatus: Bad path: ./public

报错解释 这个错误表明在使用 ssh2-sftp-client 这个Node.js库进行目录上传时遇到了问题。具体来说&#xff0c;是指定的本地路径&#xff08;./public&#xff09;不正确或者不存在。 解决方法&#xff1a; 确认当前工作目录&#xff1a;确保你在执行上传操作时的当前工作目…

Vue指令的综合案例

Vue指令的综合案例 参考文献&#xff1a; Vue的快速上手 Vue指令上 Vue指令下 文章目录 Vue指令的综合案例记事本 列表渲染删除功能添加功能底部统计和清空总代码 结语 博客主页: He guolin-CSDN博客 关注我一起学习&#xff0c;一起进步&#xff0c;一起探索编程的无限可能…

2025新春烟花代码(二)HTML5实现孔明灯和烟花效果

效果展示 源代码 <!DOCTYPE html> <html lang"en"> <script>var _hmt _hmt || [];(function () {var hm document.createElement("script");hm.src "https://hm.baidu.com/hm.js?45f95f1bfde85c7777c3d1157e8c2d34";var …

# 网络编程 - 轻松入门不含糊

网络编程 - 轻松入门 介绍 网络编程指的是&#xff0c;在网络通信协议下。实现 不同计算机之间 的数据传输&#xff0c;例如 通信、聊天、视频通话 等。 1. 网络编程概述 学习网络编程过程 中我们可以将网络编程理解为 计算机之间的数据交互 但 前提是通…

SAP BC 同服务器不同client之间的传输SCC1

源配置client不需要释放 登录目标client SCC1

【大数据基础】大数据概述

【作者主页】Francek Chen 【专栏介绍】 ⌈ ⌈ ⌈大数据技术原理与应用 ⌋ ⌋ ⌋专栏系统介绍大数据的相关知识&#xff0c;分为大数据基础篇、大数据存储与管理篇、大数据处理与分析篇、大数据应用篇。内容包含大数据概述、大数据处理架构Hadoop、分布式文件系统HDFS、分布式数…

【ROS2】☆ launch之Python

☆重点 ROS1和ROS2其中一个很大区别之一就是launch的编写方式。在ROS1中采用xml格式编写launch&#xff0c;而ROS2保留了XML 格式launch&#xff0c;还另外引入了Python和YAML 编写方式。选择哪种编写取决于每位开发人员的爱好&#xff0c;但是ROS2官方推荐使用Python方式编写…

Shell编程详解

文章目录 一、Linux系统结构二、Shell介绍1、Shell简介2、Shell种类3、Shell查询和切换 三、Shell基础语法1、注释2、本地变量3、环境变量3.1、查看环境变量3.2、临时设置环境变量3.3、永久设置环境变量 4、特殊变量5、控制语句5.1、shell中的中括号5.2、if语句5.3、for循环5.4…

Zemax 序列模式下的扩束器

扩束器结构原理 扩束器用于增加准直光束&#xff08;例如激光束&#xff09;的直径&#xff0c;同时保持其准直。它通常用于激光光学和其他需要修改光束大小或发散度的应用。 在典型的扩束器中&#xff0c;输入光束是准直激光器&#xff0c;或光束进入第一个光学元件。当光束开…

react-quill 富文本组件编写和应用

index.tsx文件 import React, { useRef, useState } from react; import { Modal, Button } from antd; import RichEditor from ./RichEditor;const AnchorTouchHistory: React.FC () > {const editorRef useRef<any>(null);const [isModalVisible, setIsModalVis…

【深度学习】多目标融合算法(二):底部共享多任务模型(Shared-Bottom Multi-task Model)

目录 一、引言 1.1 往期回顾 1.2 本期概要 二、Shared-Bottom Multi-task Model&#xff08;SBMM&#xff09; 2.1 技术原理 2.2 技术优缺点 2.3 业务代码实践 三、总结 一、引言 在朴素的深度学习ctr预估模型中&#xff08;如DNN&#xff09;&#xff0c;通常以一个行…