<Linux进程通信之共享内存>——《Linux》

news2025/1/11 10:58:16

目录

一、system V共享机制

1.共享内存示意图

2.共享内存数据结构

3.共享内存函数

3.1shmget函数

3.2 shmat函数

3.3 shmdt函数

3.4 shmctl函数

3.5 实例代码:

3.6 结果演示:

4. 创建共享内存

5. 基于共享内存与管道进行访问控制的共享内存读写

6. system V消息队列 

7. system V信号量 

8. 进程互斥

后记:●由于作者水平有限,文章难免存在谬误之处,敬请读者斧正,俚语成篇,恳望指教!

                                                                           ——By 作者:新晓·故知


一、system V共享机制

共享内存区是最快的 IPC 形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据。

1.共享内存示意图

(1)创建共享内存——删除共享内存      ------>OS内部实现

(2)关联共享内存——去关联共享内存   ------->进程实现

2.共享内存数据结构

Linux内核源码:

struct shmid_ds
{
    struct ipc_perm shm_perm;    /* operation perms */
    int shm_segsz;               /* size of segment (bytes) */
    __kernel_time_t shm_atime;   /* last attach time */
    __kernel_time_t shm_dtime;   /* last detach time */
    __kernel_time_t shm_ctime;   /* last change time */
    __kernel_ipc_pid_t shm_cpid; /* pid of creator */
    __kernel_ipc_pid_t shm_lpid; /* pid of last operator */
    unsigned short shm_nattch;   /* no. of current attaches */
    unsigned short shm_unused;   /* compatibility */
    void *shm_unused2;           /* ditto - used by DIPC */
    void *shm_unused3;           /* unused */
};

3.共享内存函数

3.1shmget函数

功能:用来创建共享内存
原型:int shmget(key_t key, size_t size, int shmflg);
参数:
key:这个共享内存段名字
size:共享内存大小
shmflg:由九个权限标志构成,它们的用法和创建文件时使用的mode模式标志是一样的
返回值:成功返回一个非负整数,即该共享内存段的标识码;失败返回-1

 3.2 shmat函数

功能:将共享内存段连接到进程地址空间

原型:
void *shmat(int shmid, const void *shmaddr, int shmflg);
参数:
shmid: 共享内存标识
shmaddr:指定连接的地址
shmflg:它的两个可能取值是SHM_RND和SHM_RDONLY
返回值:成功返回一个指针,指向共享内存第一个节;失败返回-1
说明:
shmaddr为NULL,核心自动选择一个地址
shmaddr不为NULL且shmflg无SHM_RND标记,则以shmaddr为连接地址。
shmaddr不为NULL且shmflg设置了SHM_RND标记,则连接的地址会自动向下调整为SHMLBA的整数倍。公式:
shmaddr - (shmaddr % SHMLBA)
shmflg=SHM_RDONLY,表示连接操作用来只读共享内存

3.3 shmdt函数

功能:将共享内存段与当前进程脱离
原型:
int shmdt(const void *shmaddr);
参数:
shmaddr: 由shmat所返回的指针
返回值: 成功返回0;失败返回-1
注意:将共享内存段与当前进程脱离不等于删除共享内存段

3.4 shmctl函数

功能:用于控制共享内存
原型:
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
参数:
shmid:由shmget返回的共享内存标识码
cmd:将要采取的动作(有三个可取值)
buf:指向一个保存着共享内存的模式状态和访问权限的数据结构
返回值:成功返回0;失败返回-1

3.5 实例代码:

测试代码结构
makefile:
.PHONY:all
all:server client
client:client.c comm.c
 gcc -o $@ $^
server:server.c comm.c
 gcc -o $@ $^
.PHONY:clean
clean:
 rm -f client server

comm.h:

#ifndef _COMM_H_
#define _COMM_H_
# include <stdio.h>
# include <sys/types.h>
# include <sys/ipc.h>
# include <sys/shm.h>
# define PATHNAME "."
# define PROJ_ID 0x6666
int createShm(int size);
int destroyShm(int shmid);
int getShm(int size);
# endif

comm.c:

#include "comm.h"

static int commShm(int size, int flags)
{
    key_t key = ftok(PATHNAME, PROJ_ID);
    if (key < 0)
    {
        perror("ftok");
        return -1;
    }
    int shmid = 0;
    if ((shmid = shmget(_key, size, flags)) < 0)
    {
        perror("shmget");
        return -2;
    }
    return shmid;
}
int destroyShm(int shmid)
{
    if (shmctl(shmid, IPC_RMID, NULL) < 0)
    {
        perror("shmctl");
        return -1;
    }
    return 0;
}
int createShm(int size)
{
    return commShm(size, IPC_CREAT | IPC_EXCL | 0666);
}
int getShm(int size)
{
    return commShm(size, IPC_CREAT);
}

server.c:

#include "comm.h"
int main()
{
    int shmid = createShm(4096);
    char *addr = shmat(shmid, NULL, 0);
    sleep(2);
    int i = 0;
    while (i++ < 26)
    {
        printf("client# %s\n", addr);
        sleep(1);
    }
    shmdt(addr);
    sleep(2);
    destroyShm(shmid);
    return 0;
}

client.c:

#include "comm.h"
int main()
{
    int shmid = getShm(4096);
    sleep(1);
    char *addr = shmat(shmid, NULL, 0);
    sleep(2);
    int i = 0;
    while (i < 26)
    {
        addr[i] = 'A' + i;
        i++;
        addr[i] = 0;
        sleep(1);
    }
    shmdt(addr);
    sleep(2);
    return 0;
}

3.6 结果演示:

 ctrl+c终止进程,再次重启

# ./server
shmget: File exists
# ipcs -m
------ Shared Memory Segments --------
key          shmid    owner   perms  bytes  nattch status
0x66026a25   688145   root    666    4096   0
       
# ipcrm -m 688145 #删除shm ipc资源,注意,不是必须通过手动来删除,这里只为演示相关指令,删除IPC资源是进程该做的事情
注意:共享内存没有进行同步与互斥!

4. 创建共享内存

 

 获取同一key值:

 现象:

  • 当我们运行完毕创建全新的共享内存的代码后(进程退出),但是之后在第二次、第三次、第n次的时候,再去申请就显示file存在。

这是因为systemV下的共享内存,它得生命周期是跟随内核的!

  • 释放共享内存的解决方法:

(1)显式删除

(2)重启kernel(OS)

  • 如何知道有哪些是IPC资源?

        通过指令查看:

        ipcs -m:查看shm list

        ipcs -q

        ipcs -s等

  • 如何显式删除?

      (1) ipcrm -m (shmid号)

         例如:ipcrm -m 5

     (2)通过系统接口

shell脚本:

while :; do ipcs -m; sleep 1;done

 

 

共享内存实际上是映射到了进程地址空间的用户区(堆、栈之间)!对于每一个进程而言,挂接到自己的上下文中的共享内存,属于自己的空间,类似于堆空间或者栈空间,可以被用户直接使用。

IpcShmCli.cc:

 

#include "Comm.hpp"
#include "Log.hpp"
using namespace std;

// 充当使用内存的角色
int main()
{

    //1.创建相同的key值
    key_t key = CreateKey();
    //cout << "key(hex): 0x" << hex << key << endl;
    Log() << "key(hex): 0x" << hex << key << endl;

    //2.获取共享内存
    int shmid = shmget(key,MEM_SIZE,IPC_CREAT);
    if(shmid < 0)
    {
        Log() << "shmget: " << strerror(errno) << "\n";
        return 2;
    }

    //3.挂接(关联)
    char *str = (char*)shmat(shmid,nullptr,0);
    
    //4.使用
    // sleep(6);  //仅用于观察
    
    //测试1
    int cnt = 0;      //未使用任何系统调用接口,这是因为共享内存实际上是映射到了进程地址空间的用户区(堆、栈之间)
    while(cnt <= 26)           
    {
        str[cnt] = 'A' +cnt;
        ++cnt;
        str[cnt] = '\0';
        sleep(1);
    }

    //5.去关联
    shmdt(str);
    return 0;
}

IpcShmSer.cc:

 

#include "Comm.hpp"
#include "Log.hpp"
using namespace std;

// 充当创建内存的角色

// 创建全新的共享内存
const int flags = IPC_CREAT | IPC_EXCL;
int main()
{
    key_t key = CreateKey();
    Log() << "key(hex): 0x" << hex << key << endl;
    Log() << "Begin create share memory!\n";
    int shmid = shmget(key, MEM_SIZE, flags | 0666); //"| 0666"权限控制
    if (shmid < 0)
    {
        Log() << "shmget: " << strerror(errno) << "\n";
        return 2;
    }
    //sleep(3);
    Log() << "Create shm success! shmid: " << shmid << "\n";
    // 使用
    // 1.将共享内存和自己的进程产生关联attach
    char *str = (char *)shmat(shmid, nullptr, 0);
    Log() << "Attach shm: " << shmid << " success!\n";
    //sleep(3);

    //测试共享内存通信
    while(true)
    {
        printf(".%s\n",str);
        sleep(1);
    }

    // 2.去关联
    shmdt(str);
    Log() << "Dttach shm: " << shmid << " success!\n";
    //sleep(3);

    // 删除
    shmctl(shmid, IPC_RMID, nullptr);
    Log() << "Delete shm,shmid: " << shmid << " success!\n";

    // sleep(3);
    return 0;
}

Comm.hpp:

 

#pragma once

#include <iostream>
#include <cstring>
#include <cstdlib>
#include <cstdio>
#include <cerrno>
#include <cassert>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include "Log.hpp"

#define PATH_NAME "/home/study/lesson/lesson26"
#define PROJ_ID 0x14
#define MEM_SIZE 4096 // 单位是字节

key_t CreateKey()
{
    key_t key = ftok(PATH_NAME, PROJ_ID);
    if (key < 0)
    {
        std::cerr << "ftok:" << strerror(errno) << std::endl;
        exit(1);
    }
    return key;
}

Log.hpp:

 

#pragma once

#include <iostream>
#include <ctime>

std::ostream &Log()
{
    std::cout << "For Debug | "
              << "timestamp: " << (uint64_t)time(nullptr) << " | ";
    return std::cout;
}

makefile:

 

.PHONY:all
all:IpcShmCli IpcShmSer

IpcShmCli:IpcShmCli.cc
	g++ -o $@ $^ -std=c++11
IpcShmSer:IpcShmSer.cc
	g++ -o $@ $^ -std=c++11

.PHONY:clean
clean:
	rm -f IpcShmCli IpcShmSer

共享内存,因为其自身的特性,使得它没有访问控制!
共享内存被双方直接看到,属于双方的用户空间,可以直接通信,但是不安全!
共享内存,是所有进程间通信,速度最快的!

 

5. 基于共享内存与管道进行访问控制的共享内存读写

将命名管道和共享内存创建好,把缓冲区数据写好,再通知读取。建立命名管道进行通信控制,管道具有同步访问策略,借此控制共享内存的访问。

IpcShmCli.cc:

#include "Comm.hpp"
#include "Log.hpp"
using namespace std;

// 充当使用内存的角色
int main()
{
    //打开以管道通信方式文件
    int fd = Open(FIFO_FILE,WRITER);

    //1.创建相同的key值
    key_t key = CreateKey();
    //cout << "key(hex): 0x" << hex << key << endl;
    Log() << "key(hex): 0x" << hex << key << endl;

    //2.获取共享内存
    int shmid = shmget(key,MEM_SIZE,IPC_CREAT);
    if(shmid < 0)
    {
        Log() << "shmget: " << strerror(errno) << "\n";
        return 2;
    }

    //3.挂接(关联)
    char *str = (char*)shmat(shmid,nullptr,0);
    
    //4.使用
    // sleep(6);  //仅用于观察
    
    // //测试1
    // int cnt = 0;      //未使用任何系统调用接口,这是因为共享内存实际上是映射到了进程地址空间的用户区(堆、栈之间)
    // while(cnt <= 26)           
    // {
    //     str[cnt] = 'A' +cnt;
    //     ++cnt;
    //     str[cnt] = '\0';
    //     sleep(1);
    // }

    // //测试2
    // while(true)
    // {
    //     cout<<"Please Enter# ";
    //     fflush(stdout);
    //     ssize_t s= read(0,str,MEM_SIZE);
    //     if(s > 0)
    //     {
    //         str[s] = '\0';
    //     }
    // }

    //测试管道通信
    while(true)
    {
        printf("Please Enter# ");
        fflush(stdout);
        ssize_t s = read(0, str, MEM_SIZE);
        if(s > 0)
        {
            str[s] = '\0';
        }
        Signal(fd);
    }

    //5.去关联
    shmdt(str);

    return 0;
}

 IpcShmSer.cc:

#include "Comm.hpp"
#include "Log.hpp"
using namespace std;

// 充当创建内存的角色

// 创建全新的共享内存
const int flags = IPC_CREAT | IPC_EXCL;
int main()
{
    // 测试管道通信
    CreateFifo();
    int fd = Open(FIFO_FILE, READER);
    assert(fd >= 0);

    key_t key = CreateKey();
    Log() << "key(hex): 0x" << hex << key << endl;
    Log() << "Begin create share memory!\n";
    int shmid = shmget(key, MEM_SIZE, flags | 0666); //"| 0666"权限控制
    if (shmid < 0)
    {
        Log() << "shmget: " << strerror(errno) << "\n";
        return 2;
    }
    //sleep(3);
    Log() << "Create shm success! shmid: " << shmid << "\n";
    // 使用
    // 1.将共享内存和自己的进程产生关联attach
    char *str = (char *)shmat(shmid, nullptr, 0);
    Log() << "Attach shm: " << shmid << " success!\n";
    //sleep(3);

    // //测试共享内存通信
    // while(true)
    // {
    //     printf(".%s\n",str);
    //     sleep(3);
    // }

    // 测试管道通信
   while(true)
    {
        // 让读端进行等待
        if(Wait(fd) <= 0) break; 
        printf(".%s\n", str);
        sleep(1);
    }
    // 2.去关联
    shmdt(str);
    Log() << "Dttach shm: " << shmid << " success!\n";
    //sleep(3);

    // 删除
    shmctl(shmid, IPC_RMID, nullptr);
    Log() << "Delete shm,shmid: " << shmid << " success!\n";

    Close(fd, FIFO_FILE);
    // sleep(3);
    return 0;
}

Comm.hpp:

#pragma once

#include <iostream>
#include <cstring>
#include <cstdlib>
#include <cstdio>
#include <cerrno>
#include <cassert>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include "Log.hpp"

#define PATH_NAME "/home/study/lesson/lesson26"
#define PROJ_ID 0x14
#define MEM_SIZE 4096 // 单位是字节
#define FIFO_FILE ".fifo"  //建立命名管道进行通信控制,管道具有同步访问策略,借此控制共享内存的访问



key_t CreateKey()
{
    key_t key = ftok(PATH_NAME, PROJ_ID);
    if (key < 0)
    {
        std::cerr << "ftok:" << strerror(errno) << std::endl;
        exit(1);
    }
    return key;
}

//下面测试通过管道进行通信
void CreateFifo()
{
    umask(0);
    if(mkfifo(FIFO_FILE,0666) < 0)
    {
        Log() << strerror(errno) << "\n";
        exit(2);
    }
}
//打开文件为了读、写
#define READER O_RDONLY
#define WRITER O_WRONLY
int Open(const std::string &filename, int flags)
{
    return open(filename.c_str(), flags);
}

int Wait(int fd)
{
    uint32_t values = 0;
    ssize_t s = read(fd, &values, sizeof(values));
     //assert(s == sizeof(values));
    //(void)s;  //消除告警
    return s;   //让另一方去判断警告
}

int Signal(int fd)
{
    uint32_t cmd = 1;
    write(fd, &cmd, sizeof(cmd));
}

int Close(int fd, const std::string filename)
{
    close(fd);
    unlink(filename.c_str());
}

Log.hpp:

#pragma once

#include <iostream>
#include <ctime>

std::ostream &Log()
{
    std::cout << "For Debug | "
              << "timestamp: " << (uint64_t)time(nullptr) << " | ";
    return std::cout;
}

makefile:

.PHONY:all
all:IpcShmCli IpcShmSer

IpcShmCli:IpcShmCli.cc
	g++ -o $@ $^ -std=c++11
IpcShmSer:IpcShmSer.cc
	g++ -o $@ $^ -std=c++11

.PHONY:clean
clean:
	rm -f IpcShmCli IpcShmSer

共享内存不作访问控制,可以通过信号量进行对资源保护!

6. system V消息队列 

消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法
每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值
特性方面
IPC 资源必须删除,否则不会自动清除,除非重启,所以 system V IPC 资源的生命周期随内核

 

 

 

 

7. system V信号量 

信号量主要用于同步和互斥的,下面先来看看什么是同步和互斥。

 

 

 

 以上的生命周期全部是跟随OS内核,有各自的系统调用接口。

总结:

 

8. 进程互斥

由于各进程要求共享资源,而且有些资源需要互斥使用,因此各进程间竞争使用这些资源,进程的这种关系为进程的互斥
系统中某些资源一次只允许一个进程使用,称这样的资源为临界资源或互斥资源。
在进程中涉及到互斥资源的程序段叫临界区
特性方面
*IPC 资源必须删除,否则不会自动清除,除非重启,所以 system V IPC 资源的生命周期随内核
扩展:
mmap

 

查看系统接口函数我们发现:每一个函数都包含有struct ipc_perm{ }结构。

每一种IPC资源描述结构体第一个都是一样的。

 

以上就是Linux内核对IPC资源的管理机制。不断地继承,用父类指针指向子类,需要使用哪种就强转获取即可(类似于C++中的多态或C++切片) 。这种方式的进程间通信使用不太多。设计思想很好,但实现和现代主流先进的操作系统编码不匹配。

后记:
●由于作者水平有限,文章难免存在谬误之处,敬请读者斧正,俚语成篇,恳望指教!

                                                                           ——By 作者:新晓·故知

 

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

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

相关文章

(P4)Python plt显示图像

Python plt显示图片 本地可以显示&#xff0c;服务器操作不太行。 plt.plot(x,y) plt.imshow(image) plt.show() # 显示图像&#xff08;曲线/图片都需要这一行&#xff09; plt.savefig(xx.png) # 保存图片 fig.savefig(xx.png) 功能相同 # 保存图片在本地执行失败&#xff…

Elasticsearch学习笔记

Elasticsearch学习笔记 ----第1章 Elasticsearch概述-01-开篇02-技术选型03-教学大纲-第2章 Elasticsearch入门-04-入门-环境准备05-入门-RESTful & JSON06-入门-Postman客户端工具07-入门-倒排索引08-入门-HTTP-索引-创建09-入门-HTTP-索引-查询 & 删除10-入门-HTTP-…

Linux 调试之 TRACE_EVENT(一)

文章目录前言一、TRACE_EVENT简介二、TRACE_EVENT() 结构2.1 TRACE_EVENT简介2.2 trace_sched_switch示例2.2.1 Name2.2.2 Prototype2.2.3 Arguments2.2.4 Structure2.2.5 Assignment2.2.6 Print2.3 Format file2.3.1 tracing/events/2.3.2 perf2.3.3 bpftrace2.3.4 bcc2.3.5 S…

逻辑回归 鸢尾花分类预测

目录 一&#xff1a;加载数据 二&#xff1a;数据集划分 三&#xff1a;选择算法 四&#xff1a;网格模型 超参数最优解 五&#xff1a;鸢尾花分类预测 六&#xff1a;预测与实际比对 七&#xff1a;完整源码分享 一&#xff1a;加载数据 from sklearn.datasets import…

Jenkins启动项目时报错问题

问题 在工作中使用jenkins进行项目部署启动时遇到问题&#xff0c;项目构建成功但是发布时一直显示时间增加&#xff0c;但是项目本身并没有问题&#xff0c;使用传统方法部署项目也能正常启动。报错如下图所示&#xff1a; 解决方法 是由于jenkins文件中打印日志的问题&am…

PayPal,Stripe,Square轮询支付系统

轮询展示 展示我们轮询的页面 轮询套餐 根据不同的用户和需求&#xff0c;可以选择不同的套餐 普通版 1500元 1年 1个用户 支持Paypal/Stripe 不限制A站个数 不限制B站个数 不限制提交模式 订单管理 物流管理 风控管理 必要的网站数据处理 24小时远程协助 开始…

springcloud服务消费与熔断

今天与大家分享服务消费与熔断&#xff0c;就是说当我们服务消费者转发到服务生产者时&#xff0c;如果有那一步出现了问题或者error&#xff0c;可以进行服务熔断&#xff08;服务降级&#xff09;&#xff0c;为了补救系统问题&#xff0c;不让用户使用时看见error报错信息&a…

《Java》String、StringBuffer、StringBuilder有什么区别?

目录 String StringBuffer StringBuilder 总结 ps&#xff1a;昨天在讨论完String的不可变性之后突然想要做一份总结笔记&#xff0c;总结一下String、StringBuffer、StringBuilder的区别 String String是Java中的基础类&#xff0c;提供了各种构造和对字符串的基本操作&am…

Cocos Creater(3.6.1)开发笔记——Typscript

文章目录项目入门关于cocos creator 3.x关于TypeScript新建项目VS setting json配置屏蔽项事件节点添加脚本&#xff08;事件&#xff09;案例代码素材使用技巧素材组合固定布局&#xff08;类似css&#xff09;项目入门 关于cocos creator 3.x 相当于cocos所有版本的功能的综…

初探MapReduce

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录MapReduce核心思想MapReduce编程模型MapReduce编程实例——词频统计思路1、Map阶段&#xff08;映射阶段&#xff09;2、Reduce阶段&#xff08;归并阶段&#xff09…

VS2019下C#调用C++ DLL详解+数据转换

VS2019下C#调用C DLL详解数据转换 -C#调用OpenCV&#xff08;c的.dll主要有两种常见的方式&#xff1a;托管和非托管两种形式&#xff01; 非托管的形式即是采用[DllImport]的形式&#xff0c;这种形式只能调用的C的函数&#xff0c;适合用于简单的图形处理调用&#xff0c;这也…

正交编码器溢出处理

文章目录1.正交编码器1.1 参数特性1.2 应用范围2.正交编码器使用2.1 溢出问题2.2 中断模式2.3 循环模式延伸1.正交编码器 正交编码器一般指的是增量式光栅&#xff08;磁栅&#xff09;编码器&#xff0c;通常有三路输出信号&#xff0c;A相、B相、Z相&#xff0c;俗称ABZ编码器…

【2022年度系列工作总结】「国内软件质量调查问卷」针对于本年度软件质量分析总结报告

前提背景 针对于目前的软件行业而言&#xff0c;软件的质量目前越来越被大家所在关注&#xff0c;慢慢的QA以及SQA的角色也变得愈加重要。接下来我就针对于我司&#xff08;XXX&#xff09;的相关的实际开发情况对应的【2022年国内软件质量调查问卷】&#xff0c;为大家梳理和…

[附源码]计算机毕业设计Python大学生考勤管理系统论文(程序+源码+LW文档)

该项目含有源码、文档、程序、数据库、配套开发软件、软件安装教程 项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等…

听说Linux基础指令很多?这里都帮你总结好了

✨个人主页&#xff1a; Yohifo &#x1f389;所属专栏&#xff1a; Linux学习之旅 &#x1f38a;每篇一句&#xff1a; 图片来源 &#x1f383;操作环境&#xff1a; CentOS 7.6 阿里云远程服务器 You can’t use up creativity. The more you use, the more you have. 创造力…

微软数据科学家助理(Data Scientist Associate)认证考试通过经验分享(DP-100)

今天冒着大疫情&#xff0c;去海淀的test center参加考试&#xff0c;通过了微软DP-100 在 Azure 上设计和实现数据科学解决方案&#xff0c;并且获得了经 Microsoft 认证&#xff1a;Azure 数据科学家助理 的证书。 经 Microsoft 认证&#xff1a;Azure 数据科学家助理 考试结…

Hadoop编译源码

文章目录第一章 Hadoop编译源码1.1 前期准备工作1.2 Jar包安装配置maven的环境变量在 mirrors节点中添加阿里云镜像安装gcc make配置环境变量1.3编译源码第二章 常见错误及解决方案第一章 Hadoop编译源码 1.1 前期准备工作 1&#xff09;CentOS联网 配置CentOS能连接外网。Li…

力扣(LeetCode)1703. 得到连续 K 个 1 的最少相邻交换次数(C++)

贪心 将至少连续 KKK 个 111 放在一起。首先考虑他们是相邻着放在一起的&#xff0c;然后考虑性质 : 设相邻摆放后&#xff0c;起始 111 的位置是 mid{mid}mid &#xff0c;对于每个 111 的位置 aia_iai​ &#xff0c;它需要被摆放的位置是 amidia_{mid}iamid​i 。考虑一个等…

【Effective_Objective-C_2对象,消息,运行期2】

文章目录前言12.理解消息转发机制消息转发动态方法解析动态方法解析的前提备援接受者完整的消息转发消息转发全部流程要点总结13.用“方法调配技术”调试“黑盒方法”方法调配动态消息派发系统和IMP如何交换方法实现要点总结14.理解“类对象”的用意在类的继承体系中查询类型信…

【经济学】【综合篇】经济机器是怎样运行的

原视频&#xff1a;经济机器是怎样运行的 (时长30分钟) Ray Dalio 前言&#xff1a;经济与我们每一个人息息相关&#xff0c;经济社会的一些变革或举措也会直接或间接的反映到我们每个个体身上。了解经济&#xff0c;提高自己的认知&#xff0c;可以帮助我们更好的参与经济活动…