OpenCV 图像处理六(傅里叶变换、模板匹配与霍夫变换)

文章目录

  • 一、傅里叶变换
    • 1.1 NumPy实现和逆实现
      • 1.1.1 NumPy实现傅里叶变换
        • Demo
      • 1.1.2 NumPy实现逆傅里叶变换
        • Demo
    • 1.2 OpenCV实现和逆实现
      • 1.2.1 OpenCV实现傅里叶变换
        • Demo
      • 1.2.2 OpenCV实现逆傅里叶变换
        • Demo
    • 1.3 频域滤波
      • 1.3.1低频、高频
      • 1.3.2 高通滤波器
        • 构造高通滤波器
        • Demo
      • 1.3.3 低通滤波器
        • 设置低通滤波器
        • Demo
  • 二、模板匹配
    • 2.1 模板匹配是使用函数 cv2.matchTemplate()实现的。
    • 2.2 依据method获取想要的最值及其位置需要以使cv2.minMaxLoc()函数实现
      • 2.2.1 选择
      • 2.2.2 显现标记匹配位置
      • 2.2.3 Demo
  • 三、霍夫变换
    • 3.1 霍夫直线变换 -- HoughLines函数
      • 3.1.1 cv2.line()可以在图像上绘制直线,即使直线的端点坐标超出了图像的范围,也可以正确地绘制。
      • 3.1.2 Demo
    • 3.2 概率霍夫变换 cv2.HoughLinesP
    • 3.3 霍夫圆环变换

一、傅里叶变换

任何周期函数都可以表示为不同频率的正弦函数和的形式。

在这里插入图片描述

傅里叶变换(空间域变换到频率域)、 逆傅里叶变换(频率域信息变换到空间域)
(1)幅度图像(Magnitude Image)和相位图像(Phase Image):幅度图像显示了每个频率分量的强度或能量信息,它的取值范围是非负的。相位图像显示了每个频率分量的相位信息,它的取值范围通常是[-π, π]。通过显示幅度图像和相位图像可以很直观地观察到图像中不同频率分量的贡献和相位信息。
(2)实数图像和虚数图像:实数图像和虚数图像是将复数的傅里叶变换结果分别表示为实部和虚部的图像。实数图像显示了图像中的低频信息,而虚数图像则显示了图像中的高频信息。这种可视化方法将复数形式的傅里叶变换结果拆分为两个部分,更加直观地展示了图像中不同频率分量的特点。

1.1 NumPy实现和逆实现

通过傅里叶变换可以将图像分解为不同频率分量,并且频率越高的分量对应于图像中变化越快的部分,即边缘和细节等。而频率越低的分量对应于图像中变化缓慢的部分,即平滑区域和低频信号等。这种分解可以帮助我们进行频域滤波、图像恢复和特征提取等图像处理任务。

1.1.1 NumPy实现傅里叶变换

函数:
(1)

#实现变化,返回的是一个复数数组
np.fft.fft2

在这里插入图片描述

(2)

# 将零频率分量移动到中心
np.fft.ffshift

不加shift
在这里插入图片描述
加shift
在这里插入图片描述

(3)

# 设置批频谱的范围
magnitude_spectrum = 20*np.log(np.abs(fshift))
Demo
import numpy as np
import cv2  as cv
import matplotlib.pyplot as pltimg=cv.imread('./img/test_img.jpg',0)
f=np.fft.fft2(img)
fshift = np.fft.fftshift(f)
magnitude_spectrum = 20*np.log(np.abs(fshift))
plt.subplot(121)
plt.imshow(img, cmap = 'gray')
plt.title('original')
plt.axis('off')
plt.subplot(122)
plt.imshow(magnitude_spectrum, cmap = 'gray')
plt.title('result')
plt.axis('off')
plt.show()

1.1.2 NumPy实现逆傅里叶变换

由频谱回到图像,在频域对图像进行处理,在频域的处理会反映在逆变换图像
(1)

# 实现逆傅里叶变换,返回应该复数数组
np.fft.ifft2

(2)

# 是fftshift逆函数
np.fft.ifftshift

(3)

# 设置值的范围
iimg=np.abs(逆傅里叶变换的结果)
Demo
import numpy as np
import  cv2 as cv
import matplotlib.pyplot as pltimg=cv.imread('./img/test_img.jpg',0)
#逆傅里叶变换
f=np.fft.fft2(img)
fshift = np.fft.fftshift(f)
ishift = np.fft.ifftshift(fshift)
iimg=np.fft.ifft2(ishift)
iimg=np.abs(iimg)
plt.subplot(131)
magnitude_spectrum = 20*np.log(np.abs(fshift))
plt.imshow(img, cmap = 'gray')
plt.title('original')
plt.axis('off')plt.subplot(132)
plt.imshow(magnitude_spectrum, cmap = 'gray')
plt.title('magnitude_spectrum')
plt.axis('off')plt.subplot(133)
plt.imshow(iimg, cmap = 'gray')
plt.title('result')
plt.axis('off')

在这里插入图片描述

1.2 OpenCV实现和逆实现

1.2.1 OpenCV实现傅里叶变换

(1)

#双通道,实数 、虚数
result=cv2.dft(原始图像,转换标识)

原始图像
首先被转换成np.float32格式,可以使用np.float32(img)来进行转换。

转换标识flags=cv2.DFT_COMPLEX_OUTPUT用于指定输出一个复数数组。

(2)

np.fft.ffshift

(3)用于计算复数的幅度的函数

result=cv2.magnitude(参数1,参数2

参数1和参数2是两个具有相同形状和数据类型的数组,分别表示复数的实部和虚部。

a + bi(其中 a 是实部,b 是虚部)
sqrt(a^2 + b^2)

Demo
import cv2 as cv
import numpy as np
import matplotlib.pyplot as plt
img = cv.imread('./img/test_img.jpg', 0)
# 进行傅里叶变换
dft = cv.dft(np.float32(img), flags=cv.DFT_COMPLEX_OUTPUT)
dft_shift = np.fft.fftshift(dft)
# 计算幅度谱
magnitude_spectrum = 20 * np.log(cv.magnitude(dft_shift[:, :, 0], dft_shift[:, :, 1]))
plt.subplot(121)
plt.imshow(img, cmap='gray')
plt.title('Original Image')
plt.axis('off')
plt.subplot(122)
plt.imshow(magnitude_spectrum, cmap='gray')
plt.title('Magnitude Spectrum')
plt.axis('off')
plt.show()

在这里插入图片描述
这段代码首先使用cv.imread()加载图像,然后使用cv.dft()进行傅里叶变换。接下来,使用np.fft.fftshift()将频率分量移动到图像中心。最后,通过cv.magnitude()计算复数的幅度,并使用20 * np.log()对结果进行缩放以便于显示。
注意:在这个例子中,傅里叶变换的结果是复数,cv.magnitude()用于计算复数的幅度并生成幅度谱。

1.2.2 OpenCV实现逆傅里叶变换

(1)

取决于元数据的类型和大小,双通道的
返回=cv.idft(原始数据)

(2)

 np.fft.ifftshift(magnitude_spectrum)

(3)

# 双通道的形式换成255的
cv.magnitude(img_back[:, :, 0], img_back[:, :, 1])
Demo
import cv2 as cv
import numpy as np
import matplotlib.pyplot as plt
# 读取图像
img = cv.imread('./img/test_img.jpg', 0)
# 进行傅里叶变换
dft = cv.dft(np.float32(img), flags=cv.DFT_COMPLEX_OUTPUT)
dft_shift = np.fft.fftshift(dft)
magnitude_spectrum = 20 * np.log(cv.magnitude(dft_shift[:, :, 0], dft_shift[:, :, 1]))
# 进行逆傅里叶变换
fshift_inverse = np.fft.ifftshift(dft_shift)
img_back = cv.idft(fshift_inverse)
img_back = cv.magnitude(img_back[:, :, 0], img_back[:, :, 1])
# 显示图像对比
plt.subplot(131)
plt.imshow(img, cmap='gray')
plt.title('Original Image')
plt.axis('off')
plt.subplot(132)
plt.imshow(magnitude_spectrum, cmap='gray')
plt.title('Magnitude Spectrum')
plt.axis('off')
plt.subplot(133)
plt.imshow(img_back, cmap='gray')
plt.title('Inverse Fourier Transform')
plt.axis('off')
plt.tight_layout()
plt.show()

在这里插入图片描述

1.3 频域滤波

目的:图像增强、图像去噪、边缘检测、特征提取、压缩、加密等。

在图像中,低频分量表示颜色或灰度变化较为平缓的区域,通常对应图像中的平滑部分;

假设你正在拍摄一片平静的湖面。湖水的平静部分将对应低频分量,因为它们变化缓慢且颜色趋于一致。

高频分量表示变化较快、灰度过渡尖锐的区域,通常对应图像中的细节部分

观察一幅包含文字的图像时,文字的边缘对应高频分量。这是因为文字边缘通常具有锐利的灰度过渡,由于字体的笔画和字母之间的间隔。

1.3.1低频、高频

低通滤波器允许低频分量通过,同时衰减高频分量,从而模糊图像。这种滤波器可以用来平滑图像,去除噪声或减少图像中不必要的细节。

相反,高通滤波器允许高频分量通过,同时衰减低频分量,以增强图像中的尖锐细节。这会使图像更加清晰,但可能会导致对比度的降低。

1.3.2 高通滤波器

去除中心的fftshift

构造高通滤波器
# 构造高通滤波器
rows, cols = img.shape
crow, ccol = int(rows / 2), int(cols / 2)  # 中心位置
filter = np.ones_like(img)  # 创建与图像大小相同的滤波器
d = 30  # 高通滤波半径阈值
filter[crow - d: crow + d, ccol - d: ccol + d] = 0  #将中心d半径内的值设为0
Demo
import numpy as np
import cv2 as cv
import matplotlib.pyplot as plt
# 读取图像
img = cv.imread('./img/test_img.jpg', 0)
# 进行傅里叶变换
f = np.fft.fft2(img)
fshift = np.fft.fftshift(f)
# 构造高通滤波器
rows, cols = img.shape
crow, ccol = int(rows / 2), int(cols / 2)  # 中心位置
filter = np.ones_like(img)  # 创建与图像大小相同的滤波器
d = 30  # 高通滤波半径阈值
filter[crow - d: crow + d, ccol - d: ccol + d] = 0  #将中心d半径内的值设为0
# 进行滤波操作
fshift_filtered = fshift * filter
# 进行傅里叶逆变换
f_ishift = np.fft.ifftshift(fshift_filtered)
img_filtered = np.fft.ifft2(f_ishift)
img_filtered = np.abs(img_filtered)
# 显示结果
plt.subplot(131)
plt.imshow(img, cmap='gray')
plt.title('Original')
plt.axis('off')
plt.subplot(132)
plt.imshow(np.log(1 + np.abs(fshift)), cmap='gray')
plt.title('Magnitude Spectrum')
plt.axis('off')
plt.subplot(133)
plt.imshow(img_filtered, cmap='gray')
plt.title('Filtered Image')
plt.axis('off')
plt.show()

在这里插入图片描述

1.3.3 低通滤波器

去除边缘的fftshift

设置低通滤波器
# 设置低通滤波器
rows, cols = img.shape
crow, ccol = rows//2, cols//2
mask = np.zeros((rows, cols, 2), np.uint8)
mask[crow-30:crow+30, ccol-30:ccol+30] = 1  # 选择中心区域作为低频成分
Demo
import cv2 as cv
import numpy as np
import matplotlib.pyplot as plt
# 读取图像
img = cv.imread('./img/test_img.jpg', 0)
# 进行傅里叶变换
dft = cv.dft(np.float32(img), flags=cv.DFT_COMPLEX_OUTPUT)
dft_shift = np.fft.fftshift(dft)
# 设置滤波器
rows, cols = img.shape
crow, ccol = rows//2, cols//2
mask = np.zeros((rows, cols, 2), np.uint8)
mask[crow-30:crow+30, ccol-30:ccol+30] = 1  # 选择中心区域作为低频成分
# 进行频域滤波
fshift = dft_shift * mask
f_ishift = np.fft.ifftshift(fshift)
img_back = cv.idft(f_ishift)
img_back = cv.magnitude(img_back[:, :, 0], img_back[:, :, 1])
# 显示原图和滤波后的图像
plt.subplot(121)
plt.imshow(img, cmap='gray')
plt.title('Original Image')
plt.axis('off')
plt.subplot(122)
plt.imshow(img_back, cmap='gray')
plt.title('Low Pass Filtered Image')
plt.axis('off')
plt.tight_layout()
plt.show()

模糊了,平滑图像,去除噪声在这里插入图片描述

二、模板匹配

在模板匹配中,通常将图像A称为输入图像或源图像,将图像B称为模板图像或搜索图像。模板匹配的目标是在输入图像中寻找与模板图像最相似的部分。

2.1 模板匹配是使用函数 cv2.matchTemplate()实现的。

result = cv2.matchTemplate(image, templ, method, result=None, mask=None)

参数
image – 运行搜索的图像。它必须是 8 位或 32 位浮点。
templ – 搜索的模板。它不能大于源图像,并且具有相同的数据类型。
result是一个单通道 32 位浮点型。如果输入图像(原始图像)尺寸是 WH,模板的尺寸是 wh, 则返回值的大小为(W-w+1)*(H-h+1)。

method
在这里插入图片描述
说明
在这里插入图片描述
(1)对于cv2.TM_SQDIFFcv2.TM_SQDIFF_NORMED方法,匹配结果的值越小,表示匹配程度越高。即匹配结果最小的位置是最佳的匹配位置。
(2)对于cv2.TM_CCORRcv2.TM_CCORR_NORMEDcv2.TM_CCOEFFcv2.TM_CCOEFF_NORMED方法,匹配结果的值越大,表示匹配程度越高。即匹配结果最大的位置是最佳的匹配位置。
(3)通过比较匹配结果的值,可以确定模板与输入图像的匹配程度。较小的值或较大的值表示较好的匹配。具体的阈值和判断标准可以根据具体的应用场景和需求来确定。通常,我们会选择一个阈值,当匹配结果超过或低于该阈值时,判定为有效的匹配或无效的匹配。这个阈值的选择可以根据实验和经验来进行调整。

2.2 依据method获取想要的最值及其位置需要以使cv2.minMaxLoc()函数实现

minVal, maxVal, minLoc, maxLoc = cv2.minMaxLoc( src [, mask] )

src:单通道的输入数组。

minVal:返回的最小值,如果没有最小值,则可以是空值。

maxVal:返回的最大值,如果没有最大值,则可以是空值。

minLoc:最小值的位置,如果没有最小值,则可以是空值。

maxLoc:最大值的位置,如果没有最大值,则可以是空值。

mask 为用来选取掩模的子集

2.2.1 选择

(1)当 method 的值为 cv2.TM_SQDIFFcv2.TM_SQDIFF_NORMED 时,0 表示最佳匹配,值越大表示匹配效果越差。因此,在使用这两种方法时,要寻找最小值所在的位置作为最佳匹配

minVal, maxVal, minLoc, maxLoc = cv2.minMaxLoc(matchTemplate 函数的返回值
topLeft = minLoc# 以 topLeft 点为模板匹配位置的左上角坐标,
#根据模板图像的宽度和高度确定匹配位置的右下角坐标
bottomRight = (topLeft[0] + w, topLeft[1] + h) #w 和 h 是模板图像的宽度和高度

(2)当 method 的值为cv2.TM_CCORRcv2.TM_CCORR_NORMEDcv2.TM_CCOEFFcv2.TM_CCOEFF_NORMED 时,cv2.matchTemplate() 函数的返回值越小表示匹配度越差,而返回值越大表示匹配度越好。

minVal, maxVal, minLoc, maxLoc = cv2.minMaxLoc(matchTemplate 函数的返回值
topLeft = maxLoc# 以 topLeft 点为模板匹配位置的左上角坐标,
#根据模板图像的宽度和高度确定匹配位置的右下角坐标
bottomRight = (topLeft[0] + w, topLeft[1] + h)  # w 和 h 是模板图像的宽度和高度

2.2.2 显现标记匹配位置

rectangle(img, pt1, pt2, color, thickness=None, lineType=None, shift=None)

在这里插入图片描述

2.2.3 Demo

TM_SQDIFF
topLeft = minLoc

import cv2
import numpy as np
from matplotlib import pyplot as plt
img = cv2.imread('img/x.jpg')
template = cv2.imread('img/x1.jpg')
th, tw,tl = template.shape
rv = cv2.matchTemplate(img,template,cv2.TM_SQDIFF)
minVal, maxVal, minLoc, maxLoc = cv2.minMaxLoc(rv)
topLeft = minLoc
bottomRight = (topLeft[0] + tw, topLeft[1] + th)
cv2.rectangle(img,topLeft, bottomRight, (0,255,0), 30)
plt.subplot(121),plt.imshow(rv,cmap = 'gray')
plt.title('Matching Result'), plt.xticks([]), plt.yticks([])
plt.subplot(122),plt.imshow(img[:,:,::-1])
plt.title('Detected Point'), plt.xticks([]), plt.yticks([])
plt.show()

在这里插入图片描述

cv2.TM_CCOEFF
topLeft = maxLoc

import cv2
import numpy as np
from matplotlib import pyplot as plt
img = cv2.imread('img/x.jpg')
template = cv2.imread('img/x1.jpg')
th, tw ,tl= template.shape
rv = cv2.matchTemplate(img, template, cv2.TM_CCOEFF)
minVal, maxVal, minLoc, maxLoc = cv2.minMaxLoc(rv)
# 将变量名更新为对应的最大值
topLeft = maxLoc
bottomRight = (topLeft[0] + tw, topLeft[1] + th)
# 绘制矩形框
cv2.rectangle(img, topLeft, bottomRight, (0, 255, 0), 30)
# 显示结果
plt.subplot(121), plt.imshow(rv, cmap='gray')
plt.title('Matching Result'), plt.xticks([]), plt.yticks([])
plt.subplot(122), plt.imshow(img[:,:,::-1])
plt.title('Detected Point'), plt.xticks([]), plt.yticks([])
plt.show()

三、霍夫变换

霍夫变换(Hough Transform)是一种用于在数字图像中检测几何形状的技术。用于检测直线形状,还有其他如圆、椭圆等。

3.1 霍夫直线变换 – HoughLines函数

lines = cv2.HoughLines(image, rho, theta, threshold)

image:输入图像,通常为二值化图像(单通道,黑白)

rho:以像素为单位的距离精度,一般为1

theta:以弧度为单位的角度精度,一般为 np.pi/180

threshold:阈值,检测到的直线必须满足的投票数阈值

函数返回一个包含检测到的直线参数(rho和theta值)的数组,每一行表示一条检测到的直线。
使用霍夫直线变换函数,我们可以在图像中检测直线并对其进行分析、处理或绘制。

3.1.1 cv2.line()可以在图像上绘制直线,即使直线的端点坐标超出了图像的范围,也可以正确地绘制。

cv2.line(image, start_point, end_point, color, thickness)

image:要在其上绘制线段的图像

start_point:线段的起始点坐标,格式为 (x, y)

end_point:线段的结束点坐标,格式为 (x, y)

color:线段的颜色,可以是一个整数值(灰度图像)或一个包含三个整数值的元组/列表(彩色图像),表示蓝色、绿色、红色的值(BGR顺序)

thickness:线段的粗细,默认值为1

3.1.2 Demo

import cv2
import numpy as np
# 读取图像
image = cv2.imread('img/S1.jpg')
# 转换为灰度图像
gray = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)
# 进行边缘检测
edges = cv2.Canny(gray, 50, 150)
# 进行霍夫直线变换
lines = cv2.HoughLines(edges, 1, np.pi/180, 150)
# 绘制直线
if lines is not None:for line in lines:rho, theta = line[0]a = np.cos(theta)b = np.sin(theta)x0 = a * rhoy0 = b * rhopt1 = (int(x0 + 1000*(-b)), int(y0 + 1000*(a)))pt2 = (int(x0 - 1000*(-b)), int(y0 - 1000*(a)))cv2.line(image, pt1, pt2, (0, 0, 255), 2)
# 显示结果
cv2.imshow('Lines', image)
cv2.waitKey(0)
cv2.destroyAllWindows()

在这里插入图片描述

3.2 概率霍夫变换 cv2.HoughLinesP

lines =cv2.HoughLinesP(image, rho, theta, threshold, minLineLength,
maxLineGap)

在这里插入图片描述

cv2.HoughLinesP()函数是Hough变换的改进版本,主要在直线检测算法上进行了改进和优化。
传统的Hough变换是基于参数空间的方法,需要遍历所有像素点和所有可能的直线参数来进行计算,计算量较大,效率较低。而cv2.HoughLinesP()函数采用了概率霍夫变换(Probabilistic Hough Transform)的方法,通过随机采样的方式来快速寻找直线。
概率霍夫变换通过随机选择一部分边缘点来计算直线方程,并根据这些点的相对位置进行线段的连接,而不是遍历整张图像的所有像素点。这种随机采样的方式能够大大减少计算量,提高直线检测的速度。
另外,cv2.HoughLinesP()函数还引入了两个新的参数:minLineLength和maxLineGap。minLineLength参数用于指定线段的最小长度,比这个长度短的线段将被舍弃。maxLineGap参数用于定义在同一行中被认为是一条线段的最大间隙。这两个参数的引入使得直线检测更加准确和灵活。

import cv2
import numpy as np
# 读取图像
image = cv2.imread("img8/testSobelimg.jpg")
gray = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)
# 进行边缘检测
edges = cv2.Canny(gray, 50, 150)
# 进行直线检测
lines = cv2.HoughLinesP(edges, 1, np.pi/180, 100, minLineLength=100, maxLineGap=10)
# 绘制检测到的直线
if lines is not None:for line in lines:x1, y1, x2, y2 = line[0]cv2.line(image, (x1, y1), (x2, y2), (0, 0, 255), 2)
# 显示图像
cv2.imshow("Image", image)
cv2.waitKey(0)
cv2.destroyAllWindows()

在这里插入图片描述

3.3 霍夫圆环变换

cv2.HoughCircles(image, method, dp, minDist[, circles[, param1[, param2[, minRadius[, maxRadius]]]]]) 

在这里插入图片描述

在这里插入图片描述

import cv2
import numpy as np# 读取图像
image = cv2.imread('Pic/show.png', cv2.IMREAD_GRAYSCALE)# 霍夫圆环变换参数设置
dp = 1  # 累加器分辨率与图像分辨率之比的倒数
minDist = 50  # 检测到的圆心之间的最小距离
param1 = 100  # Canny边缘检测阈值上限
param2 = 30  # 圆心累加器阈值
minRadius = 10  # 最小圆半径
maxRadius = 100  # 最大圆半径# 进行霍夫圆环变换
circles = cv2.HoughCircles(image, cv2.HOUGH_GRADIENT, dp, minDist,param1=param1, param2=param2,minRadius=minRadius, maxRadius=maxRadius)if circles is not None:circles_rounded = np.uint16(np.around(circles))for circle in circles_rounded[0, :]:center = (circle[0], circle[1])radius = circle[2]# 绘制圆和中心点cv2.circle(image, center, radius, (0, 255, 0), 3)cv2.circle(image, center, 3, (0, 0, 255), -1)# 显示结果图像cv2.imshow('Hough Circles', image)cv2.waitKey(0)cv2.destroyAllWindows()

(1)当在这里插入图片描述
在这里插入图片描述

(2)当
在这里插入图片描述

在这里插入图片描述

(3)

在这里插入图片描述

在这里插入图片描述

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

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

相关文章

Django前后端分离之后端实践

django-admin startproject djweb 生成djweb项目 django-admin startapp news 生成news应用 配置models文件 class NewInfo(models.Model):title models.CharField(max_length30)content models.TextField()b_date models.DateField()read models.IntegerFie…

Blazor Wasm Gitee 码云登录

目录: OpenID 与 OAuth2 基础知识Blazor wasm Google 登录Blazor wasm Gitee 码云登录Blazor SSR/WASM IDS/OIDC 单点登录授权实例1-建立和配置IDS身份验证服务Blazor SSR/WASM IDS/OIDC 单点登录授权实例2-登录信息组件wasmBlazor SSR/WASM IDS/OIDC 单点登录授权实例3-服务端…

ClickHouse--01--简介

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 1. ClickHouse 简介1.1 大数据处理场景1.2 什么是 ClickHouse1.3 OLAP 场景的特征 2. ClickHouse 特性2.1 完备的 DBMS 功能2.2 列式存储行式存储: 在数据写入和修改…

OpenEuler20.03LTS SP2 上安装 OpenGauss3.0.0 单机部署过程(二)

开始安装 OpenGauss 数据库 3.1.7 安装依赖包 (说明:如果可以联网,可以通过网络 yum 安装所需依赖包,既可以跳过本步骤。如果网络无法连通,请把本文档所在目录下的依赖包上传到服务器上,手工安装后,即无需通过网络进行 Yum 安装了): 上传:libaio-0.3.111-5.oe1.x8…

三、搜索与图论

DFS 排列数字 #include<iostream> using namespace std; const int N 10; int a[N], b[N]; int n;void dfs(int u){if(u > n){for(int i 1; i < n; i)cout<<a[i]<<" ";cout<<endl;return;}for(int i 1; i < n; i){if(!b[i]){b[…

自己DIY制作耳机壳一般用哪种材料比较好,性价比比较高

在选择耳机壳的材料时&#xff0c;除了考虑材料本身的性能外&#xff0c;还需要考虑成本、加工难度、耐用性、环保性等方面的因素。 从性能方面来看&#xff1a; 制作耳机壳的UV树脂和塑料材质各有其优缺点。UV树脂具有高硬度、耐磨、耐高温、环保等优点&#xff0c;能够提供更…

【FFmpeg】ffplay 命令行参数 ⑤ ( 设置音频滤镜 -af 参数 | 设置统计信息 -stats 参数 | 设置同步时钟类型 -sync 参数 )

文章目录 一、ffplay 命令行参数 - 音频滤镜1、设置音频滤镜 -af 参数2、常用的 音频滤镜 参数3、音频滤镜链 示例 二、ffplay 命令行参数 - 统计信息1、设置统计信息 -stats 参数2、关闭统计信息 -nostats 参数 三、ffplay 命令行参数 - 同步时钟类型1、设置同步时钟类型 -syn…

【前沿技术杂谈:多模态文档基础模型】使用多模态文档基础模型彻底改变文档 AI

【前沿技术杂谈&#xff1a;多模态文档基础模型】使用多模态文档基础模型彻底改变文档 AI 从文本到多模态模型&#xff1a;文档 AI 逐渐发展新技能。行业领先的型号Document AI 的下一步&#xff1a;开发通用和统一框架 您是否曾经被包含不同信息&#xff08;如应付账款、日期、…

Web后端开发:事务与AOP

事务管理 在学习数据库时&#xff0c;讲到&#xff1a;事务是一组操作的集合&#xff0c;它是一个不可分割的工作单位。事务会把所有的操作作为一个整体&#xff0c;一起向数据库提交或者是撤销操作请求&#xff0c;要么同时成功&#xff0c;要么同时失败。 事务的操作主要有三…

C#向数组指定索引位置插入新的元素值:自定义插入方法 vs List<T>.Add(T) 方法

目录 一、使用的方法 1.自定义插入方法 2.使用List.Add(T) 方法 二、实例 1.示例1&#xff1a;List.Add(T) 方法 2.示例&#xff1a;自定义插入方法 一、使用的方法 1.自定义插入方法 首先需要定义一个一维数组&#xff0c;然后修改数组的长度(这里使用Length属性获取…

网关中全局过滤器实现jwt校验

意味着有很多相同接口的实现类&#xff0c;那么必定会有优先级的问题。于是Spring就提供了Ordered这个接口&#xff0c;来处理相同接口实现类的优先级问题。 public class AuthorizeFilter implements Ordered, GlobalFilter {Overridepublic Mono<Void> filter(ServerW…

Linux应用程序几种参数传递方式

大家好&#xff0c;今天给大家介绍Linux应用程序几种参数传递方式&#xff0c;文章末尾附有分享大家一个资料包&#xff0c;差不多150多G。里面学习内容、面经、项目都比较新也比较全&#xff01;可进群免费领取。 在Linux中&#xff0c;应用程序可以通过多种方式接收参数。以下…

栈的基本操作

&#x1d649;&#x1d65e;&#x1d658;&#x1d65a;!!&#x1f44f;&#x1f3fb;‧✧̣̥̇‧✦&#x1f44f;&#x1f3fb;‧✧̣̥̇‧✦ &#x1f44f;&#x1f3fb;‧✧̣̥̇:Solitary-walk ⸝⋆ ━━━┓ - 个性标签 - &#xff1a;来于“云”的“羽球人”。…

【Java面试】数据类型常见面试题

什么是包装类型 将基本类型包装进了对象中得到的类型 基本类型和包装类型有什么区别 用途不同&#xff1a;基本类型一般用于局部变量&#xff0c;包装类型用于其他地方存储方式不同&#xff1a;用于局部变量的基本类型存在虚拟机栈中的局部变量表中&#xff0c;用于成员变量…

物联网数据隐私保护技术

在物联网&#xff08;IoT&#xff09;的世界中&#xff0c;无数的设备通过互联网连接在一起&#xff0c;不断地收集、传输和处理数据。这些数据有助于提高生产效率、优化用户体验并创造新的服务模式。然而&#xff0c;随着数据量的剧增&#xff0c;数据隐私保护成为了一个不能忽…

力扣刷题之旅:进阶篇(三)

力扣&#xff08;LeetCode&#xff09;是一个在线编程平台&#xff0c;主要用于帮助程序员提升算法和数据结构方面的能力。以下是一些力扣上的入门题目&#xff0c;以及它们的解题代码。 --点击进入刷题地址 一、动态规划&#xff08;DP&#xff09; 首先&#xff0c;让我们来…

《PCI Express体系结构导读》随记 —— 第II篇 第4章 PCIe总线概述(10)

接前一篇文章&#xff1a;《PCI Express体系结构导读》随记 —— 第II篇 第4章 PCIe总线概述&#xff08;9&#xff09; 4.2 PCIe体系结构的组成部件 PCIe总线作为处理器系统的局部总线&#xff0c;其作用与PCI总线类似&#xff0c;主要目的是为了连接处理器系统中的外部设备&…

【机器学习】基于集成学习的 Amazon 用户评论质量预测

实验六: 基于集成学习的 Amazon 用户评论质量预测 1 案例简介 ​ 随着电商平台的兴起&#xff0c;以及疫情的持续影响&#xff0c;线上购物在我们的日常生活中扮演着越来越重要的角色。在进行线上商品挑选时&#xff0c;评论往往是我们十分关注的一个方面。然而目前电商网站的…

Golang的for循环变量和goroutine的陷阱,1.22版本的更新

先来看一段golang 1.22版本之前的for循环的代码 package mainimport "fmt"func main() {done : make(chan bool)values : []string{"chen", "hai", "feng"}for _, v : range values {fmt.Println("start")go func() {fmt.P…

【05】C++ 内存管理

文章目录 &#x1f308; Ⅰ C 内存分布&#x1f308; Ⅱ C 内存管理方式1. new 和 delete 操作内置类型2. new 和 delete 操作自定义类型 &#x1f308; Ⅲ operator new 和 operator delete&#x1f308; Ⅳ new 和 delete 的实现原理1. 内置数据类型2. 自定义数据类型 &#…