博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
前端面试的一道算法题(使用canvas解答)
阅读量:5826 次
发布时间:2019-06-18

本文共 4755 字,大约阅读时间需要 15 分钟。


据了解,现在前端面试也喜欢考算法题了。前几天去面试,果不其然的,面试官给我四道算法题,让我自己回去做。下面说一个跟前端有点相关并且有点趣的一道算法题。

题目:

平面上有若干个不特定的形状,如下图所示。请写程序求出物体的个数,以及每个不同物体的面积。

image.png

分析

想要知道有多少个图形,想到的就是先获取图片中的每一个像素点然后判获取像素点的背景颜色(RGBA)。想要获得图片中的每一个像素点,那就可以联想到使用h5的canvas。

如下:

  • 书写html标签。

对不你,你的浏览器不支持Canvas
  • js获取canvas对象

let ctxt = canvas.getContext('2d');
  • js创建image对象

let img = new Image;img.src = './image.png'; //图片路径img.onload = function(){}  //加载成功后的执行函数,之后的代码就写在其中
  • 创建存储图片像素点的二维数组

let coordinates = [];for(let i=0; i<200; i++){       coordinates[i] = [];}
  • 获取像素点,也就是使用getimagedata方法。

ctxt.drawImage(img, 0, 0);  //将图片画如canvaslet data = ctxt.getImageData(0, 0, 350, 200).data;//读取整张图片的像素。
  • 将像素存入二维数组

let x=0,y=0;  //二维数组的行和列, x:列  y:行for(let i =0,len = data.length; i
= 350){ x = 0; y++; }}
  • 目前代码如下:

(function(){        let ctxt = canvas.getContext('2d');        let img = new Image;        let coordinates = [];        let h = 200,            w = 350;        for(let i=0; i<200; i++){            coordinates[i] = [];        }        img.src = './image.png'; //图片路径        img.onload = function(){            ctxt.drawImage(img, 0, 0);            let data = ctxt.getImageData(0, 0, 350, 200).data;//读取整张图片的像素。            let x=0,y=0;            for(let i =0,len = data.length; i
= 350){ x = 0; y++; } } console.log(coordinates); } })();
  • 如图:

image.png

  • 构成类似如下二维数组:

    0,0,0,0,0,0,0,0,0,0,0,0 0,0,1,1,1,0,0,0,0,0,0,0 0,1,1,1,1,0,0,0,0,0,0,0 0,1,1,1,0,0,0,1,1,1,1,0 0,0,0,0,0,0,1,1,1,0,0,0 0,0,0,0,0,0,1,1,1,0,0,0 0,0,0,0,0,0,0,0,0,0,0,0

那么我们就只需要知道二维数组中这种连续为1的块有多少个就知道了图片中形状有多少个,并且块中有多少个1,那么这个块的面积就是1的个数。

递归回溯算法

//计算连续的面积和个数const linkSum = (i,j,num)=>{        //走过的路就置0      coordinates[i][j] = 0;      num++;      //向上      if((i+1 < h) && coordinates[i+1][j] == 1){        num = linkSum(i+1 , j , num);      }      //向下      if((j+1 < w) && coordinates[i][j+1] == 1){        num = linkSum(i , j+1 , num);      }      //向左      if((i-1 >= 0) && coordinates[i-1][j] == 1){        num = linkSum(i-1 , j , num);      }      //向右    if((j-1 >= 0) && coordinates[i][j-1] == 1){        num = linkSum(i , j-1 , num);    }    return num;}

不熟悉的,直接百度就好,这里就不多说了,其实代码就反应了很多信息。

使用算法,统计并计算出结果。

const getCountAndArea = () =>{    let sum = [];    let count = 0;    for(let i = 0; i < h; i++)  //遍历二维数组    {      for(let j = 0; j < w; j++)      {       //连续1的个数       if(coordinates[i][j] == 1)       {        let buf = 0;  //连续1的个数        buf = linkSum(i,j,buf);        count++;  //形状的总数        sum.push({            index: count,   //第几个形状            area: buf         //形状的面积        });       }      }    }    return {        count,        sum    };}

最后的代码

(function(){        let ctxt = canvas.getContext('2d');        let img = new Image;        let coordinates = [];        let h = 200,            w = 350;        for(let i=0; i<200; i++){            coordinates[i] = [];        }        img.src = './image.png'; //图片路径        img.onload = function(){            ctxt.drawImage(img, 0, 0);            let data = ctxt.getImageData(0, 0, 350, 200).data;//读取整张图片的像素。            let x=0,y=0;            for(let i =0,len = data.length; i
= 350){ x = 0; y++; } } // console.log(coordinates); let rst = getCountAndArea(); // console.log(rst); console.log('个数: ' + rst.count); for(let i=0; i
{ let sum = []; let count = 0; for(let i = 0; i < h; i++) { for(let j = 0; j < w; j++) { //连续1的个数 if(coordinates[i][j] == 1) { let buf = 0; buf = linkSum(i,j,buf); count++; sum.push({ index: count, area: buf }); } } } return { count, sum }; } //计算连续的面积和个数 const linkSum = (i,j,num)=>{ //走过的路就置0 coordinates[i][j] = 0; num++; //向上 if((i+1 < h) && coordinates[i+1][j] == 1){ num = linkSum(i+1 , j , num); } //向下 if((j+1 < w) && coordinates[i][j+1] == 1){ num = linkSum(i , j+1 , num); } //向左 if((i-1 >= 0) && coordinates[i-1][j] == 1){ num = linkSum(i-1 , j , num); } //向右 if((j-1 >= 0) && coordinates[i][j-1] == 1){ num = linkSum(i , j-1 , num); } return num; } })();

运行的结果:

image.png

转载地址:http://xlsdx.baihongyu.com/

你可能感兴趣的文章
直播源码开发视频直播平台,不得不了解的流程
查看>>
Ubuntu上的pycrypto给出了编译器错误
查看>>
聊聊flink的RestClientConfiguration
查看>>
在CentOS上搭建git仓库服务器以及mac端进行克隆和提交到远程git仓库
查看>>
測試文章
查看>>
Flex很难?一文就足够了
查看>>
【BATJ面试必会】JAVA面试到底需要掌握什么?【上】
查看>>
CollabNet_Subversion小结
查看>>
mysql定时备份自动上传
查看>>
Linux 高可用集群解决方案
查看>>
17岁时少年决定把海洋洗干净,现在21岁的他做到了
查看>>
linux 启动oracle
查看>>
《写给大忙人看的java se 8》笔记
查看>>
倒计时:计算时间差
查看>>
Linux/windows P2V VMWare ESXi
查看>>
Windows XP倒计时到底意味着什么?
查看>>
tomcat一步步实现反向代理、负载均衡、内存复制
查看>>
运维工程师在干什么学些什么?【致菜鸟】
查看>>
Linux中iptables详解
查看>>
java中回调函数以及关于包装类的Demo
查看>>