【网络】:网络套接字(TCP)

news2025/1/13 13:06:21

网络套接字(TCP)

  • 一.编写TCP服务器
  • 二.编写Tcp客户端
  • 三.多进程
  • 四.多线程版本
  • 五.线程池版+完整源代码
  • 六.使用示例

一.编写TCP服务器

1.先搭一个架子

在这里插入图片描述

2.创建sockfd

在这里插入图片描述

domain参数依然是AF_INET(因为是IPV4)

在这里插入图片描述

type方式选择SOCK_STREAM(提供可靠的连接方式)

在这里插入图片描述

protocol协议类型默认缺省0

当然创建套接字是有可能失败的,所以这里加了一个日志(lg)文件,如果失败就打印日志(下面也会有该日志的完整源代码),注意:也可以直接打印在显示器上,但用日志更规范。

在这里插入图片描述

3创建并填充结构体

这里需要注意的是,填port时要使用htons函数转换成网络字节序(也就是大端),同时ip也要从字符串类型转换成int类型(有多种函数,这里选择inet_aton)。具体原因可以看上一篇博客(Udp)。

在这里插入图片描述

4.绑定内核

在这里插入图片描述

参数:1.绑定的套接字,2.绑定的结构体,3.结构体大小。

在这里插入图片描述

5.进行监听

TCP与UDP不同的是,它是面向连接的,服务器一般是比较“被动”的,服务器一直要处于等待连接的状态,我们一般称为监听。

在这里插入图片描述

在这里插入图片描述

6.对main函数进行构建,使用命令行传参方式传入端口号

在这里插入图片描述

7.接着编写Start部分代码

主要分为两步:1.获取新连接,2.根据新连接来通信。

在这里插入图片描述

这里的三个参数不必多说,关键在它的返回值也是一个套接字。它与最开始创建的套接字的功能划分不同,返回的这个套接字才是主要提供服务的。而最开始的套接字是帮助创建监听的,所以一般称它为listensock_。

在这里插入图片描述

8.编写服务代码

由于Tcp是面向字节流的,所以我们可以直接使用文件操作函数进行操作。

在这里插入图片描述

9.进行阶段性测试

由于还没有写客户端,所以我们可以使用telnet,它可以指定服务远程登陆,ip:127.0.0.1表示本地环回,端口号8888。如果没有该命令,可以yum -y install telnet。注意:下面我将Main.cc文件编译成了Tcpserver,启动该文件,在后面跟上端口号8888.

在这里插入图片描述

接着ctrl+6+],然后回车,就可以发消息了。

在这里插入图片描述

退出:ctrl+6+],输入quit。

在这里插入图片描述

二.编写Tcp客户端

1.创建套接字

在这里插入图片描述

2.绑定

客户端需要进行绑定吗?答案是需要的,但是不需要像服务器一样显示的绑定。因为无论是Udp还是Tcp,它都是socket进行通信,一定需要ip+port来进行区别,这样多个主机间才能进行区分。对于服务器来说,这个端口号要确定,因为其它客户端要对服务器发消息,就需要知道服务器的端口号。对于客户端来说,这个端口号是多少不重要,重要的是唯一,因为客户端对服务器发消息后,服务器就能通过结构体知道你的端口号和ip,所以客户端不需要显示的绑定。

Udp是在第一次发消息时绑定,而Tcp在发起connect时进行绑定。

在这里插入图片描述

在生活中,我们通过各种广告推广或者商店来获取服务器的ip和port,在这里我们修改一下main函数来收到传入服务器的ip和port。

在这里插入图片描述

3.测试

在这里插入图片描述

在这里插入图片描述

这样算是完成了一个框架,但有一个问题是目前是单进程,也就是说同时只能有一个客户端,很明显不符合预期,接下来改成多进程版。

三.多进程

在这里插入图片描述

在这里插入图片描述

首先为什么要close(listensockfd_)呢?因为我们只需要父进程一个提供监听功能就行了。为什么要close(sockfd)呢?因为我们将具体的服务功能交给了子进程,所以父进程就不需要这个fd了。当然这并没有完成多进程版,因为父进程会一直阻塞,有多种解决办法,这里选择再创建孙子进程,让子进程直接释放,孙子进程交给OS管理,这样父进程就不会阻塞了。(不了解的可以看看博客进程等待)

在这里插入图片描述

但是这种创建进程的成本实在太高了,一般连个三四十个就无了,所以为了高性能,我们可以搞一个多线程版本。

四.多线程版本

在创建线程后,我们使用pthread_detach,将主线程和其它线程分离,这样在其它线程提供服务时,主线程依然能自主运行。(不了解该函数的可以看博客线程(二))

在这里插入图片描述

在这里插入图片描述

对于线程创建不熟悉的可以看看博客线程(一),这里因为线程函数是在类内创建的,会自带一个this指针,不符合参数要求,所以改成静态的。接下来给线程函数进行传参。(由于线程函数是静态的,不能直接使用类内函数,所以我们可以将Tcpserver类一并当参数传入)

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

在这里插入图片描述

显然创建线程依然有代价,服务器在运行时一直有用户进入退出,所以会不断创建销毁,那么我们如何将线程多次利用呢?下面来线程池版。

五.线程池版+完整源代码

具体线程池代码如何写,可以看我的博客线程池,这里就直接使用了。简单来说就是向线程池里不断push任务,然后多个线程去竞争任务。 可以使用该命令检测线程个数ps -aL | head -1 && ps -aL | grep 程序名。

注意:为了保证服务器不被长时间连接,当前服务是只提供一次,使用客户端发消息时,每发一条就会重新建立一个连接。

在这里插入图片描述

makefile

.PHONY:all
all:Tcpserver Tcpclient

Tcpserver:Main.cc
	g++ -o $@ $^ -std=c++11 -lpthread
Tcpclient:Tcpclient.cc
	g++ -o $@ $^ -std=c++11


.PHONY:clean
clean:
	rm -f Tcpclient Tcpserver

log.hpp

#pragma once

#include <iostream>
#include <time.h>
#include <stdarg.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>

#define SIZE 1024

#define Info 0
#define Debug 1
#define Warning 2
#define Error 3
#define Fatal 4

#define Screen 1
#define Onefile 2
#define Classfile 3

#define LogFile "log.txt"

class Log
{
public:
  Log()
  {
    printMethod = Screen;
    path = "./log/";
  }
  void Enable(int method)
  {
    printMethod = method;
  }
  std::string levelToString(int level)
  {
    switch (level)
    {
    case Info:
      return "Info";
    case Debug:
      return "Debug";
    case Warning:
      return "Warning";
    case Error:
      return "Error";
    case Fatal:
      return "Fatal";
    default:
      return "None";
    }
  }

  void printLog(int level, const std::string &logtxt)
  {
    switch (printMethod)
    {
    case Screen:
      std::cout << logtxt << std::endl;
      break;
    case Onefile:
      printOneFile(LogFile, logtxt);
      break;
    case Classfile:
      printClassFile(level, logtxt);
      break;
    default:
      break;
    }
  }
  void printOneFile(const std::string &logname, const std::string &logtxt)
  {
    std::string _logname = path + logname;
    int fd = open(_logname.c_str(), O_WRONLY | O_CREAT | O_APPEND, 0666); // "log.txt"
    if (fd < 0)
      return;
    write(fd, logtxt.c_str(), logtxt.size());
    close(fd);
  }
  void printClassFile(int level, const std::string &logtxt)
  {
    std::string filename = LogFile;
    filename += ".";
    filename += levelToString(level); // "log.txt.Debug/Warning/Fatal"
    printOneFile(filename, logtxt);
  }

  ~Log()
  {
  }
  void operator()(int level, const char *format, ...)
  {
    time_t t = time(nullptr);
    struct tm *ctime = localtime(&t);
    char leftbuffer[SIZE];
    snprintf(leftbuffer, sizeof(leftbuffer), "[%s][%d-%d-%d %d:%d:%d]", levelToString(level).c_str(),
             ctime->tm_year + 1900, ctime->tm_mon + 1, ctime->tm_mday,
             ctime->tm_hour, ctime->tm_min, ctime->tm_sec);

    va_list s;
    va_start(s, format);
    char rightbuffer[SIZE];
    vsnprintf(rightbuffer, sizeof(rightbuffer), format, s);
    va_end(s);

    // 格式:默认部分+自定义部分
    char logtxt[SIZE * 2];
    snprintf(logtxt, sizeof(logtxt), "%s %s", leftbuffer, rightbuffer);

    // printf("%s", logtxt); // 暂时打印
    printLog(level, logtxt);
  }

private:
  int printMethod;
  std::string path;
};

Log lg;

ThreadPool.hpp

#pragma once

#include <iostream>
#include <vector>
#include <string>
#include <queue>
#include <pthread.h>
#include <unistd.h>

struct ThreadInfo
{
  pthread_t tid;
  std::string name;
};

static const int defalutnum = 5;

template <class T>
class ThreadPool
{
public:
  void Lock()
  {
    pthread_mutex_lock(&mutex_);
  }
  void Unlock()
  {
    pthread_mutex_unlock(&mutex_);
  }
  void Wakeup()
  {
    pthread_cond_signal(&cond_);
  }
  void ThreadSleep()
  {
    pthread_cond_wait(&cond_, &mutex_);
  }
  bool IsQueueEmpty()
  {
    return tasks_.empty();
  }
  std::string GetThreadName(pthread_t tid)
  {
    for (const auto &ti : threads_)
    {
      if (ti.tid == tid)
        return ti.name;
    }
    return "None";
  }

public:
  static void *HandlerTask(void *args)
  {
    ThreadPool<T> *tp = static_cast<ThreadPool<T> *>(args);
    std::string name = tp->GetThreadName(pthread_self());
    while (true)
    {
      tp->Lock();

      while (tp->IsQueueEmpty())
      {
        tp->ThreadSleep();
      }
      T t = tp->Pop();
      tp->Unlock();

      t();
    }
  }
  void Start()
  {
    int num = threads_.size();
    for (int i = 0; i < num; i++)
    {
      threads_[i].name = "thread-" + std::to_string(i + 1);
      pthread_create(&(threads_[i].tid), nullptr, HandlerTask, this);
    }
  }
  T Pop()
  {
    T t = tasks_.front();
    tasks_.pop();
    return t;
  }
  void Push(const T &t)
  {
    Lock();
    tasks_.push(t);
    Wakeup();
    Unlock();
  }
  static ThreadPool<T> *GetInstance()
  {
    if (nullptr == tp_) // ???
    {
      pthread_mutex_lock(&lock_);
      if (nullptr == tp_)
      {
        std::cout << "log: singleton create done first!" << std::endl;
        tp_ = new ThreadPool<T>();
      }
      pthread_mutex_unlock(&lock_);
    }

    return tp_;
  }

private:
  ThreadPool(int num = defalutnum) : threads_(num)
  {
    pthread_mutex_init(&mutex_, nullptr);
    pthread_cond_init(&cond_, nullptr);
  }
  ~ThreadPool()
  {
    pthread_mutex_destroy(&mutex_);
    pthread_cond_destroy(&cond_);
  }
  ThreadPool(const ThreadPool<T> &) = delete;
  const ThreadPool<T> &operator=(const ThreadPool<T> &) = delete; // a=b=c
private:
  std::vector<ThreadInfo> threads_;
  std::queue<T> tasks_;

  pthread_mutex_t mutex_;
  pthread_cond_t cond_;

  static ThreadPool<T> *tp_;
  static pthread_mutex_t lock_;
};

template <class T>
ThreadPool<T> *ThreadPool<T>::tp_ = nullptr;

template <class T>
pthread_mutex_t ThreadPool<T>::lock_ = PTHREAD_MUTEX_INITIALIZER;

Task.hpp

#pragma once
#include <iostream>
#include <string>
#include <string.h>
#include "log.hpp"

extern Log lg;

class Task
{
public:
  Task(int sockfd, const std::string &clientip, const uint16_t &clientport)
      : sockfd_(sockfd), clientip_(clientip), clientport_(clientport)
  {
  }
  Task()
  {
  }
  void run()
  {
    // 测试代码,收一个消息再发回去
    char buffer[4096];
    ssize_t n = read(sockfd_, buffer, sizeof(buffer));
    if (n > 0)
    {
      buffer[n] = 0;
      std::cout << "client say#" << buffer << std::endl;
      std::string echo_string = "tcpserver echo#";
      echo_string += buffer;

      n=write(sockfd_, echo_string.c_str(), echo_string.size());
      if(n<0)
      {
        lg(Warning, "write error,errorno:%d,errstring:%s", errno, strerror(errno));
      }
    }
    else if (n == 0) // 结束服务
    {
      lg(Info, "quit,sever close sockfd:%d", sockfd_);
    }
    else // 出错
    {
      lg(Warning, "read error,sockfd:%d,client port:%d,client ip:%s", sockfd_, clientport_, clientip_.c_str());
    }
    close(sockfd_);
  }
  void operator()()
  {
    run();
  }
  ~Task()
  {
  }

private:
  int sockfd_;
  std::string clientip_;
  uint16_t clientport_;
};

Tcpserver.hpp

#pragma once

#include <iostream>
#include <string>
#include <cstdlib>
#include <cstring>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <pthread.h>
#include <signal.h>
#include <signal.h>
#include "Task.hpp"
#include "log.hpp"
#include "ThreadPool.hpp"
const int defaultfd = -1;
const std::string defaultip = "0.0.0.0";
const int backlog = 10; // 一般不要设置太大



// 枚举错误代码,方便查阅
enum
{
  UseageError = 1,
  SocketError,
  BindError,
  ListenError,
};


// //线程传参
// class ThreadData
// {
// public:
//   ThreadData(int fd,const std::string &ip,uint16_t &p,TcpServer*tp): sockfd(fd),clientip(ip),clientport(p),tsvr(tp)
//   {}
// public:
//   int sockfd;
//   std::string clientip;
//   uint16_t clientport;
//   TcpServer* tsvr;
// };


class TcpServer
{
public:
  TcpServer(const uint16_t &port, const std::string &ip = defaultip) : listensockfd_(defaultfd), port_(port), ip_(ip)
  {
  }
  void InitServer()
  {
    listensockfd_ = socket(AF_INET, SOCK_STREAM, 0);
    if (listensockfd_ < 0) // 创建失败
    {
      lg(Fatal, "create listensockfd,errno:%d,errstring:%s", errno, strerror(errno));
      exit(SocketError);
    }

    // 如果成功,打印成功信息
    lg(Info, "create listensockfd sucess:%d", listensockfd_);

    // 创建并填充结构体
    struct sockaddr_in local;                  // 创建结构体
    memset(&local, 0, sizeof(local));          // 将内部清零
    local.sin_family = AF_INET;                // 填充类型
    local.sin_port = htons(port_);             // 填充端口号
    inet_aton(ip_.c_str(), &(local.sin_addr)); // 填充ip地址

    // 绑定
    if (bind(listensockfd_, (struct sockaddr *)&local, sizeof(local))) // 如果绑定失败
    {
      lg(Fatal, "bind listensockfd,errno:%d,errstring:%s", errno, strerror(errno));
      exit(BindError);
    }

    // 监听
    if (listen(listensockfd_, backlog) < 0) // 如果失败
    {
      lg(Fatal, "listen error,errno:%d,errstring:%s", errno, strerror(errno));
      exit(ListenError);
    }
    lg(Info, "listen sucess,listensockfd:%d", listensockfd_);
  }

  // //线程函数
  // static void* Routine(void*args)
  // {
  //   pthread_detach(pthread_self());//分离线程

  //   ThreadData*td=static_cast<ThreadData*>(args);
  //   td->tsvr->Serverce(td->sockfd,td->clientip,td->clientport);//提供服务
  //   delete td;
  //   return nullptr;
  // }

  void Start()
  {
    signal(SIGPIPE,SIG_IGN);
    ThreadPool<Task>::GetInstance()->Start();//启动线程池
    lg(Info, "tcpServer if running.....");
    while (true)
    {
      // 1.获取新连接
      struct sockaddr_in client;
      socklen_t len = sizeof(client);
      int sockfd = accept(listensockfd_, (struct sockaddr *)&client, &len);
      if (sockfd < 0)
      {
        lg(Warning, "accept error,errorno:%d,errstring:%s", errno, strerror(errno));
        continue;
      }

      // 接收客户端的port和ip
      uint16_t clientport = ntohs(client.sin_port);
      char clientip[32];
      inet_ntop(AF_INET, &(client.sin_addr), clientip, sizeof(clientip));

      // 2.根据新连接进行通信
      lg(Info, "get a new link,sockfd:%d,client port:%d,client ip:%s", sockfd, clientport, clientip);

      // 3.服务
      //3.1 单进程版
      //Serverce(sockfd, clientip, clientport);
      //close(sockfd);

      //3.2 多进程版
      // pid_t id=fork();
      // if(id==0)//子进程
      // {
      //   close(listensockfd_);
      //   if(fork()>0) exit(0);//创建孙子进程,释放子进程
      //   Serverce(sockfd, clientip, clientport);//孙子进程提供服务
      //   close(sockfd);//服务完成,关闭服务
      //   exit(0);//退出进程
      // }
      // //父进程
      // close(sockfd);
      // pid_t rid=waitpid(id,nullptr,0);
      // (void)rid;

      //3.3多线程版本
      // ThreadData*td=new ThreadData(sockfd,clientip,clientport,this);
      // pthread_t tid;
      // pthread_create(&tid,nullptr,Routine,td);

      //3.4线程池版本
      Task t(sockfd, clientip, clientport);
      ThreadPool<Task>::GetInstance()->Push(t);

    }
  }

  // void Serverce(const int sockfd, const std::string &clientip, const uint16_t clientport)
  // {
  //   // 测试代码,收一个消息再发回去
  //   while (true)
  //   {
  //     char buffer[4096];
  //     ssize_t n = read(sockfd, buffer, sizeof(buffer));
  //     if (n > 0)
  //     {
  //       buffer[n] = 0;
  //       std::cout << "buffer say#" << buffer << std::endl;
  //       std::string echo_string = "tcpserver echo#";
  //       echo_string += buffer;

  //       write(sockfd, echo_string.c_str(), echo_string.size());
  //     }
  //     else if(n==0)//结束服务
  //     {
  //       lg(Info,"quit,sever close sockfd:%d",sockfd);
  //       break;
  //     }
  //     else//出错
  //     {
  //       lg(Warning, "read error,sockfd:%d,client port:%d,client ip:%s", sockfd, clientport, clientip.c_str());
  //       break;
  //     }
  //   }
  // }

  ~TcpServer()
  {
  }

private:
  int listensockfd_; // 监听套接字
  std::string ip_;   // ip地址
  uint16_t port_;    // 端口号
};

Tcpclient.cc

#include <iostream>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>

void Usage(char *proc)
{
  std::cout << "\n\rUsage:" << proc << "severip serverport" << std::endl;
}

int main(int argc, char *argv[])
{
  if (argc != 3)
  {
    Usage(argv[0]);
    exit(1);
  }

  // 获取服务器的ip和port
  std::string serverip = argv[1];
  uint16_t serverport = std::stoi(argv[2]);

  // 填充结构体
  struct sockaddr_in server;
  memset(&server, 0, sizeof(server));

  server.sin_family = AF_INET;
  server.sin_port = htons(serverport);
  inet_pton(AF_INET, serverip.c_str(), &(server.sin_addr));

  while (true)
  {

    // 创建套接字
    int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0)
    {
      std::cerr << "socket error" << std::endl;
      return 1;
    }

    // 进行连接
    int n = connect(sockfd, (struct sockaddr *)&server, sizeof(server));
    if (n < 0) // 如果连接失败
    {
      std::cerr << "connect error...." << std::endl;
      return 2;
    }

    // 测试代码
    std::string message;

    std::cout << "Please Enter#";
    getline(std::cin, message);

    n = write(sockfd, message.c_str(), message.size());
    if (n < 0)
    {
      std::cerr << "write error....." << std::endl;
      continue;
    }

    char buffer[4096];
    n = read(sockfd, buffer, sizeof(buffer));
    if (n > 0)
    {
      buffer[n] = 0;
      std::cout << buffer << std::endl;
    }
    close(sockfd);
  }
  return 0;
}

Main.cc

#include"Tcpserver.hpp"
#include<memory>

void Usage(std::string proc)
{
  std::cout<<"\n\rUsage:"<<proc<<"port[1024+]\n"<<std::endl;
}

int main(int argc,char*argv[])
{
  if(argc!=2)//如果传参数不对,提醒一下
  {
    Usage(argv[0]);
    exit(UseageError);
  }
  uint16_t port=std::stoi(argv[1]);
  std::unique_ptr<TcpServer> tcp_svr(new TcpServer(port));
  tcp_svr->InitServer();
  tcp_svr->Start();

  return 0;
}

六.使用示例

在这里插入图片描述

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

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

相关文章

记一次某竞赛中的渗透测试(Windows Server 2003靶机漏洞)

靶机简介 Windows Server 2003是微软公司于2003年3月28日发布的服务器操作系统&#xff0c;它基于Windows XP/Windows NT 5.1进行开发&#xff0c;并在同年4月底上市。以下是关于Windows Server 2003的详细介绍&#xff1a; 系统名称与发布历程&#xff1a; 该产品最初被命名为…

Gaussian_Splatting 项目脚本指令

准备好一个稀疏重建的目录&#xff08;如Colmap的sparse文件&#xff09;&#xff0c;高斯泼溅需要稀疏重建的点云结果来作为输入&#xff0c;进行进一步训练和渲染。 可以参考&#xff1a;gaussian-splatting原理 有一点需要注意&#xff1a;Gaussian_Splatting 需要稀疏重建…

力扣之2621.睡眠函数

/*** param {number} millis* return {Promise}*/ async function sleep(millis) {return new Promise(resolve > setTimeout(resolve, millis)); }/** * let t Date.now()* sleep(100).then(() > console.log(Date.now() - t)) // 100*/ 这样的异步休眠功能在实际应用…

vue中使用html2canvas配合jspdf导出pdf(以及在导出时遇到的导出样式问题)

指定页面中导出为pdf格式并打包&#xff0c;使用html2canvas先转为图片格式&#xff0c;在利用jspdf转为pdf&#xff0c;最后下载打包为本地压缩包 yarn add html2canvas yarn add jspdf1. 注册一个插件并挂载 import html2Canvas from html2canvas import JsPDF from jspdf …

【零基础学习CAPL】——CAN报文的发送(配合Panel面板单次发送)

&#x1f64b;‍♂️【零基础学习CAPL】系列&#x1f481;‍♂️点击跳转 文章目录 1.概述2.面板创建2.1.新建一个Panel2.2. 在Panel中调出控件窗口2.3.控件添加与配置 3.系统变量创建4.系统变量与Panel绑定5.CAPL脚本实现6.效果 1.概述 使用场景&#xff0c;按下面板按钮同时…

人工智能与机器学习——开启智能时代的里程碑

写在前面 前言人工智能与机器学习的概述监督学习、无监督学习和强化学习的基本原理监督学习&#xff1a;无监督学习&#xff1a;强化学习&#xff1a; 机器学习的算法和方法常见的机器学习算法和方法线性回归&#xff1a;决策树&#xff1a;支持向量机&#xff1a;神经网络&…

个人多域名SSL证书推荐

SSL数字证书和通配符SSL证书、多域名通配符SSL证书一样&#xff0c;可以同时保护多个域名站点&#xff0c;但是它们之间还是存在一些区别。其中&#xff0c;最明显的区别就是它们的保护域名网站的类型和适用场景。今天就随SSL盾小编来了解多域名SSL证书。 1.多域名SSL证书可以…

【C/C++】深入理解--函数重载(什么是函数重载?为什么要有函数重载?)

目录 一、前言 二、 函数重载 &#x1f34e;什么是函数重载 &#x1f350;函数重载的条件 &#x1f347;函数重载的注意点 &#x1f349;为什么要有函数重载 &#x1f353;为何C语言不支持函数重载&#xff0c;反倒C可以&#xff1f; &#x1f4a6; Linux环境下演示函数重…

云纱网签约百望云,联手打造数字化产业闭环

近日&#xff0c;百望云签约广东云纱数字科技有限公司&#xff0c;共建数字化发票管理系统&#xff0c;赋能产业链上下游供应商的协同交易与运营&#xff0c;助力企业实现数字化四流合一交易&#xff0c;打造数字化产业闭环。 云纱网是广东云纱数字科技有限公司依托于深厚的产业…

Halcon 几何测量

文章目录 算子Halcon 计算两点之间的距离案例Halcon 计算点到直线的距离Halcon 计算点到区域的距离Halcon 线到区域的距离Halcon 线到线的距离 算子 distance_pp 两点之间的距离算子 distance_pp( : : Row1, Column1, Row2, Column2 : Distance) Row1 点1的行坐标 Column1 点1的…

Django视图函数技巧,从入门到实战

文章目录 Django视图函数1.request对象的方法2.视图函数的常用的返回对象&#xff08;1&#xff09;response对象&#xff08;2&#xff09;JsonResponse对象&#xff08;3&#xff09;redirect() &#xff1a;给浏览器了一个30x的状态码 3.设置响应头和状态码&#xff08;1&am…

【Vue】前端项目引入阿里图标

【Vue&React】前端项目引入阿里图标 方式11、登录自己的iconfont-阿里巴巴矢量图标库&#xff0c;把需要的图标加入到自己的项目中去&#xff1b;2、加入并进入到项目中去选择Font class 并下载到本地3、得到的文件夹如下4. 把红框中的部分粘贴到自己的项目中&#xff08; …

2万块的郎酒,都是我们惯的

文 | 琥珀酒研社 作者 | 五画 当我看到郎酒拿出快2万一瓶纪念酒的时候&#xff0c;我就知道&#xff0c;这场高价酒的喧嚣和吵闹&#xff0c;又到了一个新的高度。 和别的行业有所不同&#xff0c;白酒很少谈智商税&#xff0c;再高的价格&#xff0c;总有个冠冕堂皇的理由。…

28个炫酷的纯CSS特效动画示例(含源代码)

CSS是网页的三驾马车之一&#xff0c;是对页面布局的总管家&#xff0c;2024年了&#xff0c;这里列出28个超级炫酷的纯CSS动画示例&#xff0c;让您的网站更加炫目多彩。 文章目录 1. 涌动的弹簧效果2. 超逼真的3D篮球弹跳&#xff0c;含挤压弹起模态3. 鼠标放div上&#xff0…

力扣hot100 数组中的第K个最大元素 堆 三路划分

Problem: 215. 数组中的第K个最大元素 文章目录 思路复杂度Code 思路 &#x1f468;‍&#x1f3eb; 参考 复杂度 时间复杂度: O ( n ) O(n) O(n) 空间复杂度: O ( log ⁡ n ) O(\log{n}) O(logn) Code class Solution {public int findKthLargest(int[] nums, int k…

【Django自学】Django入门:如何使用django开发一个web项目(非常详细)

测试机器&#xff1a;windows11 x64 python版本&#xff1a;3.11 一、安装Django 安装步骤非常简单&#xff0c;使用pip安装就行 pip install django安装完成之后&#xff0c;python的 Scripts 文件夹下&#xff0c;会多一个 django-admin.exe (管理创建django项目的工具)。…

Linux其实不难,做个四层板设计的终端设备就搞懂了!

前言 想掌握一个技能&#xff0c;还得靠实践。 作者想学习Linux&#xff0c;于是边学边做&#xff0c;成功开源出了一个——还不错的 四层板设计的 终端设备。 项目成本可压缩至100元以内&#xff0c;便于复刻与参考学习&#xff01; 01 开源项目描述 做了一个基于V3S芯片的…

02-opencv简单实例效果和基本介绍-上

机器视觉概述 机器视觉是人工智能正在快速发展的一个分支。简单说来,机器视觉就是用机器代替人眼来做测量和判断。机器视觉系统是通过机器视觉产品(即图像摄取装置,分CMOS和CCD两种)将被摄取目标转换成图像信号,传送给专用的图像处理系统,得到被摄目标的形态信息,根据像素…

聚类(Clustering)理论

一、无监督学习介绍 在这小节中&#xff0c;我将开始介绍聚类算法&#xff0c;这是我们学习的第一个非监督学习算法&#xff0c;我们将要让计算机学习无标签数据而不是此前的标签数据。那么什么是非监督学习呢&#xff1f;在学习机器学习知识的开始我曾简单地介绍过非监督学习&…

【Algorithms 4】算法(第4版)学习笔记 02 - 1.4 算法分析

文章目录 前言参考目录学习笔记1&#xff1a;科学方法2&#xff1a;观察举例&#xff1a;三数之和3&#xff1a;近似4&#xff1a;增长数量级4.1&#xff1a;二分查找 demo4.2&#xff1a;二分查找代码实现4.3&#xff1a;二分查找比较次数的证明&#xff08;比较次数最多为lgN…