MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《9》

news2025/1/12 1:55:11

MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《1》:论文源地址,克隆MXNet版本的源码,安装环境与测试,以及对下载的源码的每个目录做什么用的,做个解释。

MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《2》:对论文中的区域提议、平移不变锚、多尺度预测等概念的了解,对损失函数、边界框回归的公式的了解,以及共享特征的训练网络的方法。

MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《3》:加载模型参数,给目标加锚框可视化操作,对参数文件的了解,以及感兴趣区域ROI和泛洪填充的方法(FLOODFILL_FIXED_RANGE,FLOODFILL_MASK_ONLY)

MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《4》:下载与熟悉Pascal VOC2007,2012语义分割数据集,明白实例分割除了分类之外,还可以细分到像素级别的所属类别。

MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《5》:主要就是熟悉转置卷积与大家所熟知的卷积有什么区别,作用是什么,以及双线性插值等相关知识

MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《6》:主要讲解关于参数解析的安全执行(ast.literal_eval),ROI池化以及计算图的可视化的处理

MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《7》:打印内容(比如参数文件里的东西)的三种方式以及对奇异值分解(SVD,Singular Value Decomposition)的熟悉,了解SVD的作用和运用

MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《8》:主要是通过参数的设置进一步熟悉模型,以及对于符号式编程的复习,另外关于损失函数之类,这里用到了自定义评价函数,然后通过自带的mx.metric来做,有示例让大家熟悉。

从上面的文章一路看过来的伙伴们,应该对这个框架有一定的了解了,这里先概括下整体是怎么个大概流程,当然在前面的可视化当中也可以具体知道。

首先输入数据进入到多层卷积神经网络,通过Conv--relu--Pooling获取特征图,这些特征图是用来给后续的区域提议网络RPN(Region Proposal Networks)和全连接层(Fully Conneted Layers)使用的

我们知道这个RPN是用来生成区域提议(region proposals)的,通过Softmax判断锚框是正类锚框还是负类锚框,然后利用边界框回归来修正锚框获得精确的提议。

在MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《6》 这节我们介绍了ROI池化,这层收集输入的特征图和proposals提议,通过这些信息提取提议特征图(proposal feature maps),最后输入到全连接层判定目标类别。
最后我们利用上面获取到的提议特征图计算proposal的类别,同时再次使用边界框回归(bounding box regression)获得检测框最终的精确位置

这节我们从test.py测试模型来了解下有哪些相关知识点,由于配置低在训练的时候没法训练,在测试的时候也只能将test_net(sym, imdb, args)注释掉,但不影响我们接下来的学习,我们先来看下这个加载的测试数据方法TestLoader
测试命令:python test.py --dataset voc --network vgg16 --params model/vgg16-0010.params

# 加载测试数据
test_data = TestLoader(imdb.roidb, batch_size=1, short=args.img_short_side, max_size=args.img_long_side,
                           mean=args.img_pixel_means, std=args.img_pixel_stds)

然后我们查看TestLoader,这个类class TestLoader(mx.io.DataIter)的参数是一个MXNet的迭代基类mx.io.DataIter
MXNet中的所有I/O都由该类的专门化处理。通过调用next来返回DataBatch,然后我们接着TestLoader往下看可以看到,它的属性、方法以及我们需要关注的next方法中的getdata方法,这个获取数据的方法出现im_tensor = mx.nd.array(tensor_vstack(im_tensor, pad=0))这个tensor_vstack方法。接下来会专门介绍它!

张量的垂直叠加

 这个张量的垂直叠加,跟numpy中自带的np.vstack有区别,我们贴出源码:

def tensor_vstack(tensor_list, pad=0):
    """
    通过添加一个新轴来垂直叠加张量
    :tensor_list: 张量列表
    :pad: 填充的值
    :返回最大形状张量
    """
    if len(tensor_list) == 1:
        return tensor_list[0][np.newaxis, :]

    ndim = len(tensor_list[0].shape)
    dimensions = [len(tensor_list)]  # first dim is batch size
    for dim in range(ndim):
        dimensions.append(max([tensor.shape[dim] for tensor in tensor_list]))

    dtype = tensor_list[0].dtype
    if pad == 0:
        all_tensor = np.zeros(tuple(dimensions), dtype=dtype)
    elif pad == 1:
        all_tensor = np.ones(tuple(dimensions), dtype=dtype)
    else:
        all_tensor = np.full(tuple(dimensions), pad, dtype=dtype)
    if ndim == 1:
        for ind, tensor in enumerate(tensor_list):
            all_tensor[ind, :tensor.shape[0]] = tensor
    elif ndim == 2:
        for ind, tensor in enumerate(tensor_list):
            all_tensor[ind, :tensor.shape[0], :tensor.shape[1]] = tensor
    elif ndim == 3:
        for ind, tensor in enumerate(tensor_list):
            all_tensor[ind, :tensor.shape[0], :tensor.shape[1], :tensor.shape[2]] = tensor
    else:
        raise Exception('Sorry, unimplemented.')
    return all_tensor

如果是在这个源码当中测试的话,也可以直接使用from symdata.image import tensor_vstack导入即可。从代码得知,按照最大形状来垂直叠加张量,一维变二维、二维变三维、三维变四维。

这里默认的填充是0,如果指定了其他值的话,填充的就是指定的值。我们来看几个示例: 

a=np.array([[1,2,3,4],[7,8,9,10]])
b=np.array([[11,22,33,44],[77,88,99,1010],[333,444,555,666]])
np_vstack=np.vstack((a,b))
print(type(np_vstack),np_vstack.shape)#<class 'numpy.ndarray'> (5, 4)
print(np_vstack)
'''
[[   1    2    3    4]
 [   7    8    9   10]
 [  11   22   33   44]
 [  77   88   99 1010]
 [ 333  444  555  666]]
'''

这个是np的张量的垂直堆叠,跟np.concatenate((a,b))效果一样。我们看到的形状是(5,4),二维的,维度是没有变化的!

接下来我们测试上面的函数tensor_vstack有什么不一样,从参数(tensor_list, pad=0)我们看出,是张量的列表,也就是说,列表里面的元素是由张量构成。

c=[]
c.append(a)
c.append(b)
print(c)
'''
[array([[ 1,  2,  3,  4],
       [ 7,  8,  9, 10]]), array([[  11,   22,   33,   44],
       [  77,   88,   99, 1010],
       [ 333,  444,  555,  666]])]
'''
t_vstack=tensor_vstack(c)
print(type(t_vstack),t_vstack.shape)#<class 'numpy.ndarray'> (2, 3, 4)
print(t_vstack)
'''
[[[   1    2    3    4]
  [   7    8    9   10]
  [   0    0    0    0]]

 [[  11   22   33   44]
  [  77   88   99 1010]
  [ 333  444  555  666]]]
'''

这里出来的是三维数组了,其中里面的张量的形状是一样的,都是(3,4),其中第一个只有两行,于是最后一行使用0填充。
我们再添加一个二维数组来看下有什么变化:

c.append(np.array([[4,2,1,6]]))
t_vstack=tensor_vstack(c)
print(type(t_vstack),t_vstack.shape)#<class 'numpy.ndarray'> (3, 3, 4)
print(t_vstack)

'''
[[[   1    2    3    4]
  [   7    8    9   10]
  [   0    0    0    0]]

 [[  11   22   33   44]
  [  77   88   99 1010]
  [ 333  444  555  666]]

 [[   4    2    1    6]
  [   0    0    0    0]
  [   0    0    0    0]]]
'''

添加的这个1行4列,同样将变换成3行4列,其余两行都使用0填充。

回过来看,这个方法提供到了三维的数组是吧,然后我们看下三维生成四维的效果:

a=np.array([[[1,2,3,4],[7,8,9,10]]])#(1,2,4)
b=np.array([[[11,22,33,44],[77,88,99,1010],[333,444,555,666]]])#(1, 3, 4)
c=[]
c.append(a)
c.append(b)
c.append(np.array([[[4,2,1,6]]]))#再添加一个(1,1,4)
t_vstack=tensor_vstack(c)
print(type(t_vstack),t_vstack.shape)#<class 'numpy.ndarray'> (3, 1, 3, 4)
print(t_vstack)
'''
[[[[   1    2    3    4]
   [   7    8    9   10]
   [   0    0    0    0]]]


 [[[  11   22   33   44]
   [  77   88   99 1010]
   [ 333  444  555  666]]]


 [[[   4    2    1    6]
   [   0    0    0    0]
   [   0    0    0    0]]]]
'''

三个三维的数组就这样愉快的垂直堆叠在了一起,变换成了四维的数组,形状为(3,1,3,4)的张量,其中不够的行列,就使用0填充了,如果使用其他值,比如用1来填充,只需指定pad参数即可:tensor_vstack(c,pad=1)
这样做的目的是使得不同样本数的输入,可以让它们变得形状一样(通过填充),然后垂直堆叠起来形成一个增加一个维度的张量,便于计算。

我们接着从上到下的顺序来看下每个文件都涉及到了哪些需要学习的知识点,symdata/anchor.py包含两个类AnchorGenerator和AnchorSampler。
AnchorGenerator:生成锚框,锚框数是按照len(anchor_scales) * len(anchor_ratios)【锚框尺度*锚框比例】,这里是3*3=9,也就是在特征图上面是按照这个尺度与比例来生成锚框的,由于特征图是原图抽取的,比如缩小了16倍,那我们需要扩大16倍以及做一些形状变换,得到所有的锚框数,当然具体的细节在最后一篇结束的文章里我将会把源码全部放到github以及有详细的注释,便于帮助大家更快的理解,当然由于水平有限,错误在所难免,欢迎大家指出。
AnchorSampler:锚框采样器,主要就是筛选的作用,保留大于交并比阈值(0.7)的前景锚框和小于交并比阈值0.3的背景锚框,找出最接近真实框的锚框。

交并比IoU的计算

锚框的筛选,需要用到交并比IoU这样一个很关键的方法,也就是衡量锚框跟真实框的交集,重叠部分越多,表示效果越好,越接近真实框,那这个交并比就是交集与并集的比值,另外这里需要注意的是这个坐标值的问题,跟数学中的坐标有区别,计算机里面一般都是从左上角开始的,也就是说往右和往下都是正的值,这点需要注意。
我们来看下symdata/bbox.py里面的交并比方法bbox_overlaps

def bbox_overlaps(boxes, query_boxes):
    """
    determine overlaps between boxes and query_boxes
    :param boxes: n * 4 bounding boxes
    :param query_boxes: k * 4 bounding boxes
    :return: overlaps: n * k overlaps
    """
    n_ = boxes.shape[0]
    k_ = query_boxes.shape[0]
    overlaps = np.zeros((n_, k_), dtype=np.float64)
    for k in range(k_):
        query_box_area = (query_boxes[k, 2] - query_boxes[k, 0] + 1) * (query_boxes[k, 3] - query_boxes[k, 1] + 1)
        print(query_box_area)
        for n in range(n_):
            iw = min(boxes[n, 2], query_boxes[k, 2]) - max(boxes[n, 0], query_boxes[k, 0]) + 1
            if iw > 0:
                ih = min(boxes[n, 3], query_boxes[k, 3]) - max(boxes[n, 1], query_boxes[k, 1]) + 1
                if ih > 0:
                    box_area = (boxes[n, 2] - boxes[n, 0] + 1) * (boxes[n, 3] - boxes[n, 1] + 1)
                    all_area = float(box_area + query_box_area - iw * ih)
                    overlaps[n, k] = iw * ih / all_area
    return overlaps

两个锚框的维度是2维,形状是(左上角坐标x1,y1,右下角坐标x2,y2),N*4的锚框。
我们来测试下这个2组锚框的交并比是怎么样的:

a=np.array([[0,0,100,100],[0,0,200,120]])
b=np.array([[10,10,110,110],[80,80,220,280]])
print(bbox_overlaps(a,b))
'''
[[0.68319446 0.0115745 ]
 [0.41943177 0.10400201]]
'''

交并比的个数就是a.shape[0]*b.shape[0]

大家如果稍微细心点,就会发现,query_boxes[k, 2] - query_boxes[k, 0] + 1等其他位置都有+1,这个表示的是边框线的像素。那如果是没有+1的话,计算结果如下,我们来看下结果:

a=np.array([[0,0,5,5]])
b=np.array([[1,1,6,6]])
print(bbox_overlaps(a,b))#[[0.47058824]]

这个大家手动也可以验证下,没有问题。

我们通过画矩形框图来直观感受下,上面是两组锚框(a是生成的锚框,b是真实锚框),然后a中的锚框分别与b中的锚框做IoU

from PIL import ImageDraw,Image
img = Image.open("1.jpg")#为了让锚框显示更清晰,使用一张白色背景图片
draw = ImageDraw.Draw(img)
img_boxes = np.concatenate((a,b))
color = [(0,0,0),(255,255,0),(255,0,255),(255,0,0),(0,0,255)]
i=0
for box in img_boxes:
    x1,y1,x2,y2 = box
    if i==5:
        i=0
    draw.rectangle([x1,y1,x2,y2],outline=color[i],width=4)
    i=i+1
img.show()

如下图:

这里的a是黑色框和黄色框,b是粉色框与红色框,我们可以看到它们的交集(黑框与粉色、红色的交并比,黄框与粉色、红色的交并比)和对比上面输出的IoU的值,这样就很清晰明了了。

裁剪越界的锚框

生成的锚框会存在一些超出图片边界的锚框,这个时候我们做一个裁剪的操作,将超过了图片宽高边界的部分去掉!这里比较简单,直接看代码:

def clip_boxes(boxes, im_shape):
    """
    裁剪掉超出图片边界的锚框部分
    :param boxes: [N, 4* num_classes]
    :param im_shape: 高宽元组
    :return: [N, 4* num_classes]
    """
    # x1 >= 0
    boxes[:, 0::4] = np.maximum(np.minimum(boxes[:, 0::4], im_shape[1] - 1), 0)
    # y1 >= 0
    boxes[:, 1::4] = np.maximum(np.minimum(boxes[:, 1::4], im_shape[0] - 1), 0)
    # x2 < im_shape[1]
    boxes[:, 2::4] = np.maximum(np.minimum(boxes[:, 2::4], im_shape[1] - 1), 0)
    # y2 < im_shape[0]
    boxes[:, 3::4] = np.maximum(np.minimum(boxes[:, 3::4], im_shape[0] - 1), 0)
    return boxes

边界框回归

对于边界框回归我们在 MXNet的Faster R-CNN(基于区域提议网络的实时目标检测)《2》 这篇文章中有讲到这个边界框回归的公式,有兴趣的可以去看看,然后我们来看下实际的代码:

def bbox_transform(ex_rois, gt_rois, box_stds):
    """
    计算从ex_rois到gt_rois的边界框回归目标(锚框到真实边界框的边界框回归)
    :param ex_rois: [N, 4]
    :param gt_rois: [N, 4]
    :return: [N, 4]
    """
    assert ex_rois.shape[0] == gt_rois.shape[0], 'inconsistent rois number'
    # 预测锚框的宽高与中心位置
    ex_widths = ex_rois[:, 2] - ex_rois[:, 0] + 1.0
    ex_heights = ex_rois[:, 3] - ex_rois[:, 1] + 1.0
    ex_ctr_x = ex_rois[:, 0] + 0.5 * (ex_widths - 1.0)
    ex_ctr_y = ex_rois[:, 1] + 0.5 * (ex_heights - 1.0)
    # 真实框的宽高与中心位置
    gt_widths = gt_rois[:, 2] - gt_rois[:, 0] + 1.0
    gt_heights = gt_rois[:, 3] - gt_rois[:, 1] + 1.0
    gt_ctr_x = gt_rois[:, 0] + 0.5 * (gt_widths - 1.0)
    gt_ctr_y = gt_rois[:, 1] + 0.5 * (gt_heights - 1.0)

    targets_dx = (gt_ctr_x - ex_ctr_x) / (ex_widths + 1e-14) / box_stds[0]	#平移量
    targets_dy = (gt_ctr_y - ex_ctr_y) / (ex_heights + 1e-14) / box_stds[1]	#平移量
    targets_dw = np.log(gt_widths / ex_widths) / box_stds[2]		#尺度因子
    targets_dh = np.log(gt_heights / ex_heights) / box_stds[3]		#尺度因子

    targets = np.vstack((targets_dx, targets_dy, targets_dw, targets_dh)).transpose()
    return targets

当然我们从它的代码实践中可以看到,box_stds标准差都设置成了1.0,所以可以忽略,这样对比公式就更清晰了。相当于是从预测锚框先通过平移然后缩放来接近真实锚框,输出的就是(targets_dx, targets_dy, targets_dw, targets_dh)这样的平移量和尺度因子,这就对预测锚框进行了修正!

NMS非极大值抑制

非极大值抑制的目的是去掉很多重复(交并比比较大)的边界框,首先找出类别分数最大的(降序排序),然后跟其他的边界框进行交并比IoU的计算,如果其值大于设定的阈值就忽略掉,只保留交并比小于阈值的,这样一直循环直到将所有的边界框都比较全部完为止。

贴出代码来看下:

def nms(dets, thresh):
    """
    greedily select boxes with high confidence and overlap with current maximum <= thresh
    rule out overlap >= thresh
    :param dets: [[x1, y1, x2, y2 score]]
    :param thresh: retain overlap < thresh
    :return: indexes to keep
    """
    x1 = dets[:, 0]
    y1 = dets[:, 1]
    x2 = dets[:, 2]
    y2 = dets[:, 3]
    scores = dets[:, 4]# 最后一列的分数

    areas = (x2 - x1 + 1) * (y2 - y1 + 1)
    order = scores.argsort()[::-1]# 降序

    keep = []
    while order.size > 0:
        i = order[0]
        keep.append(i)
        xx1 = np.maximum(x1[i], x1[order[1:]])
        yy1 = np.maximum(y1[i], y1[order[1:]])
        xx2 = np.minimum(x2[i], x2[order[1:]])
        yy2 = np.minimum(y2[i], y2[order[1:]])

        w = np.maximum(0.0, xx2 - xx1 + 1)
        h = np.maximum(0.0, yy2 - yy1 + 1)
        inter = w * h
        ovr = inter / (areas[i] + areas[order[1:]] - inter)

        inds = np.where(ovr <= thresh)[0]
        order = order[inds + 1]

    return keep

然后我们示例4个边界框(带上类别分数)来测试下这个效果:

dets=np.array([[10, 10, 100, 100, 0.61],[5, 20, 100, 100, 0.88],[20, 80, 188, 180, 0.66],[150, 150, 200, 200, 0.55]])
print(nms(dets=dets,thresh=0.7))#[1, 2, 3]

按照分数从大到小排序(0.88,0.66,0.61,0.55)对应的索引值是(2,0,1,3),0.88分数的框跟余下的框进行交并比计算。结果就只剩下了索引值为1,2,3的锚框了

因为第一个跟第二个锚框的重复度很高,大于了设置的阈值,所以去掉第一个,我们也可以使用上面的画图方法来更直观的看下:

from PIL import ImageDraw,Image
img = Image.open("1.jpg")#为了让锚框显示更清晰,使用一张白色背景图片
draw = ImageDraw.Draw(img)
img_boxes = dets[:,0:4]#去掉分数列
color = [(0,0,0),(255,255,0),(255,0,255),(255,0,0),(0,0,255)]
i=0
for box in img_boxes:
    x1,y1,x2,y2 = box
    if i==5:
        i=0
    draw.rectangle([x1,y1,x2,y2],outline=color[i],width=4)
    i=i+1
img.show()

如下图:

黑色框跟黄色框重叠部分很多,所以去掉黑色框,其余的粉红和红色框这些交并比很小,就属于其他类别的锚框了。
最后就是将这些方法一起运用到目标检测即可。新的一年祝大家有新的收获,由于本人水平有限,文章中有错误之处,还望指正!

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

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

相关文章

【手写 Vue2.x 源码】第四十一篇 - 组件部分 - 生成组件的真实节点

一&#xff0c;前言 上篇&#xff0c;介绍了组件部分-组件的生命周期&#xff0c;主要涉及以下几部分&#xff1a; 本篇&#xff0c;组件部分-生成组件的真实节点&#xff1b; 二&#xff0c;生成组件的真实节点 1&#xff0c;前文回顾 前篇&#xff0c;在 createElement 方…

【剧前爆米花--爪哇岛寻宝】Java中有关异常类的详细讲解

作者&#xff1a;困了电视剧 专栏&#xff1a;《JavaSE语法与底层详解》 文章分布&#xff1a;这是一篇关于Java中异常类的文章&#xff0c;在本篇文章中详细讲解了异常的使用逻辑和底层的执行过程&#xff0c;如有疏漏&#xff0c;欢迎大佬指正&#xff01; 目录 异常的体系结…

<C++>哈希

文章目录1. unordered系列容器1.1 unordered_map1.1.1 unordered_map的文档介绍1.1.2 unordered_map的接口说明1.2 unordered_set2. 哈希概念3. 哈希冲突4. 哈希函数5. 哈希冲突解决5.1 闭散列5.1.1 线性探测5.1.2 二次探测5.2 开散列5.3 开散列与闭散列比较6. 模拟实现1. unor…

配置野火霸道V2环境

配置野火霸道V2环境野火霸道开发板学习笔记信息说明下载安装Keil5配置Keil以使用DAP下载器DAP下载器的使用使用串口下载程序安装USB转串口驱动CH340检查是否安装成功配置MCUISP软件野火霸道开发板学习笔记 信息说明 日期 : 2023-01-23开发板: 野火霸道V2芯片型号: STM32F103Z…

[Paper Reading] Towards Conversational Recommendation over Multi-Type Dialogs

[Paper Reading] Towards Conversational Recommendation over Multi-Type Dialogs 文章目录[Paper Reading] Towards Conversational Recommendation over Multi-Type Dialogs论文简介快速回顾论文&#xff08;借助scispace&#xff09;梳理一下文章内容&#xff08;参考百度N…

自动化将Gitee的仓库导入Github

自动化将Gitee的仓库导入Github准备工作获取方式gitee的授权码github授权码工具源码用法下载gitee所有仓库到本地下载并更新到github&#xff08;自动创建仓库&#xff09;写在最后本方法能实现自动创建仓库 脚本及用法放在文章最后了&#xff0c;需要的自取 转跳到结尾 准备工…

高性能 Java 框架。Solon v1.12.3 发布(春节前兮的最后更)

一个更现代感的 Java "生态型"应用开发框架&#xff1a;更快、更小、更自由。不是 Spring&#xff0c;没有 Servlet&#xff0c;也无关 JavaEE&#xff1b;新兴独立的开放生态 &#xff08;已有150来个生态插件&#xff09; 。主框架仅 0.1 MB。 相对于 Spring Boot…

计算正整数的阶乘math.factorial()

【小白从小学Python、C、Java】【计算机等级考试500强双证书】【Python-数据分析】计算正整数的阶乘math.factorial()[太阳]选择题请问math.factorial(3)的输出结果是?import mathprint("【执行】math.factorial(3):",math.factorial(3))print("【执行】math.f…

带你玩转Jetson之Deepstream简明教程(二)Deepstream是什么?干什么?有什么优势?

1.Deepstream是什么&#xff1f; Deepstream是Nvidia公司推出的一套基于开源视频流框架Gstreamer的一套库。其本身由多个.lib.so和.h构成&#xff0c;其支持语言包括了Python和Cpp两种主流语言。你可以在任何Python或者Cpp编译器、开发环境中引用库的API构建属于你自己的推理流…

【c++之于c的优化】

目录&#xff1a;前言关键字一、命名空间1.什么是命名空间2.如何使用命名空间3.如何自己创建命名空间4.为什么要使用命名空间5.命名空间起别名6.匿名命名空间二、缺省参数定义缺省参数类型注意事项三、函数重载定义函数重载的三种方式操作系统的区分方式四、引用定义引用特性使…

【4-网络八股扩展】北京大学TensorFlow2.0

课程地址&#xff1a;【北京大学】Tensorflow2.0_哔哩哔哩_bilibiliPython3.7和TensorFlow2.1六讲&#xff1a;神经网络计算&#xff1a;神经网络的计算过程&#xff0c;搭建第一个神经网络模型神经网络优化&#xff1a;神经网络的优化方法&#xff0c;掌握学习率、激活函数、损…

【LeetCode每日一题】【2023/1/24】1828. 统计一个圆中点的数目

文章目录1828. 统计一个圆中点的数目方法1&#xff1a;枚举1828. 统计一个圆中点的数目 LeetCode: 1828. 统计一个圆中点的数目 中等\color{#FFB800}{中等}中等 给你一个数组 points &#xff0c;其中 points[i] [x_i, y_i] &#xff0c;表示第 i 个点在二维平面上的坐标。多…

【算法面试】队列算法笔试面试全解(金三银四面试专栏启动)

&#x1f4eb;作者简介&#xff1a;小明java问道之路&#xff0c;专注于研究 Java/ Liunx内核/ C及汇编/计算机底层原理/源码&#xff0c;就职于大型金融公司后端高级工程师&#xff0c;擅长交易领域的高安全/可用/并发/性能的架构设计与演进、系统优化与稳定性建设。 &#x1…

02_gpio子系统

总结 驱动程序还想控制gpio 可以不用读写寄存器 直觉用gpio子系统开发的接口就能用了 轻松做输入输出 获取当前值 详细介绍 用设备树里的节点 gpio1 介绍 imx6ull.dtsi gpio1 记录了控制器相关的寄存器基地址 gpio1: gpio209c000 {compatible "fsl,imx6ul-gpio"…

三、利用迁移学习进行模型微调(Datawhale组队学习)

文章目录安装配置环境准备图像分类数据集迁移学习微调训练图像分类模型导入环境图像预处理载入图像分类数据集建立类别和索引号之间映射关系定义数据加载器查看一个batch的图像和标注可视化一个batch的图像和标注模型的构建与测试可视化常见的迁移学习训练方式训练配置模型训练…

过完2022,依然记得仰望星空

&#x1f57a;作者&#xff1a;一名普普通通的双非大二学生迷茫的启明星&#x1f383;专栏&#xff1a;《数据库》《C语言从0到1专栏》《数据结构》《C语言杂谈》目录 ​编辑 一.2022之初 二.2022年中 三.2022年末 四.展望2023 一.2022之初 想起一年前这个时候&#xff0c…

07_plantform平台总线

总结 /sys/bus/plantform 平台总线其实就是继承 06_自己创建xbus总线 有了更多的玩法 和自己创建的xbus总线一样 平台总线也有dev和drv 需要这两个进行匹配之后 进行porbe调用 plantform_device 结构体中直觉继承了 struc device lantform_driver 继承了driver 详细介绍 plan…

树(基础部分)

章节目录&#xff1a;一、二叉树1.1 为什么要使用树&#xff1f;1.2 树的常用术语1.3 二叉树概念1.4 二叉树应用二、顺序存储二叉树2.1 概述2.2 基本应用三、线索化二叉树3.1 问题引出3.2 概述3.3 基本应用四、结束语一、二叉树 1.1 为什么要使用树&#xff1f; 数组存储方式&…

MP-2平面烟雾气体传感器介绍

MP-2平面烟雾气体传感器简介MP-2烟雾检测气体传感器采用多层厚膜制造工艺&#xff0c;在微型Al2O3陶瓷基片的两面分别制作加热器和金属氧化物半导体气敏层&#xff0c;封装在金属壳体内。当环境空气中有被检测气体存在时传感器电导率发生变化&#xff0c;该气体的浓度越高&…

【数据库概论】3.1 SQL简述、数据定义和索引

第三章 关系数据库标准语言SQL 目录第三章 关系数据库标准语言SQL3.1 SQL概述3.1.1 产生与发展3.1.2 SQL的特点3.1.3 SQL的基本概念3.2 数据库实例3.3 数据定义3.3.1 模式的定义和删除3.2.2基本表的定义、删除和修改1.常见数据类型2.定义基本表3.修改基本表4.删除基本表5.模式和…