IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 人工智能 -> OpenCV基于NCC多角度模板匹配 -> 正文阅读

[人工智能]OpenCV基于NCC多角度模板匹配

1. 废话不多说,先看测试效果图

a)模板原图:

b) 逆时针旋转30°:

?c)顺时针旋转30°:

2. 下面分享一下开发过程

a) 为了提升搜索匹配速度,需要构建金字塔模型,金字塔层数不宜过多,一般2~3层,以实际图像质量为依据,代码实现方式:

    //对模板图像和待检测图像分别进行图像金字塔下采样
            for (int i = 0; i < numLevels; i++)
            {
                Cv2.PyrDown(src, src, new Size(src.Cols / 2, src.Rows / 2));
                Cv2.PyrDown(model, model, new Size(model.Cols / 2, model.Rows / 2));
            }
          //为了提升速度,直接上采样到最底层
                for (int i = 0; i < numLevels; i++)
                {
                    Cv2.PyrUp(src, src, new Size(src.Cols * 2,
                                src.Rows * 2));//下一层,放大2倍
                    Cv2.PyrUp(model, model, new Size(model.Cols * 2,
                         model.Rows * 2));//下一层,放大2倍
                }

b) 多角度匹配时需要将图像按照一定的角度步长循环进行搜索,旋转的图像可以是当前实时图也可以是模板图,不过基于速度考虑选择了模板来作为旋转,同时模板图像也是固定不变的,代码实现方式:

   /// <summary>
        /// 图像旋转,并获旋转后的图像边界旋转矩形
        /// </summary>
        /// <param name="image"></param>
        /// <param name="angle"></param>
        /// <param name="imgBounding"></param>
        /// <returns></returns>
        public static Mat ImageRotate(Mat image, double angle,ref RotatedRect imgBounding,ref Mat maskMat)
        {
            Mat newImg = new Mat();
            Point2f pt = new Point2f((float)image.Cols / 2, (float)image.Rows / 2);
            Mat M = Cv2.GetRotationMatrix2D(pt, -angle, 1.0);
            var mIndex = M.GetGenericIndexer<double>();

            double cos = Math.Abs(mIndex[0, 0]);
            double sin = Math.Abs(mIndex[0, 1]);
            int nW = (int)((image.Height * sin) + (image.Width * cos));
            int nH = (int)((image.Height * cos) + (image.Width * sin));
            mIndex[0, 2] += (nW / 2) - pt.X;
            mIndex[1, 2] += (nH / 2) - pt.Y;

            Cv2.WarpAffine(image, newImg, M, new CVSize(nW, nH));
            //获取图像边界旋转矩形
            Rect rect = new CVRect(0, 0, image.Width, image.Height);
            Point2f[] srcPoint2Fs = new Point2f[4]
                {
                    new Point2f(rect.Left,rect.Top),
                     new Point2f (rect.Right,rect.Top),
                       new Point2f (rect.Right,rect.Bottom),
                          new Point2f (rect.Left,rect.Bottom)
                };

           Point2f[] boundaryPoints = new Point2f[4];
            var A = M.Get<double>(0, 0);
            var B = M.Get<double>(0, 1);
            var C = M.Get<double>(0, 2);    //Tx
            var D = M.Get<double>(1, 0);
            var E = M.Get<double>(1, 1);
            var F = M.Get<double>(1, 2);    //Ty

            for(int i=0;i<4;i++)
            {
                boundaryPoints[i].X = (float)((A * srcPoint2Fs[i].X) + (B * srcPoint2Fs[i].Y) + C);
                boundaryPoints[i].Y = (float)((D * srcPoint2Fs[i].X) + (E * srcPoint2Fs[i].Y) + F);
                if (boundaryPoints[i].X < 0)
                    boundaryPoints[i].X = 0;
                else if (boundaryPoints[i].X > nW)
                    boundaryPoints[i].X = nW;
                if (boundaryPoints[i].Y < 0)
                    boundaryPoints[i].Y = 0;
                else if (boundaryPoints[i].Y > nH)
                    boundaryPoints[i].Y = nH;
            }

            Point2f cenP = new Point2f((boundaryPoints[0].X + boundaryPoints[2].X) / 2,
                                         (boundaryPoints[0].Y + boundaryPoints[2].Y) / 2);
            double ang = angle;

            double width1=Math.Sqrt(Math.Pow(boundaryPoints[0].X- boundaryPoints[1].X ,2)+
                Math.Pow(boundaryPoints[0].Y - boundaryPoints[1].Y,2));
            double width2 = Math.Sqrt(Math.Pow(boundaryPoints[0].X - boundaryPoints[3].X, 2) +
                Math.Pow(boundaryPoints[0].Y - boundaryPoints[3].Y, 2));

            //double width = width1 > width2 ? width1 : width2;

            //double height = width1 > width2 ? width2 : width1;

            imgBounding = new RotatedRect(cenP, new Size2f(width1, width2), (float)ang);

            Mat mask = new Mat(newImg.Size(), MatType.CV_8UC3, Scalar.Black);
            mask.DrawRotatedRect(imgBounding, Scalar.White, 1);
            Cv2.FloodFill(mask, new CVPoint(imgBounding.Center.X, imgBounding.Center.Y), Scalar.White);
         //   mask.ConvertTo(mask, MatType.CV_8UC1);
            //mask.CopyTo(maskMat);
            //掩膜复制给maskMat       
            Cv2.CvtColor(mask, maskMat, ColorConversionCodes.BGR2GRAY);

            Mat _maskRoI = new Mat();
            Cv2.CvtColor(mask, _maskRoI, ColorConversionCodes.BGR2GRAY);
            Mat buf = new Mat();

            //# 黑白反转
            Cv2.BitwiseNot(_maskRoI, buf);
            Mat dst = new Mat();
            Cv2.BitwiseAnd(newImg, newImg, dst, _maskRoI);
            //Mat dst2 = new Mat();
            //Cv2.BitwiseOr(buf, dst, dst2);
            return dst;
      
        }

c) 旋转后的模板图像会产生无效区域,如果将他们也参与匹配计算,那么相似度会有所下降,影响匹配效果,因此,我们需要加入mask掩膜图像,屏蔽掉无效区域;上面的代码同时也计算出旋转后的无效区域同时将它作为mask。

 newtemplate = ImageRotate(model, start + step * i, ref rotatedRect, ref mask);
                        if (newtemplate.Width > src.Width || newtemplate.Height > src.Height)
                            continue;
                        Cv2.MatchTemplate(src, newtemplate, result, matchMode, mask);
                        Cv2.MinMaxLoc(result, out double minval, out double maxval, out CVPoint minloc, out CVPoint maxloc, new Mat());

d)搜索的角度步长可以根据需要的精度,在上采样的过程逐渐增加,如果底层需要的精度为1,在下采样的过程可以指数逐级递增,毕竟在实际的使用过程我们还是需要它们以较快的速度完成整个搜索过程的,上采样的过程主要是逐级提升检测精度,位置和角度。以上一层搜索到的点位为中心,左右一倍模板的宽度,上下一倍模板的高度来作为新的搜索区域,逐级来完成上采样。

     //开始上采样
            Rect cropRegion = new CVRect(0, 0, 0, 0);
            //for (int j = numLevels - 1; j >= 0; j--)
            {
                //为了提升速度,直接上采样到最底层
                for (int i = 0; i < numLevels; i++)
                {
                    Cv2.PyrUp(src, src, new Size(src.Cols * 2,
                                src.Rows * 2));//下一层,放大2倍
                    Cv2.PyrUp(model, model, new Size(model.Cols * 2,
                         model.Rows * 2));//下一层,放大2倍
                }

                location.X *= (int)Math.Pow(2, numLevels);
                location.Y *= (int)Math.Pow(2, numLevels);
                modelrrect = new RotatedRect(new Point2f((float)(modelrrect.Center.X * Math.Pow(2, numLevels)),//下一层,放大2倍
                                (float)(modelrrect.Center.Y * Math.Pow(2, numLevels))),
                                new Size2f(modelrrect.Size.Width * Math.Pow(2, numLevels),
                                modelrrect.Size.Height * Math.Pow(2, numLevels)), 0);

                CVPoint cenP = new CVPoint(location.X + modelrrect.Center.X,
                               location.Y + modelrrect.Center.Y);//投影到下一层的匹配点位中心

                int startX = cenP.X - model.Width;
                int startY = cenP.Y - model.Height;
                int endX = cenP.X + model.Width;
                int endY = cenP.Y + model.Height;
                cropRegion = new CVRect(startX, startY, endX - startX, endY - startY);
                cropRegion = cropRegion.Intersect(new CVRect(0, 0, src.Width, src.Height));
                Mat newSrc = MatExtension.Crop_Mask_Mat(src, cropRegion);
                //每下一层金字塔,角度间隔减少2倍
                step = 1;
                //角度开始和范围           
                range = 20;
                start = angle - 10;
                bool testFlag = false;
                for (int k = 0; k <= (int)range / step; k++)
                {

                    newtemplate = ImageRotate(model, start + step * k, ref rotatedRect, ref mask);
                    if (newtemplate.Width > newSrc.Width || newtemplate.Height > newSrc.Height)
                        continue;
                    Cv2.MatchTemplate(newSrc, newtemplate, result, TemplateMatchModes.CCoeffNormed);
                    Cv2.MinMaxLoc(result, out double minval, out double maxval,
                                                out CVPoint minloc, out CVPoint maxloc, new Mat());

                    if (maxval > temp)
                    {
                        //局部坐标
                        location.X = maxloc.X;
                        location.Y = maxloc.Y;
                        temp = maxval;
                        angle = start + step * k;
                        //局部坐标
                        modelrrect = rotatedRect;
                        testFlag = true;
                    }
                }
                if (testFlag)
                {
                    //局部坐标--》整体坐标
                    location.X += cropRegion.X;
                    location.Y += cropRegion.Y;
                }

            }

e)最后就是将测试的结果绘制出来,作为OVERLAY显示在图像上面

可参考代码:

                if (result.Score<= 0)
                {
                    currvisiontool.DrawText(new TextEx("模板匹配失败!") { x = 1000, y = 10, brush = new SolidBrush(Color.Red) });

                    currvisiontool.AddTextBuffer(new TextEx("模板匹配失败!") { x = 1000, y = 10, brush = new SolidBrush(Color.Red) });

                    stuModelMatchData.runFlag = false;
                    return;
                }
                currvisiontool.DrawText(new TextEx("得分:" + result.Score.ToString("f3")));
                currvisiontool.AddTextBuffer(new TextEx("得分:" + result.Score.ToString("f3")));

                currvisiontool.DrawText(new TextEx("偏转角度:" + result.T.ToString("f3")) { x = 10, y = 100 });
                currvisiontool.AddTextBuffer(new TextEx("偏转角度:" + result.T.ToString("f3")) { x = 10, y = 100 });

                currvisiontool.DrawText(new TextEx(string.Format("匹配点位X:{0},Y:{1}", result.X.ToString("f3"),
                   result.Y.ToString("f3")))
                { x = 10, y = 200 });
                currvisiontool.AddTextBuffer(new TextEx(string.Format("匹配点位X:{0},Y:{1}", result.X.ToString("f3"),
                   result.Y.ToString("f3")))
                { x = 10, y = 200 });

3. 过程效果演示图

a) 创建模板:选择具有稳定特征,和背景有一定的对比区分的图像区域来制作模板

?b)模板旋转后的图像

? c)旋转后对应的掩膜图像

?4. NCC多角度模板匹配核心代码段如下:

  /// <summary>
        /// 创建NCC模板
        /// </summary>
        /// <param name="src"></param>
        /// <param name="RegionaRect"></param>
        /// <returns></returns>
        public Mat CreateTemplateNcc(Mat src, Rect RegionaRect,ref double modelX,ref double modelY,ref Mat modelGrayMat)
        {
            Mat ModelMat = MatExtension.Crop_Mask_Mat(src, RegionaRect);
            Mat Morphological = Morphological_Proces.MorphologyEx(ModelMat, MorphShapes.Rect,
                       new OpenCvSharp.Size(5, 5), MorphTypes.Open);
            Mat dst = Morphological.CvtColor(ColorConversionCodes.GRAY2BGR);
            Morphological.CopyTo(modelGrayMat);           
            CVPoint cenP = new CVPoint(RegionaRect.Width / 2,
             RegionaRect.Height / 2);
            modelX = RegionaRect.X + RegionaRect.Width / 2;
            modelY = RegionaRect.Y + RegionaRect.Height / 2;
            Console.WriteLine(string.Format("模板中心点位:x:{0},y:{1}", RegionaRect.X + RegionaRect.Width / 2,
                  RegionaRect.Y + RegionaRect.Height / 2));
            dst.drawCross(cenP, Scalar.Red, 20, 2);
            return dst;

        }
   /// <summary>
        /// 多角度模板匹配方法(NCC:归一化相关系数)
        /// </summary>
        /// <param name="srcImage">待匹配图像</param>
        /// <param name="modelImage">模板图像</param>
        /// <param name="angleStart">起始角度</param>
        /// <param name="angleRange">角度范围</param>
        /// <param name="angleStep">角度步长</param>
        /// <param name="numLevels">金字塔层级</param>
        /// <param name="thresScore">得分阈值</param>
        /// <returns></returns>
        private NccTemplateMatchResult TempalteMatchNcc(Mat srcImage, Mat modelImage,
                      double angleStart, double angleRange, double angleStep,
                           int numLevels, double thresScore, int nccMethod=5)
        {
            double step = angleRange / ((angleRange / angleStep) / 10); //角度间隔
            double start = angleStart;                                      //起始角度
            double range = angleRange;                                        //角度范围

            //定义图片匹配所需要的参数
            //int resultCols = srcImage.Cols - modelImage.Cols + 1;
            //int resultRows = srcImage.Rows - modelImage.Cols + 1;
            //Mat result = new Mat(resultCols, resultRows, MatType.CV_8U);
            Mat result = new Mat();
            Mat src = new Mat();
            Mat model = new Mat();
            srcImage.CopyTo(src);
            modelImage.CopyTo(model);

            //Morphology
            src = Morphological_Proces.MorphologyEx(src, MorphShapes.Rect,
             new OpenCvSharp.Size(5, 5), MorphTypes.Open);
         
            //对模板图像和待检测图像分别进行图像金字塔下采样
            for (int i = 0; i < numLevels; i++)
            {
                Cv2.PyrDown(src, src, new Size(src.Cols / 2, src.Rows / 2));
                Cv2.PyrDown(model, model, new Size(model.Cols / 2, model.Rows / 2));
            }

            TemplateMatchModes matchMode = TemplateMatchModes.CCoeffNormed;
            switch (nccMethod)
            {
                case 0:
                    matchMode = TemplateMatchModes.SqDiff;
                    break;
                case 1:
                    matchMode = TemplateMatchModes.SqDiffNormed;
                    break;
                case 2:
                    matchMode = TemplateMatchModes.CCorr;
                    break;
                case 3:
                    matchMode = TemplateMatchModes.CCorrNormed;
                    break;
                case 4:
                    matchMode = TemplateMatchModes.CCoeff;
                    break;
                case 5:
                    matchMode = TemplateMatchModes.CCoeffNormed;
                    break;
            }

            //在没有旋转的情况下进行第一次匹配
            Cv2.MatchTemplate(src, model, result, matchMode);
            Cv2.MinMaxLoc(result, out double minVal, out double maxVal, out CVPoint minLoc, out CVPoint maxLoc, new Mat());

            CVPoint location = maxLoc;
            double temp = maxVal;
            if (nccMethod == 0 || nccMethod == 1)
            {
                temp = minVal;
                location = minLoc;
            }
              
            double angle = 0;

            Mat newtemplate;//旋转后新模板
            Mat mask = new Mat();//掩膜
            RotatedRect rotatedRect = new RotatedRect();
            //模板矩形
            RotatedRect modelrrect = new RotatedRect(new Point2f(model.Width / 2,
                              model.Height / 2), new Size2f(model.Width, model.Height), 0);



            //以最佳匹配点左右十倍角度步长进行循环匹配,直到角度步长小于参数角度步长
            if (nccMethod == 0 || nccMethod == 1)
            {
                do
                {

                    for (int i = 0; i <= (int)range / step; i++)
                    {

                        newtemplate = ImageRotate(model, start + step * i, ref rotatedRect, ref mask);
                        Cv2.MatchTemplate(src, newtemplate, result, matchMode, mask);              
                        Cv2.MinMaxLoc(result, out double minval, out double maxval, out CVPoint minloc, out CVPoint maxloc, new Mat());
                        if (minval < temp)//平方差值方式,值越小越好
                        {
                            location = minloc;
                            temp = minval;
                            angle = start + step * i;
                            modelrrect = rotatedRect;
                        }
                    }
                    range = step * 2;
                    start = angle - step;
                    step = step / 10;
                } while (step > angleStep);

                Console.WriteLine(string.Format("x:{0},y:{1},angle:{2},score:{3}",
                    location.X * Math.Pow(2, numLevels),
                    location.Y * Math.Pow(2, numLevels),
                    angle,
                    temp
                    ));

                //该方式计算的坐标精度无法满足,需要进行金字塔上采样
                Console.WriteLine(string.Format("x:{0},y:{1},angle:{2},score:{3}",
               modelrrect.Center.X * Math.Pow(2, numLevels) + location.X * Math.Pow(2, numLevels),
               modelrrect.Center.Y * Math.Pow(2, numLevels) + location.Y * Math.Pow(2, numLevels),
                angle,
                temp
                ));

                if (temp < thresScore)
                {
                    return new NccTemplateMatchResult(location.X * Math.Pow(2, numLevels), location.Y * Math.Pow(2, numLevels), angle, temp)
                    {
                        rotatedRect = new RotatedRect(new Point2f((float)((modelrrect.Center.X + location.X) * Math.Pow(2, numLevels)),
                      (float)((modelrrect.Center.Y + location.Y) * Math.Pow(2, numLevels))),
                      new Size2f(modelrrect.Size.Width * Math.Pow(2, numLevels),
                           modelrrect.Size.Height * Math.Pow(2, numLevels)),
                      (float)angle)
                    };
               
                }
            }
            else
            {
                //double minval;  double maxval; CVPoint minloc; CVPoint maxloc;
                do
                {

                    for (int i = 0; i <= (int)range / step; i++)
                    {

                        newtemplate = ImageRotate(model, start + step * i, ref rotatedRect, ref mask);
                        if (newtemplate.Width > src.Width || newtemplate.Height > src.Height)
                            continue;
                        Cv2.MatchTemplate(src, newtemplate, result, matchMode, mask);
                        Cv2.MinMaxLoc(result, out double minval, out double maxval, out CVPoint minloc, out CVPoint maxloc, new Mat());

                        if (double.IsInfinity(maxval))
                        {
                            Cv2.MatchTemplate(src, newtemplate, result, TemplateMatchModes.CCorrNormed, mask);
                            Cv2.MinMaxLoc(result, out minval, out maxval, out minloc, out maxloc, new Mat());

                        }
                        if (maxval > temp)
                        {
                            location = maxloc;
                            temp = maxval;
                            angle = start + step * i;
                            modelrrect = rotatedRect;
                        }
                    }
                range = step * 2;
                start = angle - step;
                step = step / 10;
            } while (step > angleStep) ;
            #region-------暂不使用---------
            //开始上采样
            //Rect cropRegion = new CVRect(0, 0, 0, 0);
            for (int j = numLevels - 1; j >= 0; j--)
            //{

            //    //为了提升速度,直接上采样到最底层
            //    for (int i = 0; i < numLevels; i++)
            //    {
            //        Cv2.PyrUp(src, src, new Size(src.Cols * 2,
            //                    src.Rows * 2));//下一层,放大2倍
            //        Cv2.PyrUp(model, model, new Size(model.Cols * 2,
            //             model.Rows * 2));//下一层,放大2倍
            //    }

            //    location.X *= (int)Math.Pow(2, numLevels);
            //    location.Y *= (int)Math.Pow(2, numLevels);
            //    modelrrect = new RotatedRect(new Point2f((float)(modelrrect.Center.X * Math.Pow(2, numLevels)),//下一层,放大2倍
            //                 (float)(modelrrect.Center.Y * Math.Pow(2, numLevels))),
            //                 new Size2f(modelrrect.Size.Width * Math.Pow(2, numLevels),
            //                 modelrrect.Size.Height * Math.Pow(2, numLevels)), 0);

            //    CVPoint cenP = new CVPoint(location.X + modelrrect.Center.X,
            //                   location.Y + modelrrect.Center.Y);//投影到下一层的匹配点位中心

            //    int startX = cenP.X - model.Width;
            //    int startY = cenP.Y - model.Height;
            //    int endX = cenP.X + model.Width;
            //    int endY = cenP.Y + model.Height;
            //    cropRegion = new CVRect(startX, startY, endX - startX, endY - startY);
            //    cropRegion = cropRegion.Intersect(new CVRect(0, 0, src.Width, src.Height));
            //    Mat newSrc = MatExtension.Crop_Mask_Mat(src, cropRegion);
            //    //每下一层金字塔,角度间隔减少2倍
            //    step = 2;
            //    //角度开始和范围           
            //    range = 20;
            //    start = angle - 10;
            //    bool testFlag = false;
            //    for (int k = 0; k <= (int)range / step; k++)
            //    {

            //        newtemplate = ImageRotate(model, start + step * k, ref rotatedRect, ref mask);
            //        if (newtemplate.Width > newSrc.Width || newtemplate.Height > newSrc.Height)
            //            continue;
            //        Cv2.MatchTemplate(newSrc, newtemplate, result, TemplateMatchModes.CCoeffNormed, mask);
            //        Cv2.MinMaxLoc(result, out double minval, out double maxval,
            //                                    out CVPoint minloc, out CVPoint maxloc, new Mat());
            //        if (double.IsInfinity(maxval))
            //        {
            //            Cv2.MatchTemplate(src, newtemplate, result, TemplateMatchModes.CCorrNormed, mask);
            //            Cv2.MinMaxLoc(result, out minval, out maxval, out minloc, out maxloc, new Mat());

            //        }
            //        if (maxval > temp)
            //        {
            //            //局部坐标
            //            location.X = maxloc.X;
            //            location.Y = maxloc.Y;
            //            temp = maxval;
            //            angle = start + step * k;
            //            //局部坐标
            //            modelrrect = rotatedRect;
            //            testFlag = true;
            //        }
            //    }
            //    if (testFlag)
            //    {
            //        //局部坐标--》整体坐标
            //        location.X += cropRegion.X;
            //        location.Y += cropRegion.Y;
            //    }

            //}
            #endregion

            //该方式计算的坐标精度无法满足,需要进行金字塔上采样
            Console.WriteLine(string.Format("x:{0},y:{1},angle:{2},score:{3}",
               (modelrrect.Center.X  + location.X) * Math.Pow(2, numLevels),
               (modelrrect.Center.Y + location.Y) * Math.Pow(2, numLevels),
                angle,
                temp
                ));

                if (temp > thresScore)
                {

                    return new NccTemplateMatchResult((modelrrect.Center.X + location.X) * Math.Pow(2, numLevels),
                       (modelrrect.Center.Y + location.Y) * Math.Pow(2, numLevels), angle, temp)
                    {
                        rotatedRect = new RotatedRect(new Point2f((float)((modelrrect.Center.X+ location.X) * Math.Pow(2, numLevels)),
                      (float)((modelrrect.Center.Y  + location.Y) * Math.Pow(2, numLevels))),
                      new Size2f(modelrrect.Size.Width * Math.Pow(2, numLevels),
                           modelrrect.Size.Height * Math.Pow(2, numLevels)),
                      (float)angle)
                    };
                };
           
            }
            return new NccTemplateMatchResult();
        }

5. 总结

? ? 好了,到这步基本完成了整个过程,当然还有一些需要优化的过程,比如:重叠覆盖,亚像素,以及搜索速度在当前金字塔的基础上进一步优化等,有兴趣的朋友以后可以一起探讨一下,网上C#的资料较少,当前也是参考了大佬们C++和python的技术资料,希望能做一个较好版本的C#解决方案。

ps: 学习如逆水行舟,不进则退!

  人工智能 最新文章
2022吴恩达机器学习课程——第二课(神经网
第十五章 规则学习
FixMatch: Simplifying Semi-Supervised Le
数据挖掘Java——Kmeans算法的实现
大脑皮层的分割方法
【翻译】GPT-3是如何工作的
论文笔记:TEACHTEXT: CrossModal Generaliz
python从零学(六)
详解Python 3.x 导入(import)
【答读者问27】backtrader不支持最新版本的
上一篇文章      下一篇文章      查看所有文章
加:2022-09-30 00:52:59  更:2022-09-30 00:57:15 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年12日历 -2024/12/28 3:22:00-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码
数据统计