首頁 > web前端 > js教程 > Js面試中容易遇到的演算法

Js面試中容易遇到的演算法

php中世界最好的语言
發布: 2020-07-30 17:13:22
原創
4087 人瀏覽過

Js面試中容易遇到的演算法

這次帶給大家Js面試中容易遇到的演算法,以下就是實戰案例,一起來看一下。

推薦相關文章2020年最全js面試題整理(最新)

素數

Q:你會如何驗證一個質數?

A:一個質數只能被它自己和1整除。所以,我將運行一個while循環並加1。 (看程式碼範例,如果你無法理解,那這不是你的菜。先回去學習javaScript基礎知識然後再回來吧。)

##方法1

1

2

3

4

5

6

7

8

9

10

11

12

13

function isPrime(n){

 var pisor = 2;

 while (n > pisor){

 if(n % pisor == 0){

  return false; 

 }

 else

  pisor++;

 }

 return true;

}

isPrime(137); // = true

isPrime(237); // = false

登入後複製
Q:你能做得更好嗎?

A:可以。除數一次增加1個。在3之後我可以增加2.如果一個數可以被任何偶數整除,它將被2整除。

補充:如果你不需要把除數增加到這個數。你可以更早停止。讓我在下面的步驟中解釋一下(如果需要可以多讀幾遍)

第一步,任何數字都不能被大於它的一半的數字整除。例如,13將永遠不能被7,8,9整除…它甚至可以是偶數的一半。例如,16將被8整除,但永遠不會被9,10,11,12 ……

結論:一個數字將永遠不能被一個大於其一半數值的數字整除。所以,我們可以少循環50%。

第二步,現在,如果一個數字不能被3整除。 (如果它可被3整除,那麼它就不是質數)。然後,它不可以被大於其值1/3的任何數整除。例如,35不能被3整除。因此,它永遠不會被大於35/3的數整除,永遠不能被12, 13, 14整除…如果你取一個像36一樣的偶數,它將永遠不能被13, 14, 15整除。

結論: 一個數字可以被其數值的三分之一整除。

第三步,例如,你有一個數字127。127不能被2整除,因此你最多應該檢查63.5。其次,127不能被3整除。因此,您將檢查到127/3大約42。它不能被5整除,除數應該小於127/5大約25,而不是7。那麼,我們該在哪裡停下來?

結論: 除數將小於math.sqrt(N)

方法2

如果你無法理解也不用擔心,別管它。如果那你不是一個研究人員就沒關係。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

function isPrime(n)

{

 var pisor = 3, 

  limit = Math.sqrt(n);

 //check simple cases

 if (n == 2 || n == 3)

  return true;

 if (n % 2 == 0)

  return false;

 while (pisor <= limit)

 {

 if (n % pisor == 0)

  return false;

 else

  pisor += 2;

 }

 return true;

}

isPrime(137); // = true

isPrime(237); // = false

登入後複製

素數因子

Q:如何求出一個數的所有質數因子?

A:執行一個while迴圈。開始除以2,如果不能整除,記錄這個除數,直到完成。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

function primeFactors(n){

 var factors = [],

  pisor = 2;

 while(n>2){

 if(n % pisor == 0){

  factors.push(pisor); 

  n= n/ pisor;

 }

 else{

  pisor++;

 }  

 }

 return factors;

}

 primeFactors(69); // = [3, 23]

登入後複製
Q:運行時間複雜度是多少?你能做得更好嗎?

A:O(n)。可以將除數從3開始,累加2。因為,如果一個數被任何偶數整除,它將被2整除。因此,你不需要除以偶數。此外,你不會有一個大於其價值一半的因素。如果你想讓它變得複雜,那就用第一題的補充概念吧。

Fibonacci(斐波那契)

#Q:如何取得第n個斐波納契數字?

A: 我建立一個陣列並從迭代開始。

斐波那契系列是初學者的最受歡迎的面試問題之一。所以,你必須學習這一個。

方法1

1

2

3

4

5

6

7

8

9

function fibonacci(n){

 var fibo = [0, 1];

 if (n <= 2) return 1;

 for (var i = 2; i <=n; i++ ){

 fibo[i] = fibo[i-1]+fibo[i-2];

 }

 return fibo[n];

}

fibonacci(12); // = 144

登入後複製

Q: 運行時間複雜度是多少?

A: O(n);

Q: 你能讓它

遞迴嗎?

方法2

1

2

3

4

5

6

7

8

function fibonacci(n){

 if(n < =1) {

  return n;

 } else {

  return fibonacci(n-1) + fibonacci (n-2);

 }

}

fibonacci(12); // = 144

登入後複製

Q: 運行時間複雜度是多少?

A: O(2n);關於時間複雜度的細節

最大公約數

Q: 你會如何找到兩個數字的最大公約數?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

function greatestCommonpisor(a, b){

 var pisor = 2,

  greatestpisor = 1;

 //if u pass a -ve number this will not work. fix it dude!!

 if (a < 2 || b < 2)

  return 1;

 while(a >= pisor && b >= pisor){

 if(a %pisor == 0 && b% pisor ==0){

  greatestpisor = pisor;  

 }

 pisor++;

 }

 return greatestpisor;

}

greatestCommonpisor(14, 21); // 7

greatestCommonpisor(69, 169); // = 1

登入後複製

演算法範式

很抱歉。我也无法解释它。 因为我自己80%的情况下都不能理解它。 我的算法分析教练告诉我这个,又从课堂笔记偷走(我是一个好学生,顺便说一句!)

1

2

3

4

5

6

function greatestCommonpisor(a, b){

 if(b == 0)

  return a;

 else 

  return greatestCommonpisor(b, a%b);

}

登入後複製

注意:用你的大脑来理解它。

去重

Q:你将如何从数组中删除重复的成员?
A: 执行一个循环,并保存一个对象/关联数组。如果我第一次找到一个元素,我会设置它的值为真(这将告诉我元素添加一次)。如果我在对象中找到这个元素,我不会将它插入到返回数组中。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

function removeDuplicate(arr){

 var exists ={},

  outArr = [], 

  elm;

 for(var i =0; i<arr.length; i++){

 elm = arr[i];

 if(!exists[elm]){

  outArr.push(elm);

  exists[elm] = true;

 }

 }

 return outArr;

}

removeDuplicate([1,3,3,3,1,5,6,7,8,1]); // = [1, 3, 5, 6, 7, 8]

登入後複製

合并两个排序的数组

Q: 怎样合并两个已排序数组?
A: 我将为每个数组保留一个指针(看代码,并注意这个)。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

function mergeSortedArray(a, b){

 var merged = [],

   aElm = a[0],

   bElm = b[0],

   i = 1,

   j = 1;

 if(a.length ==0)

  return b;

 if(b.length ==0)

  return a;

 /*

 if aElm or bElm exists we will insert to merged array

 (will go inside while loop)

  to insert: aElm exists and bElm doesn&#39;t exists

       or both exists and aElm < bElm

  this is the critical part of the example     

 */

 while(aElm || bElm){

  if((aElm && !bElm) || aElm < bElm){

   merged.push(aElm);

   aElm = a[i++];

  

  else {

   merged.push(bElm);

   bElm = b[j++];

  }

 }

 return merged;

}

mergeSortedArray([2,5,6,9], [1,2,3,29]);// = [1, 2, 2, 3, 5, 6, 9, 29]

登入後複製

不通过临时变量交换两个数的值

Q:如何在不使用临时变量的情况下交换两个数字?

1

2

3

4

5

6

7

8

9

10

function swapNumb(a, b){

 console.log(&#39;before swap: &#39;,&#39;a: &#39;, a, &#39;b: &#39;, b);

 b = b -a;

 a = a+ b;

 b = a-b;

 console.log(&#39;after swap: &#39;,&#39;a: &#39;, a, &#39;b: &#39;, b);

}

swapNumb(2, 3);

//  = before swap: a: 2 b: 3

//  = after swap: a: 3 b: 2

登入後複製

位操作:对不起,我无法向你解释这一点。 Kinjal Dave建议到 logical conjunction理解它。将浪费您30分钟。

1

2

3

4

5

6

7

8

9

10

function swapNumb(a, b){

 console.log("a: " + a + " and b: " + b);

 a = a ^ b;

 b = a ^ b;

 a = a ^ b;

 console.log("a: " + a + " and b: " + b);

}

swapNumb(2, 3);

// = a: 2 and b: 3

// = a: 3 and b: 2

登入後複製

字符串反向

Q:如何在JavaScript中反转字符串?
A:可以遍历字符串并将字母连接到新字符串。

方法1

1

2

3

4

5

6

7

8

9

function reverse(str){

 var rtnStr = &#39;&#39;;

 for(var i = str.length-1; i>=0;i--){

  rtnStr +=str[i];

 }

 return rtnStr;

}

reverse('you are a nice dude');

// = "edud ecin a era uoy"

登入後複製

Q:你知道在现代浏览器中串联效果很好,但在像IE8这样的旧浏览器中会变慢。 还有什么不同的方法,可以扭转一个字符串?

A:当然.我可以使用数组,也可以添加一些检查。如果字符串是NULL或其他字符串,这将失败。让我也做一些类型检查。使用此数组类似于在某些服务器端语言中使用字符串缓冲区。

方法2

1

2

3

4

5

6

7

8

function reverse(str){

 var rtnStr = [];

 if(!str || typeof str != 'string' || str.length < 2 ) return str;

 for(var i = str.length-1; i>=0;i--){

  rtnStr.push(str[i]);

 }

 return rtnStr.join('');

}

登入後複製

Q: 运行时间复杂度是多少?
A: O(n);
Q:可以做得更好?
A:我可以遍历索引的一半,它会节省一点点。 (这是没用的,可能不会打动面试官)

方法3

1

2

3

4

5

6

7

8

9

10

11

12

function reverse(str) {

 str = str.split('');

 var len = str.length,

   halfIndex = Math.floor(len / 2) - 1,

   revStr;

 for (var i = 0; i <= halfIndex; i++) {

  revStr = str[len - i - 1];

  str[len - i - 1] = str[i];

  str[i] = revStr;

 }

 return str.join(&#39;&#39;);

}

登入後複製

Q:这有效,但你可以做递归的方式吗?
A:可以。

方法4

1

2

3

4

5

6

7

function reverse (str) {

  if (str === "") {

    return "";

  } else {

    return reverse(str.substr(1)) + str.charAt(0);

  }

}

登入後複製

方法5

Q:你可以在方法中使用任何构建,使它更清洁吗?

1

2

3

4

function reverse(str){

 if(!str || str.length <2) return str;

 return str.split(&#39;&#39;).reverse().join(&#39;&#39;);

}

登入後複製

方法6

Q:你可以做反向函数作为字符串扩展吗?
A:我需要将这个函数添加到String.prototype,而不是使用str作为参数,我需要使用this

1

2

3

4

5

6

String.prototype.reverse = function (){

 if(!this || this.length <2) return this;

 return this.split(&#39;&#39;).reverse().join(&#39;&#39;);

}

&#39;abc&#39;.reverse();

// = &#39;cba&#39;

登入後複製

单词反转

Q:你如何在句子中颠倒单词?
A:您必须检查整个字符串的空白区域。确定是否可能有多个空格。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

//have a tailing white space

//fix this later

//now i m sleepy

function reverseWords(str){

 var rev = [],

   wordLen = 0;

 for(var i = str.length-1; i>=0; i--){

  if(str[i]==' ' || i==0){

   rev.push(str.substr(i,wordLen+1));

   wordLen = 0;

  }

  else

   wordLen++;

 }

 return rev.join(' ');

}

登入後複製

内置方法的快速解决方案:

1

2

3

function reverseWords(str){

 return str.split(' ').reverse();

}

登入後複製

原位反转

Q: 如果你有一个字符串如”I am the good boy”, 怎样变为 “I ma eht doog yob”? 注意这些单词位置不变但是被反转了。

A: 要做到这一点,我必须做字符串反向和字反转。

1

2

3

4

function reverseInPlace(str){

 return str.split(' ').reverse().join(' ').split('').reverse().join('');

}

reverseInPlace('I am the good boy');// = "I ma eht doog yob"

登入後複製

Q: ok。好的,你能不使用内置反向函数做到吗?
A: (内心独白)有没有搞错!!

1

2

3

4

5

6

//sum two methods.

//you can simply split words by ' '

//and for each words, call reverse function

//put reverse in a separate function

//if u cant do this, 

//have a glass of water, and sleep

登入後複製

第一个非重复字符

Q: 怎么在字符串中找到第一个非重复字符?
A: 有什么条件吗?
A: 比如是否区分大小写?
面试官可能会说No。
A: 是长字符串还是短字符串?
Q: 这些有什么关系吗?

A:例如,如果它是一个非常长的字符串,说一百万个字符,我想检查是否有26个英文字符正在重复。 我可能会检查是否所有字符都在每200个字母中重复(例如),而不是循环遍历整个字符串。 这将节省计算时间。
Q: 简单起见, 这个字符串是 “the quick brown fox jumps then quickly blow air”。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

function firstNonRepeatChar(str){

 var len = str.length,

   char, 

   charCount = {};

 for(var i =0; i<len; i++){

  char = str[i];

  if(charCount[char]){

   charCount[char]++;

  }

  else

   charCount[char] = 1;

 }

 for (var j in charCount){

  if (charCount[j]==1)

    return j;

 }

}

firstNonRepeatChar(&#39;the quick brown fox jumps then quickly blow air&#39;);// = "f"

登入後複製

这有一个问题,不能再循环中及时退出。

删除重复的字符

Q: 怎样删除字符串中的重复字符?

A: 这与第一个非重复字符非常相似。你应该问一些类似的问题。它是区分大小写的吗?。

如果面试官说,这是区分大小写的,那么你就很轻松了。 如果他说不。你可以使用string.toLowercase()来把字符串。面试官可能不喜欢这个方法。 因为返回字符串不会拥有相同的大小写。 所以

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

function removeDuplicateChar(str){

 var len = str.length,

   char,

   charCount = {},

   newStr = [];

 for(var i =0; i<len; i++){

  char = str[i];

  if(charCount[char]){

   charCount[char]++;

  }

  else

   charCount[char] = 1;

 }

 for (var j in charCount){

  if (charCount[j]==1)

    newStr.push(j);

 }

 return newStr.join(&#39;&#39;);

}

removeDuplicateChar(&#39;Learn more javascript dude&#39;); // = "Lnmojvsciptu"

登入後複製

回文检查

Q: 如何检查一个字符串是否是回文?

A: 把字符串反转,如果反转前后相等,那么它就是回文。

1

2

3

4

5

6

7

8

9

10

11

12

function isPalindrome(str){

 var i, len = str.length;

 for(i =0; i<len/2; i++){

  if (str[i]!== str[len -1 -i])

    return false;

 }

 return true;

}

isPalindrome(&#39;madam&#39;)

// = true

isPalindrome(&#39;toyota&#39;)

// = false

登入後複製

或者

1

2

3

function checkPalindrom(str) {

  return str == str.split(&#39;&#39;).reverse().join(&#39;&#39;);

}

登入後複製

类似的:在 O(n)时间复杂度内判断一个字符串是否包含在回文字符串内。你能在O(1)时间解决问题吗?

找缺失的数字

Q: 在一个1到100的未排序数组中找到缺失的数,你怎么做?

说明:数组中的数字为1到100。 数组中只有一个数字缺失。数组未排序。找到缺少的数字。

A: 你必须表现得像是在想很多。然后讨论n=n(n+1)/2的线性级数之和

1

2

3

4

5

6

7

8

9

10

11

function missingNumber(arr){

 var n = arr.length+1,

   sum = 0,

   expectedSum = n * (n+1)/2;

 for(var i = 0, len = arr.length; i < len; i++){

  sum += arr[i];

 }

 return expectedSum - sum;

}

missingNumber([5, 2, 6, 1, 3]);

// = 4

登入後複製

注意: 这个会返回任意长度数组中缺失的那个

两数之和

Q: 在一个未排序的数组中找出是否有任意两数之和等于给定的数?
A: 简单!双重循环。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

function sumFinder(arr, sum){

 var len = arr.length;

 for(var i =0; i<len-1; i++){

   for(var j = i+1;j<len; j++){

    if (arr[i] + arr[j] == sum)

      return true;

   }

 }

 return false;

}

sumFinder([6,4,3,2,1,7], 9);

// = true

sumFinder([6,4,3,2,1,7], 2);

// = false

登入後複製

Q: 时间复杂度?
A: O(n2)。
Q: 有更优解?
A: 我想想。我可以用一个对象来存储当前元素和和值的差值。当我拿到一个新元素,如果这个元素的差值在对象中存在,那么我就能判断出是否存在。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

function sumFinder(arr, sum){

 var differ = {},

   len = arr.length,

   substract;

 for(var i =0; i<len; i++){

   substract = sum - arr[i];

   if(differ[substract])

    return true;   

   else

    differ[arr[i]] = true;

 }

 return false;

}

sumFinder([6,4,3,2,1,7], 9);

// = true

sumFinder([6,4,3,2,1,7], 2);

// = false

登入後複製

最大和

Q: 找到任意两个元素的最大总和?

A: 这实际上非常简单直接。 找到两个最大的数字并返回它们的总和

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

function topSum(arr){

 var biggest = arr[0],

   second = arr[1],

   len = arr.length,

   i = 2;

 if (len<2) return null;

 if (biggest<second){

  biggest = arr[1];

  second = arr[0];

 }

 for(; i<len; i++){

  if(arr[i] > biggest){

   second = biggest;

   biggest = arr[i];

  }

  else if (arr[i]>second){

   second = arr[i];

  }

 }

 return biggest + second;

}

登入後複製

统计零

Q: 统计从1到n的零总数?

A: 如果 n = 100,则0的数目将是11(0,10,20,30,40,50,60,70,80,90,100)。 请注意,100有两个0.这个看起来很简单,但有点棘手

说明:所以这里的重点是。 如果你有一个1到50的数字,那么这个数值就是5,就是50除以10.然而,如果这个数值是100,这个数值是11,你将得到100/10 = 10和 10/10 = 1。 那就是你将如何在一个数字中得到更多的零,如(100,200,1000);

1

2

3

4

5

6

7

8

9

10

function countZero(n){

 var count = 0;

 while(n>0){

  count += Math.floor(n/10);

  n = n/10;

 }

 return count;

}

countZero(2014);

// = 223

登入後複製

子字符串

Q: 在字符串中匹配子字符串?

A: 在迭代字符串时将使用指针(一个用于字符串,另一个用于子字符串)。 然后用另一个变量来保存初始匹配的起始索引。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

function subStringFinder(str, subStr){

 var idx = 0,

   i = 0,

   j = 0,

   len = str.length,

   subLen = subStr.length;

  for(; i<len; i++){

   if(str[i] == subStr[j])

     j++;

   else

     j = 0;

   //check starting point or a match  

   if(j == 0)

    idx = i;

   else if (j == subLen)

    return idx;

 }

 return -1;

}

subStringFinder('abbcdabbbbbck''ab')

// = 0

subStringFinder('abbcdabbbbbck''bck')

// = 9

//doesn't work for this one.

subStringFinder('abbcdabbbbbck''bbbck'

// = -1

登入後複製

排列

Q: 如何获取字符串中的所有排列?

A: 根据您对算法的了解程度,这可能会很困难。、

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

function permutations(str){

  var arr = str.split(''),

    len = arr.length, 

    perms = [],

    rest,

    picked,

    restPerms,

    next;

  if (len == 0)

    return [str];

  for (var i=0; i<len; i++)

  {

    rest = Object.create(arr);

    picked = rest.splice(i, 1);

    restPerms = permutations(rest.join(''));

    for (var j=0, jLen = restPerms.length; j< jLen; j++)

    {

      next = picked.concat(restPerms[j]);

      perms.push(next.join(''));

    }

  }

  return perms;

}

登入後複製

相信看了本文案例你已经掌握了方法,更多精彩请关注php中文网其它相关文章!

相关学习推荐:javascript视频教程

以上是Js面試中容易遇到的演算法的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
最新問題
怎麼實作 JavaScript點與圓的位置關係
來自於 1970-01-01 08:00:00
0
0
0
JavaScript鉤子函數是什麼?
來自於 1970-01-01 08:00:00
0
0
0
c++ 呼叫javascript
來自於 1970-01-01 08:00:00
0
0
0
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板