【网络编程开发】4.socket套接字及TCP的实现框架 5.TCP多进程并发

4.socket套接字及TCP的实现框架

Socket套接字

Socket套接字是网络编程中用于实现不同计算机之间通信的一个基本构建块

在现代计算机网络中,Socket套接字扮演着至关重要的角色。它们为应用程序提供了一种方式,通过这种方式,程序能够通过网络发送和接收数据包。以下是对socket的相关介绍:

  1. 定义功能
    • Socket,译为“套接字”,是一种软件形式的"插座",使得不同主机之间的进程可以通过网络进行通信。它基本上充当了应用程序、操作系统与网络硬件之间的接口。
  2. 核心作用
    • Socket套接字允许一个程序通过网络将信息传递给另一个程序。这涉及到底层的网络协议操作,如TCP/IP,但Socket API提供了一个更简单的抽象,使得程序员无需深入了解这些复杂协议的细节即可进行网络编程。
  3. 类型应用
    • 流式套接字(SOCK_STREAM)基于TCP协议,提供面向连接的可靠数据传输服务,适用于需要确保数据完整性的应用,如网页浏览。
    • 数据报套接字(SOCK_DGRAM)基于UDP协议,提供无连接的服务,适用于对实时性要求高但可以容忍少量数据丢失的应用,如视频通话。
  4. 工作过程
    • 服务器端创建一个Socket并绑定到一个特定的IP地址和端口上,然后开始监听来自客户端的连接请求。
    • 客户端创建一个Socket,并尝试连接到服务器的IP地址和端口。一旦连接建立,双方就可以通过这个Socket进行数据的发送和接收。
  5. 高级功能
    • 使用Socket,可以实现多种网络应用程序,包括聊天应用、文件传输、远程控制等。在多用户游戏、实时数据处理等方面也广泛应用Socket技术。
    • 在多线程或多进程的环境中,Socket也能有效地处理并发连接,提高系统的响应速度和处理能力。

TCP实现框架

在这里插入图片描述

以下是C语言中常见的Socket API及其功能:

以下是C语言中常见的Socket API及其功能:

  1. socket函数:该函数用于创建一个套接字,并返回一个套接字描述符。

    #include <sys/types.h>
    #include <sys/socket.h>
    int socket(int domain, int type, int protocol);
    /*
    参数:domain:指定协议族,如AF_INET表示IPv4地址。type:指定套接字类型,如SOCK_STREAM表示提供有序、可靠、双向字节流的连接。protocol:通常设置为0,表示系统将选择与指定类型相匹配的默认协议。
    返回值:成功时返回一个非负整数,表示新创建的套接字描述符。失败时返回-1,并设置errno为相应的错误码。
    */
    
  2. bind函数:此函数用于将套接字绑定到一个特定的地址和端口号上。这对于服务器来说尤其重要,因为它们需要在已知的地址上监听来自客户端的连接请求。

    #include <sys/types.h>
    #include <sys/socket.h>
    int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
    /*
    参数:sockfd:表示套接字的描述符。addr:指向特定于域的套接字地址结构的指针,该结构包含了要绑定的地址信息。addrlen:指定地址结构的大小。
    返回值:成功时返回0。失败时返回-1,并设置errno为相应的错误码。
    */
    
  3. listen函数:服务器使用此函数将套接字设置为监听状态,等待客户端的连接请求。可以指定一个队列限制,以控制同时可处理的连接数。

    #include <sys/types.h>
    #include <sys/socket.h>
    int listen(int sockfd, int backlog);
    /*
    参数:sockfd:表示套接字的描述符。backlog:指定在未完成连接队列中允许的最大连接数。
    返回值:成功时返回0。失败时返回-1,并设置errno为相应的错误码。
    */
    
  4. accept函数:当有客户端连接请求时,服务器使用此函数接受连接,并返回一个新的套接字,用于与客户端通信。

    #include <sys/types.h>
    #include <sys/socket.h>
    int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
    /*
    参数:sockfd:表示套接字的描述符。addr:指向特定于域的套接字地址结构的指针,该结构用于接收客户端的地址信息。addrlen:指向一个变量的指针,该变量用于接收地址结构的大小。
    返回值:成功时返回一个新的套接字描述符,用于与客户端通信。失败时返回-1,并设置errno为相应的错误码。
    */
    
  5. connect函数:该函数用于建立一个到服务器的连接。对于TCP套接字,这将触发TCP的三次握手过程。

    #include <sys/types.h>
    #include <sys/socket.h>
    int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
    /*
    参数:sockfd:表示套接字的描述符。addr:指向特定于域的套接字地址结构的指针,该结构包含了服务器的地址信息。addrlen:指定地址结构的大小。
    返回值:成功时返回0。失败时返回-1,并设置errno为相应的错误码。
    */
    
  6. send和recv函数:这两个函数分别用于发送和接收数据。它们是TCP通信中常用的数据传输函数。

    #include <sys/types.h>
    #include <sys/socket.h>
    ssize_t send(int sockfd, const void *buf, size_t len, int flags);
    ssize_t recv(int sockfd, void *buf, size_t len, int flags);
    /*
    参数:sockfd:表示套接字的描述符。buf:指向要发送或接收数据的缓冲区的指针。len:指定要发送或接收的数据的长度。flags:可选的标志位,用于控制发送或接收的行为。
    返回值:成功时返回实际发送或接收的字节数。失败时返回-1,并设置errno为相应的错误码。
    */
    
  7. sendto和recvfrom函数:对于UDP通信,不需要建立连接,直接使用sendto和recvfrom函数进行数据报文的发送和接收。

    #include <sys/types.h>
    #include <sys/socket.h>
    ssize_t sendto(int sockfd, const void *buf, size_t len, int flags, const struct sockaddr *dest_addr, socklen_t addrlen);
    ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags, struct sockaddr *src_addr, socklen_t *addrlen);
    /*
    参数:sockfd:表示套接字的描述符。buf:指向要发送或接收数据的缓冲区的指针。len:指定要发送或接收的数据的长度。flags:可选的标志位,用于控制发送或接收的行为。dest_addr:指向目标地址结构的指针,用于UDP发送操作。src_addr:指向源地址结构的指针,用于UDP接收操作。addrlen:指向一个变量的指针,用于UDP接收操作,该变量用于接收源地址结构的大小。
    返回值:成功时返回实际发送或接收的字节数。失败时返回-1,并设置errno为相应的错误码。
    */
    

示例-服务端

TCP_socket.c

#include <stdio.h> // 引入标准输入输出库
#include <sys/socket.h> // 引入套接字库
#include <sys/types.h> // 引入类型定义库
#include <stdlib.h> // 引入标准库
#include <arpa/inet.h> // 引入互联网协议库
#include <unistd.h> // 引入Unix标准库
#include <string.h> #define PORT 5001 // 定义端口号为5001
#define BACKLOG 5 // 定义最大连接数为5int main(int argc, char *argv[]) // 主函数,参数为命令行参数个数和参数列表
{int fd,newfd,ret; // 定义文件描述符变量char buf[BUFSIZ] = {}; //BUFSIZ 8142struct sockaddr_in addr; // 定义IPv4地址结构体if(argc < 3){fprintf(stderr,"%s<addr><PORT>",argv[0]);}// 创建套接字fd = socket(AF_INET, SOCK_STREAM, 0);if(fd < 0){perror("socket"); exit(0); }// 设置地址结构体addr.sin_family = AF_INET; // 使用IPv4协议addr.sin_port = htons(atoi(argv[2])); // 设置端口号,atoi()将字符串转换为整数if(inet_aton(argv[1],&addr.sin_addr)==0){//inet_aton()函数将字符串形式的IP地址转换为网络字节序的二进制形式fprintf(stderr,"Invalid address\n");exit(EXIT_FAILURE);}// 绑定通信结构体if(bind(fd, (struct sockaddr *)&addr, sizeof(addr) ) == -1){perror("bind"); exit(0); }// 设置套接字为监听模式if(listen(fd, BACKLOG) == -1){perror("listen"); exit(0); }// 接受客户端的连接请求,生成新的用于和客户端通信的套接字newfd = accept(fd, NULL, NULL);if(newfd < 0){perror("accept"); exit(0);}/*循环接收*/while(1){memset(buf,0,BUFSIZ);ret = read(newfd,buf,BUFSIZ);if(ret < 0){perror("read");exit(0);}else if(ret == 0)break;elseprintf("buf = %s\n",buf);}close(newfd);close(fd); // 关闭套接字return 0; 
}

示例-客户端

TCP_socket2.c

#include <stdio.h> // 引入标准输入输出库
#include <sys/socket.h> // 引入套接字库
#include <sys/types.h> // 引入类型定义库
#include <stdlib.h> // 引入标准库
#include <arpa/inet.h> // 引入互联网协议库
#include <unistd.h> // 引入Unix标准库
#include <string.h>#define PORT 5001 // 定义端口号为5001
#define BACKLOG 5 // 定义最大连接数为5
#define STR "Hello World!" // 定义要发送的字符串为"Hello World!"int main(int argc, char *argv[]) // 主函数,参数为命令行参数个数和参数列表
{int fd; // 定义文件描述符变量struct sockaddr_in addr; // 定义IPv4地址结构体char buf[BUFSIZ] = {}; //BUFSIZ 8142if(argc < 3){fprintf(stderr,"%s<addr><port>\n",argv[0]);exit(0);}// 创建套接字fd = socket(AF_INET, SOCK_STREAM, 0);if(fd < 0){perror("socket"); // 如果创建失败,打印错误信息exit(0); // 退出程序}// 设置地址结构体addr.sin_family = AF_INET; // 使用IPv4协议addr.sin_port = htons(atoi(argv[2])); // 设置端口号,atoi()将字符串转换为整数if(inet_aton(argv[1],&addr.sin_addr)==0){//inet_aton()函数将字符串形式的IP地址转换为网络字节序的二进制形式fprintf(stderr,"Invalid address\n");exit(EXIT_FAILURE);}    // 向服务端发起连接请求if(connect(fd, (struct sockaddr *)&addr, sizeof(addr) ) == -1){perror("connect"); exit(0); }/*循环发送*/while(1){printf(">");fgets(buf,BUFSIZ,stdin);write(fd,buf,strlen(buf));}// 关闭套接字close(fd);return 0; // 返回0表示程序正常结束
}

在这里插入图片描述

通信成功

如果结束程序,再次运行服务端程序,会出现报错如下(地址已经被使用)

在这里插入图片描述

解决办法:

在程序中加入如下程序段:

/*地址快速重用*/
int flag=1,len= sizeof (int); 
if ( setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &flag, len) == -1) { perror("setsockopt"); exit(1); 
} 

5.TCP多进程并发

示例-服务端

my_sever.c

#include <stdio.h> // 引入标准输入输出库
#include <sys/socket.h> // 引入套接字库
#include <sys/types.h> // 引入类型定义库
#include <stdlib.h> // 引入标准库
#include <arpa/inet.h> // 引入网络地址库
#include <unistd.h> // 引入Unix系统调用库
#include <string.h> // 引入字符串处理库
#include <strings.h> // 引入字符串操作库
#include <signal.h> // 引入信号处理库
#include <sys/wait.h> // 引入进程等待库#define BACKLOG 5 // 定义最大连接数为5void ClinetHandle(int newfd); // 客户端处理函数/*信号处理函数,防止出现僵尸进程*/
void SigHandle(int sig){ if(sig == SIGCHLD){ // 如果接收到SIGCHLD信号printf("client exited\n"); // 打印客户端退出信息wait(NULL); // 等待子进程结束}
}/*主函数*/
int main(int argc, char *argv[])
{int fd, newfd; // 定义文件描述符变量struct sockaddr_in addr, clint_addr; // 定义套接字地址结构体socklen_t addrlen = sizeof(clint_addr); // 定义地址长度变量#if 0struct sigaction act; // 定义信号处理结构体act.sa_handler = SigHandle; // 设置信号处理函数act.sa_flags = SA_RESTART; // 设置信号处理标志sigemptyset(&act.sa_mask); // 清空信号屏蔽集sigaction(SIGCHLD, &act, NULL); // 注册信号处理函数
#elsesignal(SIGCHLD, SigHandle); // 注册信号处理函数
#endifpid_t pid; // 定义进程ID变量if(argc < 3){ // 如果参数个数小于3fprintf(stderr, "%s<addr><port>\n", argv[0]);exit(0); }/*创建套接字*/fd = socket(AF_INET, SOCK_STREAM, 0); // 创建TCP套接字if(fd < 0){ perror("socket"); }addr.sin_family = AF_INET; // 设置地址族为IPv4addr.sin_port = htons( atoi(argv[2]) ); // 设置端口号if ( inet_aton(argv[1], &addr.sin_addr) == 0) { // 如果转换IP地址失败fprintf(stderr, "Invalid address\n"); exit(EXIT_FAILURE); }/*地址快速重用*/int flag=1,len= sizeof (int); if ( setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &flag, len) == -1) { // 如果设置失败perror("setsockopt"); exit(1); } /*绑定通信结构体*/if(bind(fd, (struct sockaddr *)&addr, sizeof(addr) ) == -1){ // 如果绑定失败perror("bind"); exit(0); }/*设置套接字为监听模式*/if(listen(fd, BACKLOG) == -1){ perror("listen"); exit(0); }while(1){ // 循环监听连接请求/*接受客户端的连接请求,生成新的用于和客户端通信的套接字*/newfd = accept(fd, (struct sockaddr *)&clint_addr, &addrlen); // 接受连接请求if(newfd < 0){ perror("accept"); exit(0); }/*打印客户端地址和端口号*/ // inet_ntoa()将网络字节序转换为点分十进制IP地址格式的字符串;// ntohs()将一个16位数从网络字节顺序转换为主机字节顺序printf("addr:%s port:%d\n", inet_ntoa(clint_addr.sin_addr), ntohs(clint_addr.sin_port) ); if( (pid = fork() ) < 0){ perror("fork"); exit(0); }else if(pid == 0){ // 如果当前进程是子进程close(fd); // 关闭父进程的文件描述符ClinetHandle(newfd); // 处理客户端请求exit(0); }elseclose(newfd); // 关闭子进程的文件描述符}close(fd); // 关闭服务器套接字return 0; 
}/*客户端处理函数*/
void ClinetHandle(int newfd){int ret; // 定义返回值变量char buf[BUFSIZ] = {}; // 定义缓冲区数组并初始化为0while(1){ // 循环读取客户端数据//memset(buf, 0, BUFSIZ); // 清空缓冲区数组bzero(buf, BUFSIZ); // 使用bzero函数清空缓冲区数组ret = read(newfd, buf, BUFSIZ); // 从客户端读取数据if(ret < 0) {perror("read"); exit(0); }else if(ret == 0) // 如果读取到EOF标志break; // 跳出循环elseprintf("buf = %s\n", buf); // 打印读取到的数据}close(newfd); // 关闭与客户端的连接
}

示例-客户端

my_client.c

#include <stdio.h> // 引入标准输入输出库
#include <sys/socket.h> // 引入套接字库
#include <sys/types.h> // 引入类型定义库
#include <stdlib.h> // 引入标准库
#include <arpa/inet.h> // 引入互联网协议族库
#include <unistd.h> // 引入Unix标准库
#include <string.h> // 引入字符串处理库#define BACKLOG 5 // 定义最大连接数为5int main(int argc, char *argv[]) // 主函数,参数为命令行参数个数和参数列表
{int fd; // 定义文件描述符变量struct sockaddr_in addr; // 定义套接字地址结构体char buf[BUFSIZ] = {}; // 定义缓冲区数组并初始化为0if(argc < 3){ // 如果参数个数小于3fprintf(stderr, "%s<addr><port>\n", argv[0]); // 打印错误信息exit(0); // 退出程序}/*创建套接字*/fd = socket(AF_INET, SOCK_STREAM, 0); // 创建TCP套接字if(fd < 0){ perror("socket"); exit(0); }addr.sin_family = AF_INET; // 设置地址族为IPv4addr.sin_port = htons( atoi(argv[2]) ); // 设置端口号if ( inet_aton(argv[1], &addr.sin_addr) == 0) { fprintf(stderr, "Invalid address\n"); exit(EXIT_FAILURE); }/*向服务端发起连接请求*/if(connect(fd, (struct sockaddr *)&addr, sizeof(addr) ) == -1){ // 如果连接失败perror("connect"); exit(0); }while(1){ // 循环读取用户输入的数据printf("Input->"); // 提示用户输入fgets(buf, BUFSIZ, stdin); // 从标准输入读取数据到缓冲区write(fd, buf, strlen(buf) ); // 将缓冲区的数据发送给服务器}close(fd); // 关闭套接字return 0; 
}

在这里插入图片描述

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

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

相关文章

2024年城市建设与环境管理国际会议(ICUCEM 2024)

2024 International Conference on Urban Construction and Environmental Management 【1】大会信息 大会地点&#xff1a;中国成都 投稿邮箱&#xff1a;icucemsub-paper.com 【2】会议简介 2024年城市建设与环境管理国际会议是一个专注于探讨城市建设与环境管理前沿议题…

EasyRecovery2024终极破解指南来袭!

在数字化时代&#xff0c;数据成为了每个人、每家公司最宝贵的资产之一。无论是个人的照片、文档&#xff0c;还是公司的机密信息&#xff0c;一旦丢失&#xff0c;后果不堪设想。因此&#xff0c;数据恢复工具如EasyRecovery2024应运而生&#xff0c;成为了保护数据安全的利器…

kubeedge v1.17.0部署教程

文章目录 前言一、安装k8s平台二、部署kubeedge1.部署MetalLB(可选)2.cloud3.edge4. 部署nginx到edge端 总结参考 前言 本文主要介绍kubeedge v1.17.0的安装过程 主要环境如下表 应用版本centos7.0k8s1.28.2kubeedge1.17.0docker24.0.8centos7.0 一、安装k8s平台 本文主要参…

Doris Connector 结合 Flink CDC 实现 MySQL 分库分表

1. 概述 在实际业务系统中为了解决单表数据量大带来的各种问题&#xff0c;我们通常采用分库分表的方式对库表进行拆分&#xff0c;以达到提高系统的吞吐量。 但是这样给后面数据分析带来了麻烦&#xff0c;这个时候我们通常试将业务数据库的分库分表同步到数据仓库时&#x…

英伟达Docker 安装与GPu镜像拉取

获取nvidia_docker压缩包nvidia_docker.tgz将压缩包上传至服务器指定目录解压nvidia_docker.tgz压缩包 tar -zxvf 压缩包执行rpm安装命令&#xff1a; #查看指定rpm包安装情况 rpm -qa | grep libstdc #查看指定rpm包下的依赖包的版本情况 strings /lib64/libstdc |grep GLI…

【代码随想录】【算法训练营】【第25天】 [216]组合总和III [17] 电话号码的字母组合

前言 思路及算法思维&#xff0c;指路 代码随想录。 题目来自 LeetCode。 day 25&#xff0c;周六&#xff0c;坚持有点困难~ 题目详情 [216] 组合总和III 题目描述 216 组合总和III 解题思路 前提&#xff1a;组合子集问题 思路&#xff1a;回溯算法&#xff0c;剪枝…

347. 前 K 个高频元素

题目 给你一个整数数组 nums 和一个整数 k &#xff0c;请你返回其中出现频率前 k 高的元素。你可以按 任意顺序 返回答案。 示例 1: 输入: nums [1,1,1,2,2,3], k 2 输出: [1,2] 示例 2: 输入: nums [1], k 1 输出: [1] 提示&#xff1a; 1 < nums.length < 1…

基于聚类与统计检验深度挖掘电商用户行为

1.项目背景 在当今竞争激烈的电商市场中,了解用户的行为和需求对于制定成功的市场策略至关重要,本项目通过建立RFM模型、K-Means聚类模型,将1000个用户进行划分,针对不同类的用户,提出不同的营销策略,最后通过统计检验来探究影响用户消费行为的因素和影响用户上网行为的…

DBeaver连接Oracle报错:ORA-12514

Listener refused the connection with the following error:ORA-12514, TNS:listener does not currently know of service requested inconnect descriptor ———————————————— 1.报错信息2.配置正确结语 ———————————————— 如果是第一次连接Or…

Spring 使用SSE(Server-Sent Events)学习

什么是SSE SSE 即服务器发送事件&#xff08;Server-Sent Events&#xff09;&#xff0c;是一种服务器推送技术&#xff0c;允许服务器在客户端建立连接后&#xff0c;主动向客户端推送数据。 SSE 基于 HTTP 协议&#xff0c;使用简单&#xff0c;具有轻量级、实时性和断线重…

Go微服务: 基于rocketmq:5.2.0搭建RocketMQ环境,以及示例参考

概述 参考最新官方文档&#xff1a;https://rocketmq.apache.org/zh/docs/quickStart/03quickstartWithDockercompose以及&#xff1a;https://rocketmq.apache.org/zh/docs/deploymentOperations/04Dashboard综合以上两个文档来搭建环境 搭建RocketMQ环境 1 ) 基于 docker-c…

速通数据挖掘课程

速通 数据挖掘课程 大的分类 标签预测&#xff08;分类&#xff09; 和 数值预测&#xff08;预测呀&#xff09; 监督 非监督 是否 需要预先训练模型 然后预测 聚类&#xff1a;拿一个比一个&#xff0c;看看相似否&#xff0c;然后归一类 数据四种类型 数据属性有四种&…

simplicity studio 5 修改设备电压

工装板的soc额定输入电压为1.5v&#xff0c;而常态下ttl高电平为5v/3.3v&#xff0c;所以需要设定烧录程序时的设备电压。 确保连接设备&#xff0c;并且被识别。 进入管理员模式。 烧录.hex文件快捷方法。

人工智能芯片封装技术及应用趋势分析

简介人工智能&#xff08;AI&#xff09;、物联网&#xff08;IoT&#xff09;和大数据的融合正在开创全新的智能时代&#xff0c;以智能解决方案改变各行各业。人工智能芯片在支持人工智能学习和推理计算方面发挥着非常重要的作用&#xff0c;可实现各行各业的多样化应用。 本…

HTML+CSS+JS 动态登录表单

效果演示 实现了一个登录表单的背景动画效果,包括一个渐变背景、一个输入框和一个登录按钮。背景动画由多个不同大小和颜色的正方形组成,它们在页面上以不同的速度和方向移动。当用户成功登录后,标题会向上移动,表单会消失。 Code <!DOCTYPE html> <html lang=&q…

电阻、电容和电感测试仪设计

在现代化生产、学习、实验当中,往往需要对某个元器件的具体参数进行测量,在这之中万用表以其简单易用,功耗低等优点被大多数人所选择使用。然而万用表有一定的局限性,比如:不能够测量电感,而且容量稍大的电容也显得无能为力。所以制作一个简单易用的电抗元器件测量仪是很…

智能视频监控平台LntonCVS视频汇聚共享平台智慧楼宇应用方案

随着城市经济的迅速发展&#xff0c;大中型城市的写字楼数量不断增加。在像香港、台北、上海、北京等大城市&#xff0c;写字楼的安保成本相当高。为了降低这一成本&#xff0c;越来越多的物业公司开始采用技术手段。写字楼安防监控系统便是其中之一&#xff0c;它利用安全防范…

【control_manager】无法加载,gazebo_ros2_control 0.4.8,机械臂乱飞

删除URDF和SDRF文件中的特殊注释#, !,&#xff1a; xacro文件解析为字符串时出现报错 一开始疯狂报错Waiting for /controller_manager node to exist 1717585645.4673686 [spawner-2] [INFO] [1717585645.467015300] [spawner_joint_state_broadcaster]: Waiting for /con…

这家公司的39亿存款,无法收回了?

新闻提要 4日晚间&#xff0c;亿利洁能发布公告称&#xff0c;亿利财务公司对于公司存放在亿利财务公司的 39.06 亿元货币资金的用途主要是向亿利集团及其关联方发放贷款&#xff0c;近日公司获悉相关贷款已被划分为次级贷款&#xff08;不良贷款的一种&#xff09;&#xff0…

Pinterest免费引流实操演示

这篇文章中你将了解到 1.Pinterest网站介绍&#xff0c;用户群体&#xff0c;适合做什么品类。 2.现在的商家都在上面做什么&#xff1f;案例展示。 3.我们在这个站免费引流要怎么做以及注意事项。 1.Pinterest网站介绍&#xff0c;用户群体&#xff0c;适合做什么品类。 P…