正点原子嵌入式linux驱动开发——Linux ADC驱动
在之前的笔记中,学习了如何给ICM20608编写IIO驱动,ICM20608本质就是ADC,因此纯粹的ADC驱动也是IIO驱动框架的。本章就学习一下如何使用STM32MP1内部的ADC,并且在学习巩固一下IIO驱动。
ADC简介
ADC
ADC,Analog to Digital Converter的缩写,中文名称模数转换器。它可以将外部的模拟信号转化成数字信号。对于GPIO口来说高于某个电压值,它读出来的只有高电平,低于就是低电平。假如想知道具体的电压数值就要借助于ADC的帮助,它可以将一个范围内的电压精确的读取出来。
ADC有几个比较重要的参数:
- 测量范围:测量范围对于ADC来说就好比尺子的量程,ADC测量范围决定了外接的设备其信号输出电压范围,不能超过ADC的测量范围。如果所使用的外部传感器输出的电压信号范围和所使用的ADC测量范围不符合,那么就需要自行设计相关电压转换电路。
- 分辨率:就是尺子上的能量出来的最小测量刻度,假如ADC的测量范围为0-5V,分辨率设置为12位,那么能测出来的最小电压就是 5V除以2的12次方,也就是5/4096=0.00122V。很明显,分辨率越高,采集到的信号越精确,所以分辨率是衡量 ADC 的一个重要指标。
- 精度:是影响结果准确度的因素之一,比如在厘米尺上能测量出大概多少毫米的尺度但是毫米后一点点我们却不能准确的量出。经过计算ADC在12位分辨率下的最小测量值是0.00122V,但是ADC的精度最高只能到11位也就是0.00244V。也就是ADC测量出0.00244V的结果是要比0.00122V要可靠,也更准确。
- 采样时间:当ADC在某时刻采集外部电压信号的时候,此时外部的信号应该保持不变,但实际上外部的信号是不停变化的。所以在ADC内部有一个保持电路,保持某一时刻的外部信号,这样ADC就可以稳定采集了,保持这个信号的时间就是采样时间。
- 采样率:也就是在一秒的时间内采集多少次。很明显,采样率越高越好,当采样率不够的时候可能会丢失部分信息,所以ADC采样率是衡量ADC性能的另一个重要指标。
总之,只要是需要模拟信号转为数字信号的场合,那么肯定要用到ADC。很多数字传感器内部会集成ADC,传感器内部使用ADC来处理原始的模拟信号,最终给用户输出数字信号。
STM32MP157 ADC简介
STM32MP157有两个ADC:ADC1和ADC2,ADC1和ADC2紧密耦合,可在双重模式下运行(ADC1为主器件)。每个ADC由一个16位逐次逼近模数转换器组成,每个ADC有20个通道,每个通道支持单次、连续、扫描或不连续采样模式。转换结果存储在一个左对齐或右对
齐的32位数据寄存器中。ADC主要特性如下:
- 多达2个ADC,可在双重模式下运行。
- 可以配置为16、14、12、10或8位分辨率。
- 自校准。
- 可独立配置各通道采样时间。
- ……
ADC驱动源码简析
设备树下ADC节点
STM32MP157有2个ADC,因此对应2个ADC控制器,所以在设备树里就有2个ADC控制器节点。这2个ADC的设备树节点内容都是一样的,除了reg属性不同(毕竟不同的控制器,其地址范围不同)。本章实验使用PA5这个引脚来完成ADC实验,而PA5就是ADC1_INP19通道引脚,所以这里就以ADC1为例进行讲解,stm32mp151.dtsi文件中的adc节点信息如下:
示例代码 57.2.1.1 adc 节点内容
1 adc: adc@48003000 {
2 compatible = "st,stm32mp1-adc-core";
3 reg = <0x48003000 0x400>;
4 interrupts = <GIC_SPI 18 IRQ_TYPE_LEVEL_HIGH>,
5 <GIC_SPI 90 IRQ_TYPE_LEVEL_HIGH>;
6 clocks = <&rcc ADC12>, <&rcc ADC12_K>;
7 clock-names = "bus", "adc";
8 interrupt-controller;
9 st,syscfg = <&syscfg>;
10 #interrupt-cells = <1>;
11 #address-cells = <1>;
12 #size-cells = <0>;
13 status = "disabled";
14
15 adc1: adc@0 {
16 compatible = "st,stm32mp1-adc";
17 #io-channel-cells = <1>;
18 reg = <0x0>;
19 interrupt-parent = <&adc>;
20 interrupts = <0>;
21 dmas = <&dmamux1 9 0x400 0x80000001>;
22 dma-names = "rx";
23 status = "disabled";
24 };
25
26 adc2: adc@100 {
27 compatible = "st,stm32mp1-adc";
28 #io-channel-cells = <1>;
29 reg = <0x100>;
30 interrupt-parent = <&adc>;
31 interrupts = <1>;
32 dmas = <&dmamux1 10 0x400 0x80000001>;
33 dma-names = "rx";
34 status = "disabled";
35 };
36 };
第2行,compatible属性值为“st,stm32mp1-adc-core”,所以在整个Linux源码里面搜索这个字符串即可找到STM32MP157的ADC驱动核心文件,这个文件就是drivers/iio/adc/stm32-adc-core.c。
第16、27行,compatible属性值“st,stm32mp1-adc”,搜索这个字符串,可以找到ADC驱动文件,这个文件就是drivers/iio/adc/stm32-adc.c。
关于STM32MP157的ADC节点更为详细的信息请参考对应的绑定文档:Documentation/devicetree/bindings/iio/adc/st,stm32-adc.txt。接下来简单分析一下绑定文档,后面需要根据绑定文档修改设备树,使能ADC对应的通道。
ADC首先需要一个根节点,adc根节点属性如下:
1、必要属性
- compatible:兼容性属性,必须的,可以设置为“st,stm32mp1-adc-core”。
- reg:ADC控制器寄存器信息。
- interrupts:中断属性,ADC1和ADC2各对应一个中断信息。
- clocks:时钟属性。
- clock-names:时钟名字,可选“adc”或“bus”。
- interrupt-controller:中断控制器。
- vdda-supply:此属性对应vdda输入模拟电压句柄。
- vref-supply:此属性对应vref参考电压句柄。
- interrupt-cells:设置为1。
- address-cells:设置为1。
- size-cells:设置为0。
2、可选属性
- :pinctrl引脚配置信息。
- booster-supply:嵌入式booster调节器句柄。
- vdd-supply:vdd输入电压句柄。
- st,syscfg:系统配置控制器句柄。
- st,max-clk-rate-hz:最大时钟。
STM32MP157有两个ADC,每个ADC对应一个子节点,ADC子节点相关属性如下:
1、必要属性
- compatible:兼容性属性,必须的,可以设置为“st,stm32mp1-adc”。
- reg:不同ADC控制器寄存器地址偏移信息。
- interrupts:中断线信息,adc@0为0,adc@100为1。
- st,adc-channels:ADC通道信息,可以设置0-19,分别对应20个通道。
- st,adc-diff-channels:ADC差分通道信息,如果使用差分ADC功能的话。
- io-channel-cells:设置为1。
2、可选属性
- dmas:DMA 通道句柄。
- dma-names:dma 名字,必须设置成“rx”。
- assigned-resolution-bits:ADC分辨率,可以设置为8、10、12,、14或16。
- st,min-sample-time-nsecs:最小采样时间,单位ns。
ADC驱动源码分析
STM32MP157 ADC驱动文件有两个:stm32-adc-core.c和stm32-adc.c。stm32-adc-core.c是ADC核心层,主要用于ADC电源等初始化,需要重点关注的是 stm32-adc.c这个文件。stm32-adc.c主体框架是platform,配合IIO驱动框架实现ADC驱动。
stm32_adc结构体
ST自己将ADC外设抽象成了结构体,stm32_adc就相当于自定义的设备结构体。stm32_adc结构体贯穿于整个驱动文件,结构体内容如下:
stm32_adc_probe函数
接下来看一下stm32_adc_probe函数,内容如下(有省略):
示例代码 57.2.2.2 stm32_adc_probe 函数
1 static int stm32_adc_probe(struct platform_device *pdev)
2 {
3 struct iio_dev *indio_dev;
4 struct device *dev = &pdev->dev;
5 irqreturn_t (*handler)(int irq, void *p) = NULL;
6 struct stm32_adc *adc;
7 int ret;
8
9 if (!pdev->dev.of_node)
10 return -ENODEV;
11
12 indio_dev = devm_iio_device_alloc(&pdev->dev, sizeof(*adc));
13 if (!indio_dev)
14 return -ENOMEM;
15
16 adc = iio_priv(indio_dev);
17 adc->common = dev_get_drvdata(pdev->dev.parent);
18 spin_lock_init(&adc->lock);
19 init_completion(&adc->completion);
20 adc->cfg = (const struct stm32_adc_cfg *)
21 of_match_device(dev->driver->of_match_table, dev)->data;
22
23 indio_dev->name = dev_name(&pdev->dev);
24 indio_dev->dev.parent = &pdev->dev;
25 indio_dev->dev.of_node = pdev->dev.of_node;
26 indio_dev->info = &stm32_adc_iio_info;
27 indio_dev->modes = INDIO_DIRECT_MODE | INDIO_HARDWARE_TRIGGERED;
28
29 platform_set_drvdata(pdev, adc);
30
31 ret = of_property_read_u32(pdev->dev.of_node, "reg",
&adc->offset);
32 if (ret != 0) {
33 dev_err(&pdev->dev, "missing reg property\n");
34 return -EINVAL;
35 }
36
37 adc->irq = platform_get_irq(pdev, 0);
38 if (adc->irq < 0)
39 return adc->irq;
40
41 ret = devm_request_threaded_irq(&pdev->dev, adc->irq,
stm32_adc_isr,
42 stm32_adc_threaded_isr,
43 0, pdev->name, adc);
44 if (ret) {
45 dev_err(&pdev->dev, "failed to request IRQ\n");
46 return ret;
47 }
48
49 adc->clk = devm_clk_get(&pdev->dev, NULL);
50 if (IS_ERR(adc->clk)) {
51 ret = PTR_ERR(adc->clk);
52 if (ret == -ENOENT && !adc->cfg->clk_required) {
53 adc->clk = NULL;
54 } else {
55 dev_err(&pdev->dev, "Can't get clock\n");
56 return ret;
57 }
58 }
59
60 ret = stm32_adc_of_get_resolution(indio_dev);
61 if (ret < 0)
62 return ret;
63
64 ret = stm32_adc_chan_of_init(indio_dev);
65 if (ret < 0)
66 return ret;
67
68 ret = stm32_adc_dma_request(indio_dev);
69 if (ret < 0)
70 return ret;
71
72 if (!adc->dma_chan)
73 handler = &stm32_adc_trigger_handler;
74
75 ret = iio_triggered_buffer_setup(indio_dev,
76 &iio_pollfunc_store_time, handler,
77 &stm32_adc_buffer_setup_ops);
78 if (ret) {
79 dev_err(&pdev->dev, "buffer setup failed\n");
80 goto err_dma_disable;
81 }
82
83 /* Get stm32-adc-core PM online */
84 pm_runtime_get_noresume(dev);
85 pm_runtime_set_active(dev);
86 pm_runtime_set_autosuspend_delay(dev,
STM32_ADC_HW_STOP_DELAY_MS);
87 pm_runtime_use_autosuspend(dev);
88 pm_runtime_enable(dev);
89
90 ret = stm32_adc_hw_start(dev);
91 if (ret)
92 goto err_buffer_cleanup;
93
94 ret = iio_device_register(indio_dev);
95 if (ret) {
96 dev_err(&pdev->dev, "iio dev register failed\n");
97 goto err_hw_stop;
98 }
......
123 return ret;
124 }
第12行,调用devm_iio_device_alloc函数申请iio_dev,这里也连stm32_adc内存一起申请了。
第16行,调用iio_priv函数从iio_dev里面的到stm32_adc首地址。
第23-27行,初始化iio_dev,重点是第26行的stm32_adc_iio_info,因为用户空间读取ADC数据最终就是由stm32_adc_iio_info来完成的。
第37行,调用platform_get_irq获取中断号。
第41行,调用devm_request_threaded_irq函数申请中断,这里使用的是中断线程化。
第60行,调用stm32_adc_of_get_resolution函数获取ADC的分辨率。
第64行,调用stm32_adc_chan_of_init函数初始化ADC通道。
第68行,调用stm32_adc_dma_request函数初始化DMA。
第75行,调用iio_triggered_buffer_setup函数设置IIO触发缓冲区。
第90行,调用stm32_adc_hw_start函数开启ADC。
第94行,调用iio_device_register函数向内核注册iio_dev。
可以看出stm32_adc_probe函数核心就是初始化ADC,然后建立ADC的IIO驱动框架。
stm32_adc_iio_info结构体
stm32_adc_iio_info结构体内容如下所示:
重点来看一下第2行的stm32_adc_read_raw函数,因为此函数才是最终向用户空间发送ADC原始数据的,函数内容如下:
第9-18行,读取ADC原始数据值,第18行type值为IIO_VOLTAGE,也就是读取电压值。第14行调用stm32_adc_single_conv函数来完成ADC单次读取。stm32_adc_single_conv函数会设置采样率、配置通道、使用硬件触发、开启转换,最后等待转换完成中断发生。
第20-28行,返回ADC对应的分辨率。
第30-36行,返回差分ADC的偏移值。
stm32_adc_read_raw函数内容还是比较简单的,因为只是读取ADC原始值,不像ICM20608那么复杂。关于ADC驱动源码就讲解到这里,接下来学习如何使能ADC,然后编写应用程序读取ADC采集到的值。
硬件原理图分析
STM32MP157开发板ADC硬件原理图如下图所示:
上图中JP2是一个3P的排针,1脚连接到STM32MP157的DAC引脚上(PA4),2脚连
接到ADC引脚上(PA5),3脚连接到VR1这个可调电位器上。本章实验使用STM32MP157的ADC来采集VR1可调电位器的电压,因此要用跳线帽将JP1的2,3脚连接起来,如下图所示:
ADC驱动编写
修改设备树
ADC驱动ST已经编写好了,只需要修改设备树即可。首先在stm32mp15-pinctrl.dtsi文件中添加ADC使用的PA5引脚配置信息:
示例代码 57.4.1.1 PA5 引脚配置信息
1 adc1_in19_pins_a: adc1-in19 {
2 pins {
3 pinmux = <STM32_PINMUX('A', 5, ANALOG)>;
4 };
5 };
接下来在stm32mp157d-atk.dts文件中向根节点添加vdd子节点信息,内容如下:
示例代码 57.4.1.2 vdd 子节点
1 vdd: regulator-vdd {
2 compatible = "regulator-fixed";
3 regulator-name = "vdd";
4 regulator-min-microvolt = <3300000>;
5 regulator-max-microvolt = <3300000>;
6 regulator-always-on;
7 regulator-boot-on;
8 };
最后在stm32mp157d-atk.dts文件中向adc节点追加一些内容,内容如下:
示例代码 57.4.1.3 adc 节点
1 &adc {
2 pinctrl-names = "default";
3 pinctrl-0 = <&adc1_in19_pins_a>;
4 vdd-supply = <&vdd>;
5 vdda-supply = <&vdd>;
6 vref-supply = <&vdd>;
7 status = "okay";
8
9 adc1: adc@0 {
10 st,adc-channels = <19>;
11 st,min-sample-time-nsecs = <10000>;
12 assigned-resolution-bits = <16>;
13 status = "okay";
14 };
15 };
第3行,配置adc引脚。
第4-6行,设置电压属性。
第9-12行,adc1子节点,第10行st,adc-channels属性设置adc通道为19,第11行st,min-sample-time-nsecs属性设置最小采样时间为10000ns,第12行设置分辨率为16位。
使能ADC驱动
ST官方默认已经使能了ADC驱动,所以不需要修改,但是为了学习,看一下如何使能Linux内核自带的ADC驱动。打开Linux内核配置界面,配置路径如下:
-> Device Drivers -> Industrial I/O support (IIO [=y]) -> Analog to digital converters -> STMicroelectronics STM32 adc core (STM32_ADC_CORE [=y]) -> <*>STMicroelectronics STM32 adc //使能 STM32 ADC |
如下图所示:
编写测试APP
编译修改后的设备树,然后使用新的设备树启动系统。进入/sys/bus/iio/devices目录下,此目录下就有ADC对应的iio设备:iio:deviceX,本章例程如下图所示:
上图中的“iio:device0”就是ADC设备,因为此时并没有加载其他的IIO设备驱动,只有一个ADC。如果还加载了其他IIO设备驱动,那么就要依次进入iio设备目录,查看一下都对应的是什么设备。
进入“iio:device0”目录,内容如下图所示:
标准的IIO设备文件目录,只关心三个文件:
- in_voltage19_raw:ADC1通道19原始值文件。
- in_voltage_offset:ADC1偏移文件。
- in_voltage_scale:ADC1比例文件(分辨率),单位为mV。实际电压值(mV)=in_voltage19_raw*in_voltage_scale。
开发板此时in_voltage19_raw和in_voltage_scale这两个文件内容如下:
经过计算,上图中实际电压:20779*0.050354003≈1046.3mV,也就是1.0463V。
编写测试APP,其中需要:
新建char数组指针file_path存储iio框架对应文件路径,并enum对应的文件索引。然后欣姐设备结构体,里面存储raw、scale和act就可以了。
编写file_data_read,这个跟之前iio的很类似,就是fopen之后fscanf,到EOF处就fseek把指针调回文件头然后fclose。
之后编写adc_read,里面就是调用file_data_read读出来之后分别atoi和atof转成数字,最后把raw和scale乘起来/1000得到实际值存到adc_dev的结构体指针dev->act之中。
最后是main函数,argc就1个,在while中adc_read然后printf就好了。
运行测试
编译驱动程序和测试APP
由于不需要编写ADC驱动程序,因此也就不需要编译驱动程序。设备树前面已经编译过了,所以这里就只剩下编译测试APP。由于adcApp.c用到了浮点运算,因此编译的时候要使能硬件浮点,输入如下编译adcApp.c 这个测试程序:
arm-none-linux-gnueabihf-gcc -march=armv7-a -mfpu=neon -mfloat-abi=hard adcApp.c -o adcApp |
运行测试
注意,在测试之前一定要先按照之前的连接示意图所示,将JP2跳线帽接到左边,也就是将ADC1_CH19通道连接到开发板上的可调电位器上!
输入如下命令,使用adcApp测试程序:
./adcApp |
测试APP会不断的读取ADC值并输出到终端,可以通过调节开发板上的电位器来改变电压值,如下图所示:
从上图可以看到ADC原始值以及对应的电压值,因为STM32MP157的ADC可采集电压范围为0-3.3V,因此扭动开发板上的电位器的时候,电压会在0-3.3V之间变化。
总结
这一章的学习放在了IIO的驱动后面,所以其实比之前要简单了很多,个人感觉可以先看这个,学一下IIO的驱动框架,然后再去看上一篇笔记,基本的内容都一样,反而上一篇笔记的IIO驱动来搞ICM20608难度大很多。
相关文章:

正点原子嵌入式linux驱动开发——Linux ADC驱动
在之前的笔记中,学习了如何给ICM20608编写IIO驱动,ICM20608本质就是ADC,因此纯粹的ADC驱动也是IIO驱动框架的。本章就学习一下如何使用STM32MP1内部的ADC,并且在学习巩固一下IIO驱动。 ADC简介 ADC ADC,Analog to D…...

自动化测试介绍和分类,看这一篇就够了
📢专注于分享软件测试干货内容,欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!📢交流讨论:欢迎加入我们一起学习!📢资源分享:耗时200小时精选的「软件测试」资…...

Debian中执行脚本 提示没有那个文件或目录
原因是在脚本头有句: ~/.bash_profile这个在CentOS里执行是正常的,但在Debian中是没有的,它改成了: ~/.profile一、区别: 1、/etc/profile: 此文件为系统的每个用户设置环境信息,当用户第一次登录时,该文…...

放松鸭-技术支持
“放松鸭”利用苹果手表的HRV心率变异性和静息心率等数据进行分析,帮助您了解当前身体疲劳和心理压力程度,并及时提醒您的压力状态。我们的目标是让您更好地感知、管理和应对压力,让您的身心得到平静和放松。通过读取您的心脏数据,…...

Vue 报错error:0308010C:digital envelope routines::unsupported
你遇到的错误,error:0308010C:digital envelope routines::unsupported,与 OpenSSL 相关,表明在你的 Vue.js 应用中可能存在与加密操作相关的问题。这种错误通常出现在 OpenSSL 库存在不匹配或问题的情况下。 以下是解决此问题的一些建议&am…...

Android 9.0 隐藏设置中一级菜单“已连接的设备”
Android 9.0 隐藏设置中一级菜单“已连接的设备” 接到客户反馈需要隐藏设备设置中的“已连接的设备”一级菜单,具体修改参照如下: /vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/SettingsActivity.java somethin…...

Hive开窗函数根据特定条件取上一条最接近时间的数据(根据条件取窗口函数的值)
一、Hive开窗函数根据特定条件取上一条最接近时间的数据(单个开窗函数,实际取两个窗口) 针对于就诊业务,一次就诊,多个处方,处方结算时间可能不一致,然后会有多个AI助手推荐用药,会…...

指针与函数
指针函数:函数的返回值可以是整型值、浮点型值、字符型值等,在C语言中还允许一个函数的返回值是一个指针(地址),这种返回指针的函数称为指针函数。 指针函数语法格式: 基类型 * 函数名(参数列…...

GBase8a-GDCA-第二次阶段测试
文章目录 主要内容在这里插入图片描述 在这里插入图片描述 总结 主要内容 GBase8a-GDCA-第二次阶段测试及答案 总结 以上是今天要讲的内容,GBase8a-GDCA-第二次阶段测试…...

Go 理解零值
在 Go 语言中,零值(Zero Value)是指在声明变量但没有显式赋值的情况下,变量会被自动赋予一个默认值。这个默认值取决于变量的类型,不同类型的变量会有不同的零值。零值是 Go 语言中的一个重要概念,因为它确…...

SQL编写规范【干货】
编写本文档的目的是保证在开发过程中产出高效、格式统一、易阅读、易维护的SQL代码。 1 编写目 2 SQL书写规范 3 SQL编写原则 获取所有软件开发资料:点我获取...

2.5 Windows驱动开发:DRIVER_OBJECT对象结构
在Windows内核中,每个设备驱动程序都需要一个DRIVER_OBJECT对象,该对象由系统创建并传递给驱动程序的DriverEntry函数。驱动程序使用此对象来注册与设备对象和其他系统对象的交互,并在操作系统需要与驱动程序进行交互时使用此对象。DRIVER_OB…...

[ubuntu]ubuntu上安装jdk1.8教程
首先需要去官方网站去下载对应jdk1.8版本: https://www.oracle.com/java/technologies/downloads/ 您也可以去csdn搜索我提供jdk安装包 这里以jdk-8u201-linux-x64.tar.gz为例子,首先下载安装后解压 tar -zxvf jdk-8u201-linux-x64.tar.gz 比如我解…...

金蝶云星空其他出库单保存提示序列号不一致
文章目录 金蝶云星空其他出库单保存提示序列号不一致保存报错初步分析总结 金蝶云星空其他出库单保存提示序列号不一致 保存报错 显示单据数量0.序列号数量3 初步分析 输入实发数量没有触发序列号数量的计算 检查实发数量的值更新事件 实发数量和序列号数量的转换ÿ…...

FBI:皇家勒索软件要求350名受害者支付2.75亿美元
导语 最近,FBI和CISA联合发布的一份通告中透露,自2022年9月以来,皇家勒索软件(Royal ransomware)已经入侵了全球至少350家组织的网络。这次更新的通告还指出,这个勒索软件团伙的赎金要求已经超过了2.75亿美…...

Layout工程师们--Allegro X AI实现pcb自动布局布线
Cadence 推出 Allegro X AI,旨在加速 PCB 设计流程,可将周转时间缩短 10 倍以上 楷登电子(美国 Cadence 公司,NASDAQ:CDNS)今日宣布推出 Cadence Allegro X AI technology,这是 Cadence 新一代…...

Hive入门--学习笔记
1,Apache Hive概述 定义: Hive是由Facebook开源用于解决海量结构化日志的数据统计,它是基于大数据生态圈Hadoop的一个数据仓库工具。 作用: Hive可以用于将结构化的数据文件【映射】为一张表,并提供类SQL查询功能。 H…...

【nlp】1文本预处理总括目录(附各章节链接)
文本预处理 1. 文本预处理机器作用2. 文本预处理包含的主要环节2.1 文本处理的基本方法2.1.1 分词2.1.2 词性标注2.2.3 命名实体标注2.2 文本张量表示方法2.2.1 one-hot编码2.2.2 Word2vec2.2.3 Word Embedding2.3 文本语料的数据分析2.3.1 标签数量分布2.3.2 句子长度分布2.3.…...

《增长黑客》思维导图
增长黑客这个词源于硅谷,简单说,这是一群以数据驱动营销、以迭代验证策略,通过技术手段实现爆发式增长的新型人才。 近年来,互联网公司意识到这一角色可以发挥四两拨千斤的作用,因此对该职位的需求也如井喷式增长。本…...

oracle-buffer cache
段,区,块。 每当新建一个表,数据库会相应创建一个段。然后给这个段分配一个区。 一个区包含多个块。 区是oracle给段分配空间的最小单位。 块是oracle i\o的最小单位。 原则上,一个块包含多行数据。 dbf文件会被划分成一个一个…...

数据可视化—D3(Data Driven Documents)
链接 教程链接安装教程官方github仓库 基础知识 D3是一个Javascript库,用于在浏览器中创建可视化和可交互的各种图表。通过以下代码的对比,说明D3的使用场景以及使用效果(理论上,以下两段代码效果是一样的)…...

±15kV ESD 保护、3V-5.5V 供电、真 RS-232 收发器MS2232/MS2232T
产品简述 MS2232/MS2232T 芯片是集成电荷泵,具有 15kV ESD 保护的 RS-232 收发器,包括两路接收器、两路发送器。 芯片满足 TIA/EIA-232 标准,为异步通信控制器和串口连 接器提供通信接口。 芯片采用 3V-5.5V 供电,电荷泵仅用…...

企业版远程软件推荐
在当今的数字时代,为您的企业配备远程访问功能至关重要。通过远程访问,您的团队可以在办公室外工作,并且无论身在何处都可以保持相同的生产力水平。在本文中,我们汇总了市场上的四大选择。 我们在远程访问解决方案中寻找什么 远…...

独孤思维:没学会走就要跑,你只能一辈子是穷b
很多人眼高手低,没学会走就要跟别人比赛跑步; 很多人想要发财,没赚到钱就要喊着跟谁比有钱。 眼高手低,自命不凡,愚蠢至极。 上周团队要扩编,招一个运营。 来了一个00后女孩应聘。 上来就说自己目标三…...

鸿蒙LiteOs读源码教程+向LiteOS中添加一个系统调用
本文分为2个部分:第1部分简要介绍如何读鸿蒙Liteos源码,第2部分是实验向LiteOS中添加一个系统调用的完整过程。 前置资料: imx6ull开发板使用方式详解 源码下载 编译运行简单程序 Ubuntu虚拟机使用鸿蒙LiteOs操作系统常见错误汇总 一、鸿…...

美国站群服务器IP如何设置分配?
在配置美国站群服务器时,IP的分配是一个重要的步骤。下面将介绍一些关于美国站群服务器IP分配的相关知识。 独享IP和虚拟IP 在租用美国站群服务器之前,我们需要了解提供的IP是独享的还是虚拟的。独享IP指每个网站都有独立的IP地址,而虚…...

R语言——taxize(第二部分)
taxize(第二部分) 3. taxize 文档中译3.10. classification(根据类群ID检索分类阶元层级)示例1:传递单个ID值示例2:传递多个ID值示例3:传递单个名称示例4:传递多个名称示例5…...

Postman+Newman+Jenkins实现接口测试持续集成
近期在复习Postman的基础知识,在小破站上跟着百里老师系统复习了一遍,也做了一些笔记,希望可以给大家一点点启发。 1.新建一个项目 2.设置自定义工作空间 3.执行windows的批处理命令 4.执行系统的Groovy脚本 5.生成的HTML的报告集成到Jenkin…...

C#WPF中的实现读取和写入文件的几种方式
说明:C#中实现读取和写入的类根据需要来选择。 1、File类 File类是用于操作文件的工具类,提供了对文件进行创建、复制、删除、移动和打开单一文件的静态方法。但需要注意的是,WPF中使用File的类,需要先引用System.IO下的命名空间。…...

如何利用自动发现将现网的进程纳入到监控系统中?
进程监控是一项关键任务,旨在监测系统中运行的进程的性能和状态。通过有效的进程监控,可以实时了解进程的运行情况,及时发现问题并采取措施,确保系统的稳定性和性能。 本期EasyOps产品使用最佳实践,我们将为您揭晓&am…...