Linux网络编程(Socket)

目录

  • 网络编程(Socket)概述
    • 引入
    • 网络编程通识扫盲
    • socket套接字
    • 套接字描述符
    • 字节序
  • socket编程步骤
  • Linux提供的API简析
    • 创建套接字即连接协议[socket](服、客)
    • 绑定IP和端口[bind](服)
      • 地址转换api
      • 字节序转换api
    • 监听[listen](服)
    • 接受连接[accept](服)
    • 数据收发[read、write](服、客)
    • 客户端的[connect]函数(客)
  • socket服务端代码实现
  • socket客户端代码实现
  • 实现双方(多方)一直聊天
  • 多方消息收发

网络编程(Socket)概述

引入

前面几个章节讲的进程间通讯均基于同一台Linux内核实现的,因此无法实现多机(和手机、单片机、X86架构等)通讯,因此引入网络通讯,入门先学习Socket(又叫做套接字)网络编程
在这里插入图片描述

问题:两台计算机实现TCP(通过socket编程)通信时,要用到线么??总感觉仅仅通过代码就能建立连接不太靠谱。
回答:TCP连接的基础就是网络连接已经建立好之后,所以物理连接肯定是基础。至于物理连接有很多种,可以是有线的、也可以是无线的,只要协议支持TCP/IP协议就可以。

网络编程通识扫盲

在这里插入图片描述
在这里插入图片描述

socket套接字

socket起源于Unix,而Unix/Linux基本哲学之一就是“一切皆文件”,都可以用“打开open –> 读写write/read –> 关闭close”模式来操作。Socket就是该模式的一个实现, socket即是一种特殊的文件,一些socket函数就是对其进行的操作(读/写IO、打开、关闭)。

Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。

套接字描述符

其实就是一个整数,我们最熟悉的句柄是0、1、2三个,0是标准输入,1是标准输出,2是标准错误输出。0、1、2是整数表示的,对应的FILE *结构的表示就是stdin、stdout、stderr。当应用程序要创建一个套接字时,操作系统就返回一个小整数作为描述符,应用程序则使用这个描述符来引用该套接字需要I/O请求的应用程序请求操作系统打开一个文件。操作系统就创建一个文件描述符提供给应用程序访问文件。从应用程序的角度看,文件描述符是一个整数,应用程序可以用它来读写文件。

字节序

字节序就是字节存储的顺序(从高地址开始存储还是从低地址开始存储),在网络编程中要注意相关协议使用的字节序,防止数据传输出错。具体使用的是字节序转换api,配合端口号使用

在这里插入图片描述
在这里插入图片描述

socket编程步骤

模拟场景:

在这里插入图片描述

步骤介绍:

在这里插入图片描述

Linux提供的API简析

创建套接字即连接协议[socket](服、客)

int socket(int domain,int type,int protocol)

在这里插入图片描述

绑定IP和端口[bind](服)

int bind(int sockfd, const struct sockaddr *addr,socklen_t addrlen);

struct sockaddr 这个结构体一般同等替代成struct sockaddr_in,使用的时候注意新结构体的类型强制转换。

在这里插入图片描述

地址转换api

 int inet_aton(const char *straddr, struct in_addr *addrp);char *inet_ntoa(struct in_addr inaddr);

在这里插入图片描述

字节序转换api

在这里插入图片描述

监听[listen](服)

int listen(int sockfd, int backlog);

backlog:支持最大的连接数

在这里插入图片描述

接受连接[accept](服)

三次握手成功就建立accept连接。函数里面的结构体存放客户端的IP和端口号等信息!

 int accept(int sockfd, struct sockaddr *addr,socklen_t *addrlen, int flags);

在这里插入图片描述

数据收发[read、write](服、客)

和文件read、write用的同一个api。底下最后两对一般用于UDP在这里插入图片描述

数据收发第二套API,多了flags控制参数
在这里插入图片描述

客户端的[connect]函数(客)

 int connect(int sockfd, const struct sockaddr *addr,socklen_t addrlen);

在这里插入图片描述

socket服务端代码实现

利用telnet的方式进行通信,目前还没有写客户端的代码

#include <stdio.h>
#include <sys/types.h>      
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>int main(int argc, char const *argv[])
{int s_fd;int c_fd;int n_read;int n_write;char readBuf[128];char *returnMsg="我收到了你的信息";//发送给客户端的消息 尽量不使用数组struct sockaddr_in c_addr;struct sockaddr_in s_addr;memset(&c_addr,0,sizeof(struct sockaddr_in));memset(&s_addr,0,sizeof(struct sockaddr_in));//数据清空 再配置//1.socket  int socket(int domain, int type, int protocol);s_fd=socket(AF_INET,SOCK_STREAM,0);//ipv4   tcp协议if(s_fd == -1){printf("创建socket失败");perror("socket:");exit(-1);}//2.bind  int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);s_addr.sin_family=AF_INET;//ipv4s_addr.sin_port=htons(8687);//端口号,选择5000以上(有些端口被系统调用)。honts返回网络字节序//int inet_aton(const char *cp, struct in_addr *inp)inet_aton("192.168.103.49",&s_addr.sin_addr)//inet_aton("127.0.0.1",&s_addr.sin_addr);//sin_addr是结构体sockaddr_in里面的结构体 存放IP(下面有查找到原型)   然后转换为网络能识别的格式//或者使用ifconfig命令查到实际本机的IP也可以bind(s_fd,(struct sockaddr *)&s_addr,sizeof(struct sockaddr_in));//结构体类型转换,因为用的同等替换的结构体sockaddr_in//3.listen int listen(int sockfd, int backlog);listen(s_fd,10);//监听10个连接//4.accept int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);int client=sizeof(struct sockaddr_in);       //要求用指针(存放长度)c_fd=accept(s_fd,(struct sockaddr *)&c_addr,&client);//sockaddr_in进行类型强转 存放客户端信息if(c_fd == -1){printf("连接失败\n");perror("accept:");exit(-1);}//客户端printf("客户端的ip:%s\n",inet_ntoa(c_addr.sin_addr)); //把网络格式的ip地址打印成字符串格式//5.read  ssize_t read(int fd, void *buf, size_t count);n_read=read(c_fd,readBuf,128);  //c_fd客户端if(n_read == -1){perror("read:");}else{printf("得到的消息:%d,%s\n",n_read,readBuf);}//6.write ssize_t write(int fd, const void *buf, size_t count);n_write=write(c_fd,returnMsg,strlen(returnMsg));return 0;
}

运行结果(客户端发送 huai dan):

在这里插入图片描述

Tips:在user/include目录下查找头文件。查找结构体sockaddr_in的定义原型、使用了哪个头文件时,我们可以用grep xx* -nir 来实现(n:显示行号;i:不区分大小写;r:递归)。

在这里插入图片描述

进去就能找到这个结构体原型啦:

在这里插入图片描述

在这里插入图片描述

socket客户端代码实现

客户端向服务端发送消息,实现通信(只能进行一次通讯)。

#include <stdio.h>
#include <sys/types.h>      
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>int main(int argc, char const *argv[])
{int c_fd;int n_read;int n_write;int c_connect;char readBuf[128];char *returnMsg="这是来自客户端的信息";struct sockaddr_in c_addr;memset(&c_addr,0,sizeof(struct sockaddr_in));//数据清空//1.socket  int socket(int domain, int type, int protocol);c_fd=socket(AF_INET,SOCK_STREAM,0);//ipv4   tcp协议if(c_fd == -1){printf("创建socket失败");perror("socket:");exit(-1);}c_addr.sin_family=AF_INET;//ipv4c_addr.sin_port=htons(8687);//端口号,选择5000以上。honts返回网络字节序//int inet_aton(const char *cp, struct in_addr *inp)inet_aton("192.168.103.49",&c_addr.sin_addr);//转换为网络能识别的格式//2.connect  int connect(int sockfd, const struct sockaddr *addr,socklen_t addrlen);c_connect=connect(c_fd,(struct sockaddr *)&c_addr,sizeof(struct sockaddr));if(c_connect == -1){printf("连接失败\n");perror("connect:");}//3.write/send ssize_t write(int fd, const void *buf, size_t count);n_write=write(c_fd,returnMsg,strlen(returnMsg));//注意和sizeof的使用区别//4.read  ssize_t read(int fd, void *buf, size_t count);n_read=read(c_fd,readBuf,128);if(n_read == -1){perror("read:");}else{printf("来自服务端的消息:%d,%s\n",n_read,readBuf);}//5.close     int close(int fd);close(c_fd);return 0;
}

运行结果:
在这里插入图片描述

实现双方(多方)一直聊天

本质就是上面的代码加入while循环,实现不断的消息收发

其中服务端使用了两次fork():

  • 第一次是在accept后(即三次握手成功后)创建进程,实现和多个客户端的通信;
  • 第二次fork()创建进程应用在和客户端通信“写”的过程,而“读”放在while循环中,这样就实现了读和写并行运行。

客户端仅用fork创建了一个进程,同样应用在“写”,“读”放在while循环中,这样就实现了读和写并行运行。

服务端代码:

#include <stdio.h>
#include <sys/types.h>      
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>int main(int argc, char const *argv[])
{int s_fd;int c_fd;int n_read;int n_write;char readBuf[128];char returnMsg[128]={0};struct sockaddr_in c_addr;struct sockaddr_in s_addr;memset(&c_addr,0,sizeof(struct sockaddr_in));memset(&s_addr,0,sizeof(struct sockaddr_in));//数据清空if(argc != 3){printf("参数出错\n");exit(-1);}//1.socket  int socket(int domain, int type, int protocol);s_fd=socket(AF_INET,SOCK_STREAM,0);//ipv4   tcp协议if(s_fd == -1){printf("创建socket失败");perror("socket:");exit(-1);}//2.bind  int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);s_addr.sin_family=AF_INET;//ipv4s_addr.sin_port=htons(atoi(argv[2]));//端口号,选择5000以上。honts返回网络字节序,atoi(argv[2])防止端口被占用//int inet_aton(const char *cp, struct in_addr *inp)inet_aton(argv[1],&s_addr.sin_addr);//转换为网络能识别的格式bind(s_fd,(struct sockaddr *)&s_addr,sizeof(struct sockaddr_in));//3.listen int listen(int sockfd, int backlog);listen(s_fd,10);//监听10个连接//4.accept int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);int client=sizeof(struct sockaddr_in);while(1){//不断接收客户端c_fd=accept(s_fd,(struct sockaddr *)&c_addr,&client);if(c_fd == -1){printf("连接失败\n");perror("accept:");exit(-1);}printf("客户端的ip:%s\n",inet_ntoa(c_addr.sin_addr)); //把网络格式的ip地址打印成字符串格式if(fork() == 0){if(fork() == 0){while(1){//不断写入memset(returnMsg,0,sizeof(returnMsg));printf("请输入:\n");gets(returnMsg);//6.write ssize_t write(int fd, const void *buf, size_t count);n_write=write(c_fd,returnMsg,strlen(returnMsg));}}while(1){//不断读取//5.read  ssize_t read(int fd, void *buf, size_t count);memset(readBuf,0,sizeof(readBuf));//不断清空数据防止数据重复出现n_read=read(c_fd,readBuf,128);if(n_read == -1){perror("read:");}else{printf("得到的消息:%d,%s\n",n_read,readBuf);}}}}return 0;
}

客户端代码:

#include <stdio.h>
#include <sys/types.h>      
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>int main(int argc, char const *argv[])
{int ret;int c_fd;int n_read;int n_write;int c_connect;char readBuf[128];char returnMsg[128]={0};char *quit="quit";struct sockaddr_in c_addr;memset(&c_addr,0,sizeof(struct sockaddr_in));//数据清空if(argc != 3){printf("参数出错\n");exit(-1);}//1.socket  int socket(int domain, int type, int protocol);c_fd=socket(AF_INET,SOCK_STREAM,0);//ipv4   tcp协议if(c_fd == -1){printf("创建socket失败");perror("socket:");exit(-1);}c_addr.sin_family=AF_INET;//ipv4c_addr.sin_port=htons(atoi(argv[2]));//端口号,选择5000以上。honts返回网络字节序//int inet_aton(const char *cp, struct in_addr *inp)inet_aton(argv[1],&c_addr.sin_addr);//转换为网络能识别的格式//2.connect  int connect(int sockfd, const struct sockaddr *addr,socklen_t addrlen);c_connect=connect(c_fd,(struct sockaddr *)&c_addr,sizeof(struct sockaddr));if(c_connect == -1){printf("连接失败\n");perror("connect:");}while(1){if(fork() == 0){while(1){//不断写入memset(returnMsg,0,sizeof(returnMsg));printf("请输入:\n");gets(returnMsg);//3.write/send ssize_t write(int fd, const void *buf, size_t count);n_write=write(c_fd,returnMsg,strlen(returnMsg));if(strcmp(quit,returnMsg) == 0){//如果输入quit则客户端就退出exit(0);}}}while(1){//不断读取memset(readBuf,0,sizeof(readBuf));//不断清空数据防止数据重复出现//4.read  ssize_t read(int fd, void *buf, size_t count);n_read=read(c_fd,readBuf,128);if(n_read == -1){perror("read:");}else{printf("来自服务端的消息:%d,%s\n",n_read,readBuf);}}//5.close     int close(int fd);close(c_fd);}return 0;
}

结果:

在这里插入图片描述

多方消息收发

上一节代码其实已经可以实现多方通信了,不过存在两个问题:

  • 1、客户端发消息回车的那一瞬间,光标不知道被哪个进程抢到了,也就是说服务器同一时刻发送的消息,不能确定哪个客户端子进程收到消息。
  • 2、客户端之间无法进行互相通讯。

下面的demo加入了类似心跳包的功能,用来说明服务端其实是知道哪个客户端发来的消息,并且每隔两秒给每个客户端回复。

如果想要完全实现类似QQ聊天机制,思路就是将服务端作为中转站,客户端和客户端之间通过服务器完成聊天功能。当然客户端之间要提前建立“好友”关系,所谓的好友关系可以通过sqlite数据库存储每个客户端的IP、端口、账号等信息,然后服务端后台对这些进行逻辑处理,实现客户端之间的类似QQ聊天机制。

服务端代码

#include <stdio.h>
#include <sys/types.h>      
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>int main(int argc, char const *argv[])
{int mark=0;int s_fd;int c_fd;int n_read;int n_write;char readBuf[128];char returnMsg[128]={0};struct sockaddr_in c_addr;struct sockaddr_in s_addr;memset(&c_addr,0,sizeof(struct sockaddr_in));memset(&s_addr,0,sizeof(struct sockaddr_in));//数据清空if(argc != 3){printf("参数出错\n");exit(-1);}//1.socket  int socket(int domain, int type, int protocol);s_fd=socket(AF_INET,SOCK_STREAM,0);//ipv4   tcp协议if(s_fd == -1){printf("创建socket失败");perror("socket:");exit(-1);}//2.bind  int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);s_addr.sin_family=AF_INET;//ipv4s_addr.sin_port=htons(atoi(argv[2]));//端口号,选择5000以上。honts返回网络字节序,atoi(argv[2])防止端口被占用//int inet_aton(const char *cp, struct in_addr *inp)inet_aton(argv[1],&s_addr.sin_addr);//转换为网络能识别的格式bind(s_fd,(struct sockaddr *)&s_addr,sizeof(struct sockaddr_in));//3.listen int listen(int sockfd, int backlog);listen(s_fd,10);//监听10个连接//4.accept int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);int client=sizeof(struct sockaddr_in);while(1){//不断接收客户端c_fd=accept(s_fd,(struct sockaddr *)&c_addr,&client);if(c_fd == -1){printf("连接失败\n");perror("accept:");exit(-1);}printf("客户端的ip:%s\n",inet_ntoa(c_addr.sin_addr)); //把网络格式的ip地址打印成字符串格式mark++;if(fork() == 0){if(fork() == 0){while(1){sprintf(returnMsg,"欢迎第%d号客户端",mark);//6.write ssize_t write(int fd, const void *buf, size_t count);n_write=write(c_fd,returnMsg,strlen(returnMsg));sleep(20);}}while(1){//5.read  ssize_t read(int fd, void *buf, size_t count);memset(readBuf,0,sizeof(readBuf));n_read=read(c_fd,readBuf,128);if(n_read == -1){perror("read:");}else{printf("得到%d号的消息:%s\n",mark,readBuf);}}}}return 0;
}

客户端代码

#include <stdio.h>
#include <sys/types.h>      
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>int main(int argc, char const *argv[])
{int c_fd;int n_read;int n_write;int c_connect;char readBuf[128];char returnMsg[128]={0};char *quit="quit";struct sockaddr_in c_addr;memset(&c_addr,0,sizeof(struct sockaddr_in));//数据清空if(argc != 3){printf("参数出错\n");exit(-1);}//1.socket  int socket(int domain, int type, int protocol);c_fd=socket(AF_INET,SOCK_STREAM,0);//ipv4   tcp协议if(c_fd == -1){printf("创建socket失败");perror("socket:");exit(-1);}c_addr.sin_family=AF_INET;//ipv4c_addr.sin_port=htons(atoi(argv[2]));//端口号,选择5000以上。honts返回网络字节序//int inet_aton(const char *cp, struct in_addr *inp)inet_aton(argv[1],&c_addr.sin_addr);//转换为网络能识别的格式//2.connect  int connect(int sockfd, const struct sockaddr *addr,socklen_t addrlen);c_connect=connect(c_fd,(struct sockaddr *)&c_addr,sizeof(struct sockaddr));if(c_connect == -1){printf("连接失败\n");perror("connect:");}while(1){if(fork() == 0){while(1){//不断写入memset(returnMsg,0,sizeof(returnMsg));printf("请输入:\n");gets(returnMsg);//3.write/send ssize_t write(int fd, const void *buf, size_t count);n_write=write(c_fd,returnMsg,strlen(returnMsg));if(strcmp(quit,returnMsg) == 0){//如果输入quit则客户端就退出exit(0);}}}while(1){//不断读取memset(readBuf,0,sizeof(readBuf));//不断清空数据防止数据重复出现//4.read  ssize_t read(int fd, void *buf, size_t count);n_read=read(c_fd,readBuf,128);if(n_read == -1){perror("read:");}else{printf("来自服务端的消息:%s\n",readBuf);}}//5.close     int close(int fd);close(c_fd);}return 0;
}

运行结果:
在这里插入图片描述

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

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

相关文章

嵌入式开发概述(树莓派介绍)

目录嵌入式定义嵌入式芯片选型ARM架构ARM树莓派嵌入式定义 国内普遍认同的嵌入式系统的定义是以应用为中心&#xff0c;以计算机技术为基础&#xff0c;软硬件可裁剪&#xff0c;适应应用系统对功耗、可靠性、成本、体积、功耗等严格要求的专用计算机系统。 嵌入式系统是软件和…

ubuntu 16gcc g++版本降级

打算在ubuntu16上编译linux3.4.66内核&#xff0c;但是遇到 In file included from include/linux/compiler.h:48:0, from include/linux/stddef.h:4, from include/linux/posix_types.h:4, from include/linux/types.h:17, from include/linux/page-flags.h:8, from kern…

android 系统gpu 调试_【资讯】高通公布首批可OTA更新GPU驱动手机:谷歌Pixel 4/三星S10在列...

文章转载自&#xff1a;iT之家原文链接&#xff1a;https://www.ithome.com/0/479/483.htm(IT之家3月25日消息) 谷歌今天宣布了一系列面向游戏开发者的新工具&#xff1a;Google Play Asset Delivery、Android性能调节器&#xff0c;Android GPU检查器和Cloud Firestore等&…

cURL库

一、cURL库可以做什么 1.cURL是一个文件传输工具&#xff0c;支持很多协议 二、cURL库的用法 1.初始化 $urlcurl_init() 2.设置选项 curl_setopt($url,CURLOPT_URL&#xff0c;"http://www.baidu.com") curl_setopt($url,string option,mixed value) 3.执行cURL会话 …

树莓派刷机

目录准备刷机重刷准备 SD卡&#xff08;一般16G以上&#xff0c;这里我用的8G&#xff09; 读卡器 Win32DiskImager&#xff08;将镜像烧到SD卡的软件&#xff09;&#xff0c;在这里下载&#xff1a;https://download.csdn.net/download/zhuguanlin121/18329615?spm1001.201…

linux3.4.2移植总结(s3c2440)

环境&#xff1a;Linux version 3.5.0-23-generic (builddkomainu) (gcc version 4.6.3 (Ubuntu/Linaro 4.6.3-1ubuntu5) ) ubuntu12 目标板&#xff1a;JZ2440 待移植内核&#xff1a;3.4.2 交叉编译器&#xff1a;arm-linux-gcc-4.3.2 1. make s3c2410_defconfig //使…

判断图有无环_浅谈什么是图拓扑排序

1 引言 在工程实践中,一个工程项目往往由若干个子项目组成。这些子项目间往往有两种关系:  (1) 先后关系&#xff0c;即必须在某个项完成后才能开始实施另一个子项目。  (2) 子项目间无关系&#xff0c;即两个子项目可以同时进行,互不影响。例如&#xff1a;在工厂里产品的…

Source Insight 4.0安装教程(PS:附安装包及卸载重新安装等注意事项)

目录一、Source Insight 4.0安装包二、删除配置文件&#xff08;初次安装忽略此步骤&#xff09;1、清除注册表信息&#xff1a;2、删除全局配置信息&#xff1a;三、安装步骤1、解压2、安装3、替换4、破解5、安装提示unable to open or create...解决方案一、Source Insight 4…

通过远程(vnc)无法打开qtcreator

最近打算移植qt程序到板子上于是&#xff0c;在ubuntu物理机上装了qt开发环境&#xff0c;通过远程桌面链接上去写qt。 但是远程老是点不开qtcreator。于是通过命令行启动 出了报错信息&#xff0c;应该和xserver 服务有关&#xff0c;没有深入了解错误 下面是我的解决方案&a…

iOS开发UI篇—使用xib自定义UItableviewcell实现一个简单的团购应用界面布局

iOS开发UI篇—使用xib自定义UItableviewcell实现一个简单的团购应用界面布局 iOS开发UI篇—使用xib自定义UItableviewcell实现一个简单的团购应用界面布局 一、项目文件结构和plist文件 二、实现效果 三、代码示例 1.没有使用配套的类&#xff0c;而是直接使用xib文件控件tag值…

硬件模块化开发的例子:LD3320语音模块与Source Insight的使用

目录硬件模块化开发的行情LD3320语音模块使用手册介绍&#xff1a;语音模块验货测试硬件连接效果图测试结果良好Source Insight代码查看器创建工程修改字体配色方案中文乱码关联代码语音识别模块代码分析语音识别模块二次开发硬件模块化开发的行情 LD3320语音模块使用手册介绍&…

ubuntu 远程桌面及Qt安装

环境&#xff1a; ubunut14 x64 目标&#xff1a;通过vnc远程桌面&#xff0c;在ubuntu 上编写qt程序 1.安装远程桌面 apt-get install xrdp vnc4server xbase-clients dconf-editor #安装xrdp&#xff0c;vnc4server apt-get install gnome-panel gnome-settings-daemon m…

awk分割字符

为什么80%的码农都做不了架构师&#xff1f;>>> str1|_|2|_|3|_||_|4|_|5|_|6|_||_|7|_|8|_|9; 期望分割成 1|_|2|_|3 4|_|5|_|6 7|_|8|_|9 也就是以|_||_|做分割符&#xff0c;但是&#xff0c;字符段长度不确定&#xff0c;有可能分割成3段&#xff0c;也可能会有…

面经——嵌入式软件工程师面试遇到的经典题目

参考&#xff1a;嵌入式软件工程师面试遇到的经典题目 作者&#xff1a;一只青木呀 发布时间&#xff1a; 2020-11-04 23:43:16 网址&#xff1a;https://blog.csdn.net/weixin_45309916/article/details/109499825 目录1、找错误2、下面的代码输出是什么&#xff0c;为什么&am…

记一次用WPScan辅助渗透WordPress站点

记一次用WPScan辅助渗透WordPress站点 一、什么是WPScan&#xff1f; WPScan 是一个扫描 WordPress 漏洞的黑盒子扫描器&#xff0c;它可以为所有 Web 开发人员扫描 WordPress 漏洞并在他们开发前找到并解决问题。我们还使用了 Nikto &#xff0c;它是一款非常棒的 Web 服务器评…

什么是Cortex、ARMv8、arm架构、ARM指令集、soc

参考&#xff1a;到底什么是Cortex、ARMv8、arm架构、ARM指令集、soc&#xff1f;一文帮你梳理基础概念【科普】 发布时间&#xff1a; 一口Linux 网址&#xff1a;https://blog.csdn.net/daocaokafei/article/details/109008103 目录前言1. ARM公司2. ARM内核与架构1&#xff…

Windows下Qt5搭建Android开发环境笔记

Windows很大的特点是配置使用几乎都可以图形化进行&#xff0c;和Linux比起来在很多时候配置环境也要方便很多。所以&#xff0c;搭建Qt for Andorid也是十分简单的。需要以下工具&#xff1a;1.最方便的Qt官方包&#xff0c;现在还处于RC阶段&#xff0c;经过测试也是有些小bu…

对于嵌入式交叉编译总结

这几天终于搞定了老师项目里我负责的部分&#xff0c;主要是做一个图像采集的手持端&#xff0c;我选用了JZ2440。 从移植内核、制作文件系统、Qt移植总结下来发现在对于代码的交叉编译必须保证编译平台的一致性。对于s3c2440来说是armv4t&#xff0c;所以当我们编译出来程序必…

linux驱动调试--oops信息

在移植dm9000 时被一个错误困扰了很久&#xff0c;当时手里只有printk调试手段&#xff0c;觉得自己应该升级下了&#xff0c;先学习了根据oops信息来调试。 先构造一个错误&#xff0c;insmod后抛出如下信息 我们着重看这几句 PC is at memcpy0x8c/0x29c c0148080 pc : …

浏览器输入网址后发生了什么?

摘自&#xff1a;这是最全的一篇&#xff01;&#xff01;&#xff01;浏览器输入网址后发什么了什么&#xff1f; 作者&#xff1a;程序员cxuan 发布时间&#xff1a; 2021-04-15 11:59:07 网址&#xff1a;https://blog.csdn.net/qq_36894974/article/details/115720479 到现…