• 网络编程:协议拆分练习(通过协议控制机械臂 TFTP传输文件)


    一.使用代码控制机械臂

    1.1协议要求以及操作

    机械臂程序 使用 QT 写的一个windows下运行的 TCP 服务器

    运行在哪个主机上 ip地址就是哪个主机的IP地址 端口号就是自己指定的 控制端口号

    点击开启监听 就ok了 就开始等待客户端连接了

    机械臂的协议:0xff 0x02 (1) (2) 0xff

    (1) 要操作的摆臂 0x00 红色的摆臂 0x01 控制蓝色的摆臂

    (2) 要偏移的角度

    红 [-90, 90] char

    蓝 [0, 180] unsigned char

    1.2代码实现:

    1. #include
    2. #include
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. #include
    9. #include
    10. #define ERRLOG(msg) do{\
    11. printf("%s %s(%d):", __FILE__, __func__, __LINE__);\
    12. perror(msg);\
    13. exit(-1);\
    14. }while(0)
    15. int main(int argc, char const *argv[])
    16. {
    17. if(3 != argc){
    18. printf("Usage: %s \n", argv[0]);
    19. return -1;
    20. }
    21. int socketfd;
    22. if (-1 == (socketfd = socket(AF_INET, SOCK_STREAM, 0))){
    23. ERRLOG("socket error");
    24. }
    25. struct sockaddr_in clientaddr;
    26. clientaddr.sin_family = AF_INET;
    27. clientaddr.sin_port = htons(atoi(argv[2]));
    28. clientaddr.sin_addr.s_addr = inet_addr(argv[1]);
    29. socklen_t clientaddr_len = sizeof(clientaddr);
    30. if (-1 == connect(socketfd, (struct sockaddr*)&clientaddr, sizeof(clientaddr))){
    31. ERRLOG("connect error");
    32. }
    33. printf("连接成功\n");
    34. char red_buff[5] = {0xff, 0x02, 0x00, 0x00, 0xff};
    35. unsigned char blue_buff[5] = {0xff, 0x02, 0x01, 0x00, 0xff};
    36. if (-1 == send(socketfd, red_buff, 5, 0)){
    37. ERRLOG("send red_buf error");
    38. }
    39. sleep(1);
    40. if (-1 == send(socketfd, blue_buff, 5, 0)){
    41. ERRLOG("send blue_buff error");
    42. }
    43. char choose;
    44. while (1){
    45. printf("请输入您的选择(red+/- w/s blue+/- a/d)");
    46. scanf("%c", &choose);
    47. getchar();
    48. switch (choose){
    49. case 'w':
    50. if (red_buff[3] < 90){
    51. red_buff[3] += 10;
    52. if (-1 == send(socketfd, red_buff, 5, 0)){
    53. ERRLOG("send red_buf error");
    54. }
    55. }
    56. break;
    57. case 's':
    58. if (red_buff[3] > -90){
    59. red_buff[3] -= 10;
    60. if (-1 == send(socketfd, red_buff, 5, 0)){
    61. ERRLOG("send red_buf error");
    62. }
    63. }
    64. break;
    65. case 'a':
    66. if (blue_buff[3] < 180){
    67. blue_buff[3] += 10;
    68. if (-1 == send(socketfd, blue_buff, 5, 0)){
    69. ERRLOG("send red_buf error");
    70. }
    71. }
    72. break;
    73. case 'd':
    74. if (blue_buff[3] > 0){
    75. blue_buff[3] -= 10;
    76. if (-1 == send(socketfd, blue_buff, 5, 0)){
    77. ERRLOG("send red_buf error");
    78. }
    79. }
    80. break;
    81. case 'q':
    82. goto ret;
    83. }
    84. }
    85. ret:
    86. return 0;
    87. }

    TFTP协议练习

    2.1 TFTP简介、通信过程

    2.1.1 TFTP概述

    TFTP:简单文件传送协议

    最初用于引导无盘系统,被设计用来传输小文件

    2.1.2 特点:

    基于UDP协议实现

    不进行用户有效性认证

    2.1.3 数据传输模式:

    octet:二进制模式

    netascii:文本模式

    mail:已经不再支持

    2.2 TFTP通信过程

    2.3 TFTP通信过程总结(无选项)

    1、服务器在69号端口等待客户端的请求

    2、服务器若批准此请求,则使用临时端口与客户端进行通信

    3、每个数据包的编号都有变化(从1开始)

    4、每个数据包都要得到ACK的确认如果出现超时,则需要重新发送最后的包(数据或ACK)

    5、数据的长度以512Byte传输

    6、小于512Byte的数据意味着传输结束

    2.4 TFTP协议分析

     

    错误码:

    0 未定义,参见错误信息

    1 File not found.

    2 Access violation.

    3 Disk full or allocation exceeded.

    4 illegal TFTP operation.

    5 Unknown transfer ID.

    6 File already exists.

    7 No such user.

    8 Unsupported option(s) requested.

    2.5代码实现:

    1. #include
    2. #include
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. #include
    9. #include
    10. #include
    11. #include
    12. #define ERRLOG(msg) do{\
    13. printf("%s %s(%d):", __FILE__, __func__, __LINE__);\
    14. perror(msg);\
    15. exit(-1);\
    16. }while(0)
    17. int main(int argc, char const *argv[])
    18. {
    19. if (argc != 3){
    20. printf("输入的格式错误\n");
    21. exit(-1);
    22. }
    23. int socketfd;
    24. int nbytes;
    25. int fd;
    26. unsigned char buff[600] = {0};
    27. char filename[32] = {0};
    28. char text[512] = {0};
    29. unsigned short code;
    30. unsigned short number;
    31. unsigned short block_number = 0;
    32. if (-1 == (socketfd = socket(AF_INET, SOCK_DGRAM, 0))){
    33. ERRLOG("socket error");
    34. }
    35. struct sockaddr_in serveraddr;
    36. memset(&serveraddr, 0, sizeof(serveraddr));
    37. serveraddr.sin_family = AF_INET;
    38. rename:
    39. serveraddr.sin_port = htons(atoi(argv[2]));
    40. serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
    41. socklen_t serveraddr_len = sizeof(serveraddr);
    42. printf("请输入您要传输的文件");
    43. fgets(filename, sizeof(filename), stdin);
    44. filename[strlen(filename) - 1] = '\0';
    45. nbytes = sprintf(buff, "%c%c%s%c%s%c", 0, 1, filename, 0, "octet", 0);
    46. if (-1 == sendto(socketfd, buff, nbytes, 0, (struct sockaddr*)&serveraddr, serveraddr_len)){
    47. ERRLOG("sendto error");
    48. }
    49. while (1){
    50. if (-1 == (nbytes = recvfrom(socketfd, buff, sizeof(buff), 0, (struct sockaddr*)&serveraddr, &serveraddr_len))){
    51. ERRLOG("recvfrom error");
    52. }
    53. code = ntohs(*(unsigned short*)buff);
    54. number = ntohs(*(unsigned short*)(buff+2));
    55. strncpy(text, buff+4, nbytes-4);
    56. if (code == 3 && number == block_number+1){
    57. block_number++;
    58. if (number == 1){
    59. if (-1 == (fd = open(filename, O_RDWR | O_CREAT | O_TRUNC, 0666))){
    60. ERRLOG("open error");
    61. }
    62. }
    63. write(fd, text, nbytes - 4);
    64. *(unsigned short*)buff = htons(4);
    65. *(unsigned short*)(buff+2) = htons(number);
    66. if (-1 == sendto(socketfd, buff, 4, 0, (struct sockaddr*)&serveraddr, serveraddr_len)){
    67. ERRLOG("sendto error");
    68. }
    69. if (nbytes - 4< 512){
    70. close(fd);
    71. break;
    72. }
    73. }else if(code == 5){
    74. printf("错误\n");
    75. goto rename;
    76. }
    77. }
    78. close(socketfd);
    79. return 0;
    80. }

  • 相关阅读:
    记录一次爬虫接单项目【采集国际淘宝数据】
    软件测试Triangle练习题
    扩展欧几里得
    生产订单自动下达
    企业电子招投标系统源码之电子招投标系统建设的重点和未来趋势
    RFID电网资产全寿命周期管理解决方案
    【动力节点】最新rabbitmq笔记-1-6章What is RabbitMQ?
    《凤凰架构》第三章——事务处理
    Vue中动态绑定class和style
    [每周一更]-(第63期):Linux-nsenter命令使用说明
  • 原文地址:https://blog.csdn.net/a2998658795/article/details/126438180