京东2018校招技术笔试编程题参考代码及思路

以下代码均来自几位大神的,不是我写的⁄(⁄ ⁄•⁄ω⁄•⁄ ⁄)⁄ 大神们说不想露脸(✿◡‿◡)

在这里分享给大家~

回文

分析

暴力枚举一下check回文,可以确定出最后答案的一半,就可以得到答案了。

参考代码

#include <bits/stdc++.h>

using namespace std;

bool isPalindrome(string s) {
    int i = 0, j = s.size() - 1;
    while(i < j) 
        if(s[i++] != s[j--])
            return false;
    return true;
}
string s;
int main() {
    cin >> s;
    string add;
    int i, j;
    for(i = 0; i < s.size(); i++) {
        if(isPalindrome(s.substr(i))) break;
    }
    for(int j = i - 1; j >= 0; j--) add += s[j];
    s += add;
    cout << s.size() << endl;
}

两个子串

分析

暴力枚举计算字符串前缀后缀相等的最长长度,然后拼接一下就是结果。

参考代码

#include <bits/stdc++.h>

using namespace std;

string s;

int main() {
    cin >> s;
    int l = s.size(), m = 0;
    for(int i = 1; i < l; i++) {
        if(s.substr(0, i) == s.substr(l - i, i))
            m = i;
    }
    cout << s.substr(0, l - m) + s << endl;
    return 0;
}

括号匹配方案

分析

这个题字符串长度不长,可以直接爆搜。

有一种代码比较简单的解法,挨着累计'('的个数,遇到')'就完成一次匹配,把情况数乘进答案。本质是把题目所说的移除操作做了一个等价的变化。

参考代码

#include <bits/stdc++.h>

using namespace std;

string s;
int main() {
    cin >> s;
    int ans = 1, cnt = 0;
    for(int i = 0; i < s.size(); i++) {
        if(s[i] == '(') {
            cnt++;
        } else {
            ans *= cnt;
            cnt--;
        }
    }
    cout << ans << endl;
}

疯狂的序列

分析

范围比较大,考虑二分答案,等差数列求和来check。

参考代码

#include <bits/stdc++.h>

using namespace std;

long long n;

int main() {
    cin >> n;
    long long l = 1, r = 2000000000LL, mid;
    while(l < r) {
        mid = (l + r) / 2;
        if(mid * (mid + 1) / 2 >= n)
            r = mid;
        else 
            l = mid + 1;
    }
    cout << l << endl;
    return 0;
}

神奇数

分析

注意到r - l不会特别大,直接枚举范围内的数,然后暴力check。
暴力check的时候注意到所有数位和最大只有81,所以会比较快。

参考代码

#include <bits/stdc++.h>

using namespace std;

bool check(int n) {
    char s[11];
    int cur = 0, t = 0;
    while(n > 0) {
        s[cur] = n % 10;
        t += s[cur++];
        n /= 10;
    }
    if(t % 2) return false;
    t /= 2;
    bool ok[42] = {0};
    ok[s[0]] = true;
    for(int i = 1; i < cur; i++) {
        int v = s[i];
        for(int j = 41; j >= 0; j--) {
            if(ok[j] && j + v <= 41) {
                ok[j + v] = true;
            }
        }
        if(ok[t]) {
            return true;
        }
    }
    return false;
}
int l, r;
int main() {
    int res = 0;
    cin >> l >> r;
    for(int i = l; i <= r; i++) {
        if(check(i)) res++;
    }
    cout << res << endl;
    return 0;
}
import java.util.HashSet;
import java.util.Scanner;
import java.util.Set;

public class Main {


  public static boolean check(int n){
    int[] s = new int[11];
    int cur = 0, t = 0;
    while(n > 0) {
      s[cur] = n % 10;
      t += s[cur++];
      n /= 10;
    }
    if( t % 2 == 1 ) return false;
    t /= 2;
    boolean[] ok = new boolean[42];
    ok[s[0]] = true;
    for(int i = 1; i < cur; i++) {
      int v = s[i];
      for(int j = 41; j >= 0; j--) {
        if(ok[j] && j + v <= 41) {
          ok[j + v] = true;
        }
      }
      if(ok[t]) {
        return true;
      }
    }
    return false;
  }


  public static int max(int a, int b){
    return (a>b) ? a : b;
  }

  public static void main(String[] args) {
    Scanner in = new Scanner(System.in);
    int res = 0;
    int l = in.nextInt();
    int r = in.nextInt();
    for(int i = l; i <= r; i++) {
      if(check(i)) res++;
    }
    System.out.println(res);

  }
}

求幂

分析

裸暴力好像只能拿20%的分

我们考虑去枚举n范围内的所有i,然后处理出i的幂那些数。
考虑对于i ^ x, 我们需要计算满足 (i ^ x) ^ c = (i ^ y) ^ d的数量,其中i ^ x, i ^ y <= n. 这些我们可以通过预处理出来。
然后对于(i ^ x) ^ c = (i ^ y) ^ d 其实意味着x c = y d, 意味着(x / y) = (d / c), 其中x, y我们可以在预处理之后枚举出来,于是我们就可以借此计算出n范围内有多少不同这种c和d去满足等式。
其实就等于 n / max(x / gcd(x, y), y / gcd(x, y)),然后都累加进答案。gcd()表示最大公约数。
中间可能产生重复枚举,我们用一个set或者hash容器标记一下就好。

以上枚举对于2~sqrt(n)。最后对于大于sqrt(n)的部分,每个的贡献都是n。

参考代码

#include <bits/stdc++.h>

using namespace std;

const int mod = 1e9 + 7;

set<int> S;
int n;
int main() {
    cin >> n;
    long long res = 1LL * n * n % mod;
    for(int i = 2; i * i <= n; i++) {
        if(S.find(i) != S.end()) continue;
        long long tmp = i;
        int cnt = 0;
        while(tmp <= n) {
            S.insert(tmp);
            tmp = tmp * i;
            cnt++;
        }
        for(int x = 1; x <= cnt; x++) {
            for(int y = 1; y <= cnt; y++) {
                int g = __gcd(x, y);
                int tmpx = x / g;
                int tmpy = y / g;
                res = (res + n / max(tmpx, tmpy)) % mod;
            }
        }
    }
    res += 1LL * (n - S.size() - 1) * n;
    res %= mod;
    cout << res << endl;
}
import java.util.HashSet;
import java.util.Scanner;
import java.util.Set;

public class Main {

  public final static long MOD = 1000000000 + 7;

  public static int max(int a, int b){
    return (a>b) ? a : b;
  }

  public static long gcd(long a,long b){
    return (a % b == 0) ? b : gcd(b,a%b);
  }

  public static void main(String[] args) {
    Scanner in = new Scanner(System.in);
    long n = in.nextInt();
    long ans = (long)1*n*(n*2-1) % MOD;
    Set<Integer> set  = new HashSet<>();
    for (int i = 2; i*i <= n; i++){
      if ( set.contains(i)) continue;
      long tmp = i;
      int cnt = 0;

      while(tmp <= n) {
        set.add((int)tmp);
        tmp = tmp * i;
        cnt++;
      }

      for(int k = 1; k <= cnt; k++) {
        for(int j = k + 1; j <= cnt; j++) {
          ans = (ans + n / (j / gcd(k, j) ) * (long)2 ) % MOD;
        }
      }

    }

    System.out.println(ans);
  }
}

前端

购物车

function add(items) {
    var tbody = document.getElementById('jsTrolley').getElementsByTagName('tbody')[0];
    (items || []).forEach(function (item) {
        var tr = document.createElement('tr');
        tr.innerHTML = '<td>' + item.name 

 + '</td><td>' + item.price.toFixed(2) + '</td><td><a href="javascript:void(0);">删除</a></td>';
        tbody.appendChild(tr);
    });
    update();
}

function bind() {
    var table = document.getElementById('jsTrolley');
    table.addEventListener('click', function (event) {
        var el = event.target;
        if (el.tagName.toLowerCase() === 'a') {
            tr = el.parentNode.parentNode;
            tr.parentNode.removeChild(tr);
            update();
        }
    });
}

function update() {
    var table = document.getElementById('jsTrolley');
    var tbody = table.getElementsByTagName('tbody')[0];
    var tfoot = table.getElementsByTagName('tfoot')[0];
    var tr = [].slice.call(tbody.getElementsByTagName('tr'), 0);
    var total = 0;
    tr.forEach(function (tr) {
        total += +(tr.getElementsByTagName('td')[1].innerHTML.trim());
    });
    tfoot.getElementsByTagName('td')[0].innerHTML = total.toFixed(2) + '(' + tr.length + '件商品)';
}
全部评论
//疯狂序列的 #include <iostream> #include <algorithm> using namespace std; int main() { long long n = 0; cin >> n; long long rest = sqrt(n * 2); if (rest*(rest + 1) / 2 < n) { cout << rest + 1; return 0; } cout << rest; return 0; }
点赞 回复 分享
发布于 2017-09-08 21:12
括号匹配方案的理解 看到大佬做的括号匹配,才意识到这道题的本质就是求 左右括号匹配的方式有几种。我们被题意牵着走,所以总想着每次先删除左括号,其实更快的做法是每次想着先去删除右括号,删除的时候看左边有几个左括号,这些左括号便都是可以和它配对的。然后将左括号删掉一个,继续往后处理。 那么我们的做法就可以优化为:从前往后看,遇到左括号就记录(左括号的个数记为 cnt ),遇到右括号就根据 cnt 得到左边的左括号数(),那么就有 cnt 种方式进行匹配,然后将 ans 乘以 cnt ,cnt--,再往后处理,直到最后就结束。
点赞 回复 分享
发布于 2017-09-08 23:53
有没有提交的地方 想看看自己写的,能对多少
点赞 回复 分享
发布于 2017-09-08 21:16
前端购物车这个题 他的一行代码 顶我5行
点赞 回复 分享
发布于 2017-09-08 21:21
牛妹太厉害了
点赞 回复 分享
发布于 2017-09-08 21:10
更新了前端的
点赞 回复 分享
发布于 2017-09-08 21:13
算法岗位: 第一题:通过率为70%,时间复杂度不够,求赐教~~~ import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner in = new Scanner(System.in); long n = in.nextLong(); System.out.println(solution(n)); in.close(); } public static long solution(long n){ if(n == 0){ return 0; } long result = 1; while(true){ ++result; if(n>(result*(result+1)/2) && n<(result+1)*(result+2)/2){ break; } } return result+1; } } 第二题目:(不知道能不能通过,因为我没时间了,哎~~~,我感觉即使能过,时间复杂度也过不了) import java.util.Scanner; public class Main{ public static void main(String[] args) { Scanner in = new Scanner(System.in); long n = in.nextLong(); System.out.println(solution(n)); in.close(); } public static long solution(long n){ if(n == 0){ return 0; } long result = 0; long m = 10000000007L; result+= Math.pow(n, n) + (n-1)*n; for(int a = 2;a<=n;a++){ for(int b = 2;a<=n;a++){ for(int c = 2;a<=n;a++){ for(int d = 2;a<=n;a++){ if(a!=b && c!=d && a!=c){ if(Math.pow(a, b) == Math.pow(c,d)){ result ++; } } } } } } return result%m; } }
点赞 回复 分享
发布于 2017-09-08 21:20
算法岗位: 第一题:通过率为70%,时间复杂度不够,求赐教~~~ import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner in = new Scanner(System.in); long n = in.nextLong(); System.out.println(solution(n)); in.close(); } public static long solution(long n){ if(n == 0){ return 0; } long result = 1; while(true){ ++result; if(n>(result*(result+1)/2) && n<(result+1)*(result+2)/2){ break; } } return result+1; } } 第二题目:(不知道能不能通过,因为我没时间了,哎~~~,我感觉即使能过,时间复杂度也过不了 没有中间的四层for循环,通过率为10%,京东笔试也就这样了) import java.util.Scanner; public class Main{ public static void main(String[] args) { Scanner in = new Scanner(System.in); long n = in.nextLong(); System.out.println(solution(n)); in.close(); } public static long solution(long n){ if(n == 0){ return 0; } long result = 0; long m = 10000000007L; result+= Math.pow(n, n) + (n-1)*n; for(int a = 2;a<=n;a++){ for(int b = 2;a<=n;a++){ for(int c = 2;a<=n;a++){ for(int d = 2;a<=n;a++){ if(a!=b && c!=d && a!=c){ if(Math.pow(a, b) == Math.pow(c,d)){ result ++; } } } } } } return result%m; } }
点赞 回复 分享
发布于 2017-09-08 21:25
牛妹,你别装了,其实大神是我
点赞 回复 分享
发布于 2017-09-08 21:41
京东第二道题《神奇数》的非暴力解法!!!时间不够,只AC了10%,现在才做了出来 #include <iostream> using namespace std; bool Istwo(long long sum, long long cnt) { bool sta = false; long long count = cnt * 10; int datacnt = 0; while (count > 0) { if ((count % 10) > (sum % 10)) datacnt++; count /= 10; sum /= 10; } if (datacnt % 2 == 0) sta = true; return sta; } int main() { long long n, m; while (cin >> n >> m) { long long count = m / 11 - n / 11; for (long long i = n / 11; i <= m / 11; i++) if (Istwo(i * 11, i) == false) count--; if (n / 11 * 11 == n) count++; cout << count << endl; } return 0; }
点赞 回复 分享
发布于 2017-09-08 21:47
疯狂随机数: #include <iostream> #include <math.h> using namespace std; int main() { double n; while (cin>>n){ int k; k=(sqrt(1+8*n)-1)/2; if(2*n>pow(k,2) && 2*n<=(pow(k,2)+k))cout<<k; else cout<<k+1; } return 0; }
点赞 回复 分享
发布于 2017-09-08 21:59
疯狂的序列,其实是个等差数列求和(r*(r+1))/2 = n逆问题,输入n,求r if __name__ == "__main__": n = int(sys.stdin.readline().strip()) result = int((1+(1+8*n)**0.5)/2) if( (result**2 + result)/2 - result <= n <= (result**2 + result)/2): print(result) else: print(result+1)
点赞 回复 分享
发布于 2017-09-08 22:41
求幂的那个还是不是很懂
点赞 回复 分享
发布于 2017-09-08 23:42
回文可以不暴力,用kmp的next数组,因为next数组就是一种前缀后缀的关系。 求出第n+1位的next数组的值,就是需要拼接的开始位置 #include<iostream> #include<string> using namespace std; int next1[60]; void getNext(string p,int *next1,int size) {     int j,k;     next1[0]=-1;     j=0;     k=-1;     while(j<size-1){         //匹配的情况下,p[j]==p[k]         if(k==-1||p[j]==p[k])    {             j++;             k++;             next1[j]=k;         }         else                   //p[j]!=p[k]             k=next1[k];     } } int main(){ string s;     while(cin>>s){         int n = s.size();         getNext(s,next1,n+1);         string result = s;         int start = next1[n];         result += s.substr(start,n);         cout<<result<<endl;     }     return 0; }
点赞 回复 分享
发布于 2017-09-09 11:23
牛妹好厉害!
点赞 回复 分享
发布于 2017-09-08 21:12
牛妹,你好及时啊!
点赞 回复 分享
发布于 2017-09-08 21:13
function find(n){ var i=1; while(i*i+i<=n*2-1){ i++; } return i; }//疯狂序列,自己写的,各位大佬指点一下
点赞 回复 分享
发布于 2017-09-08 21:15
大神好厉害啊
点赞 回复 分享
发布于 2017-09-08 21:18
回文 static void Main(string[] args) { char[] str = Console.ReadLine().ToCharArray(); int length = str.Length; go1: while (true) { length++; char[] newStr = new char[length]; for (int i = 0; i < str.Length; i++) { newStr[i] = str[i]; } int span = length - str.Length; for (int i = 1; i <= span; i++) { newStr[length - i] = str[i - 1]; } int left = 0; int right = length - 1; while (left < length) { if(newStr[left]!=newStr[right]) goto go1; left++; right--; } break; } Console.WriteLine(length); }
点赞 回复 分享
发布于 2017-09-08 21:18
求第几位 static void Main(string[] args) { long n = Convert.ToInt64(Console.ReadLine()); long cout = 0; long num = 0; while (cout<n) { num++; cout += num ; } Console.WriteLine(num); }
点赞 回复 分享
发布于 2017-09-08 21:19

相关推荐

51 188 评论
分享
牛客网
牛客企业服务