《硅谷》第三季第一集神秘代码

2023-11-05 13:10

本文主要是介绍《硅谷》第三季第一集神秘代码,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

java详细解剖《硅谷》第三季第一集”神秘代码”


问题引出:
相信不少对程序编程感兴趣的小伙伴都或多或少地知道或了解《硅谷》这部美国电视剧(话说楼主可是熬了两个通宵才看完的)。这部电视剧讲述程序员在硅谷创业的艰辛历程,但是不可否认,硅谷是程序员的圣地,也是天才的集聚地……我们今天就其中的一段代码来进行剖析(这里我要说一下,HBO的编剧真的很走心),来巩固和学习一下。其实这段“神秘代码”就是关于花式字符串输出的程序代码,今天就用java代码来还原一下。


这是电视剧中的源代码,下面会贴出github地址

其实刚看到这段代码我是懵比的,这是什么东西,但是可以从这段程序中分析出几个很“蹊跷”的问题:

  1. 仔细看后发现这段程序进行了大量的位移操作,其实位移运算是很简单的,例如2的23次方就是 "System.out.println(2 << 23);"(其实我有这种感觉,自从学了关于计算机运算的运算符,写个数就不好好写了,但是我认为在实际开发中最好不要这样用,毕竟代码不是你自己写,会影响其他的人理解);

  2. "int c = (((s & ((dcf_t)0x1FULL << i * 5)) >> i * 5) + 65);" , 这段代码还有一个魔术数字“65”,仔细一想,“65”不就是字符“A”的ASCII码吗?这句代码每次都要加上这个魔术数字,同时又参考了知乎大神的答案,真相就是前面那一堆代码就是在计算与字符“A”的偏移量;

  3. "_ctx_iface(0x79481E6BBCC01223 + ((dcf_t0x1222DC << 64)), i);" , 大眼一看,这一串的十六进制数字是什么,如果看不懂,就先记住数字是用十六进制表示的就行了。


好的,下面就先按我的思路来逆推一下

  1. 首先由输出结果入手,”DREAM_ON_ASSHOLES”,将每个字符的ASCII写出来,分别计算与字符‘A’(65)的偏移量,然后转换为固定5位的二进制数字。如下图:
    这里写图片描述
  2. 由上面的图我们可以得到每个字符相对于字符‘A’偏移量的二进制0101代码。好的,我们再来看程序中的一句代码"printf("%c", c);",这明明就是单字符输出啊,可是程序需要用十六进制表示并从底位向高位解析,所以需要将相对应的0101代码片段从后往前依次连接起来。如下图:
    这里写图片描述
  3. 好了,终于得到一大串0101代码了,每4位将其转换为16进制,注意高位不够4位的需要补零。如下图:

这里写图片描述

好了,分析完了,就要上代码了

package mysterious.mysterious_01;import java.math.BigInteger;/*** @function 本程序纯属搞笑^_^,出自《硅谷》第三季第一集中的神秘代码,主要功能就是"花式输出字符串"。* @describe 本程序主要是利用ASCII码中的(A --> 65)计算与各个字母的偏移量,再进行掩码处理,得到字符串的十六进制数字* @author Mr.leaf* @time 2017-02-03 21:13* */
public class MysteriousCodeUtil {private String inputString = "";//由用户输入的字符串private BigInteger hexResult1 = new BigInteger("0");//经加密后生成的两个十六进制数值 private BigInteger hexResult2 = new BigInteger("0");public MysteriousCodeUtil(String inputString){  this.setInputString(inputString);}public String getInputString() {return inputString;}public void setInputString(String inputString) {if(inputString == null || "".equals(inputString)){throw new NullPointerException();}this.inputString = inputString;}/*** @function 将用户输入的字符串转换为字符数组,并逐一计算与'A'的偏移量,然后进行掩码处理,得到十六进制数值* @describe 其实过程很简单,主要是关于掩码处理问题和移位后的或运算问题 * */public void encryptString(){char[] charArray = this.inputString.toCharArray();//将用户输入的字符串转换为字符数组for(int i = 0; i < charArray.length && charArray[i] != '\n'; i++){if(i < 12){//为什么是12?因为在解码的过程中使用左移64位,掩码是每5位,12*5=60,又64-60=4,这就解释了为什么下面使用"0xF"和"0x10"hexResult1 = hexResult1.or((new BigInteger(String.valueOf(charArray[i] - 'A')).and(new BigInteger("1F", 16))).shiftLeft(i * 5));  //因为64 / 5 = 12余4,所以i小于12时数值1要与11111进行&运算}else if(i == 12){hexResult1 = hexResult1.or((new BigInteger(String.valueOf(charArray[i] - 'A')).and(new BigInteger("F", 16))).shiftLeft(i * 5));   //"F" --> 01111,i等于12时数值1要与01111进行&运算hexResult2 = hexResult2.or((new BigInteger(String.valueOf(charArray[i] - 'A')).and(new BigInteger("10", 16))).shiftRight(4));     //"10" --> 10000,i等于12时数值2要与10000进行&运算}else{hexResult2 = hexResult2.or((new BigInteger(String.valueOf(charArray[i] - 'A')).and(new BigInteger("1F", 16))).shiftLeft((i - 13) * 5 + 1));   //由于在i等于12时数值2与10000进行&运算,所以在i大于12时数值2与11111进行&运算后的移位操作要+1}}}/*** @function 将经过加密的字符串还原* @describe 这个过程主要是讲一大串二进制数字还原出来,先将hexResult2此十六进制数值移位64位,然后与hexResult1进行串接* */public void printResult(){for(int j = 0; j < this.inputString.length(); j++){this.countOffset((hexResult1.add(hexResult2.shiftLeft(64))), j);}   }/*** @function 首先将得到的十六进制数值进行去除掩码操作,然后加上起始值'A',最终还原此字符* @param s 转换后偏移量的十六进制值* @param i 将要还原字符串中第i位字符* */public void countOffset(BigInteger s, int i){BigInteger c = ((s.and(new BigInteger("11111", 2).shiftLeft(i * 5))).shiftRight(i * 5)).add(new BigInteger("65"));System.out.printf("%c", c.intValue());  }
}

我将程序代码封装成一个工具类,具体方法的解释可以看注释。好的,大致就是这样,如果程序中的注释不是很明白,可以留言给楼主!!!
告诉大家一个秘密,其实用System.out.println("DREAM_ON_ASSHOLES"); 可以实现同样的效果 ^_^


参考文章地址:
https://www.zhihu.com/question/44606486

这篇关于《硅谷》第三季第一集神秘代码的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/350074

相关文章

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

计算机毕业设计 大学志愿填报系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点赞 👍 收藏 ⭐评论 📝 🍅 文末获取源码联系 👇🏻 精彩专栏推荐订阅 👇🏻 不然下次找不到哟~Java毕业设计项目~热门选题推荐《1000套》 目录 1.技术选型 2.开发工具 3.功能

代码随想录冲冲冲 Day39 动态规划Part7

198. 打家劫舍 dp数组的意义是在第i位的时候偷的最大钱数是多少 如果nums的size为0 总价值当然就是0 如果nums的size为1 总价值是nums[0] 遍历顺序就是从小到大遍历 之后是递推公式 对于dp[i]的最大价值来说有两种可能 1.偷第i个 那么最大价值就是dp[i-2]+nums[i] 2.不偷第i个 那么价值就是dp[i-1] 之后取这两个的最大值就是d

pip-tools:打造可重复、可控的 Python 开发环境,解决依赖关系,让代码更稳定

在 Python 开发中,管理依赖关系是一项繁琐且容易出错的任务。手动更新依赖版本、处理冲突、确保一致性等等,都可能让开发者感到头疼。而 pip-tools 为开发者提供了一套稳定可靠的解决方案。 什么是 pip-tools? pip-tools 是一组命令行工具,旨在简化 Python 依赖关系的管理,确保项目环境的稳定性和可重复性。它主要包含两个核心工具:pip-compile 和 pip

D4代码AC集

贪心问题解决的步骤: (局部贪心能导致全局贪心)    1.确定贪心策略    2.验证贪心策略是否正确 排队接水 #include<bits/stdc++.h>using namespace std;int main(){int w,n,a[32000];cin>>w>>n;for(int i=1;i<=n;i++){cin>>a[i];}sort(a+1,a+n+1);int i=1

html css jquery选项卡 代码练习小项目

在学习 html 和 css jquery 结合使用的时候 做好是能尝试做一些简单的小功能,来提高自己的 逻辑能力,熟悉代码的编写语法 下面分享一段代码 使用html css jquery选项卡 代码练习 <div class="box"><dl class="tab"><dd class="active">手机</dd><dd>家电</dd><dd>服装</dd><dd>数码</dd><dd

生信代码入门:从零开始掌握生物信息学编程技能

少走弯路,高效分析;了解生信云,访问 【生信圆桌x生信专用云服务器】 : www.tebteb.cc 介绍 生物信息学是一个高度跨学科的领域,结合了生物学、计算机科学和统计学。随着高通量测序技术的发展,海量的生物数据需要通过编程来进行处理和分析。因此,掌握生信编程技能,成为每一个生物信息学研究者的必备能力。 生信代码入门,旨在帮助初学者从零开始学习生物信息学中的编程基础。通过学习常用

husky 工具配置代码检查工作流:提交代码至仓库前做代码检查

提示:这篇博客以我前两篇博客作为先修知识,请大家先去看看我前两篇博客 博客指路:前端 ESlint 代码规范及修复代码规范错误-CSDN博客前端 Vue3 项目开发—— ESLint & prettier 配置代码风格-CSDN博客 husky 工具配置代码检查工作流的作用 在工作中,我们经常需要将写好的代码提交至代码仓库 但是由于程序员疏忽而将不规范的代码提交至仓库,显然是不合理的 所

Unity3D自带Mouse Look鼠标视角代码解析。

Unity3D自带Mouse Look鼠标视角代码解析。 代码块 代码块语法遵循标准markdown代码,例如: using UnityEngine;using System.Collections;/// MouseLook rotates the transform based on the mouse delta./// Minimum and Maximum values can