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

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

相关文章

使用Dify访问mysql数据库详细代码示例

《使用Dify访问mysql数据库详细代码示例》:本文主要介绍使用Dify访问mysql数据库的相关资料,并详细讲解了如何在本地搭建数据库访问服务,使用ngrok暴露到公网,并创建知识库、数据库访... 1、在本地搭建数据库访问的服务,并使用ngrok暴露到公网。#sql_tools.pyfrom

Java springBoot初步使用websocket的代码示例

《JavaspringBoot初步使用websocket的代码示例》:本文主要介绍JavaspringBoot初步使用websocket的相关资料,WebSocket是一种实现实时双向通信的协... 目录一、什么是websocket二、依赖坐标地址1.springBoot父级依赖2.springBoot依赖

讯飞webapi语音识别接口调用示例代码(python)

《讯飞webapi语音识别接口调用示例代码(python)》:本文主要介绍如何使用Python3调用讯飞WebAPI语音识别接口,重点解决了在处理语音识别结果时判断是否为最后一帧的问题,通过运行代... 目录前言一、环境二、引入库三、代码实例四、运行结果五、总结前言基于python3 讯飞webAPI语音

什么是 Java 的 CyclicBarrier(代码示例)

《什么是Java的CyclicBarrier(代码示例)》CyclicBarrier是多线程协同的利器,适合需要多次同步的场景,本文通过代码示例讲解什么是Java的CyclicBarrier,感... 你的回答(口语化,面试场景)面试官:什么是 Java 的 CyclicBarrier?你:好的,我来举个例

基于Canvas的Html5多时区动态时钟实战代码

《基于Canvas的Html5多时区动态时钟实战代码》:本文主要介绍了如何使用Canvas在HTML5上实现一个多时区动态时钟的web展示,通过Canvas的API,可以绘制出6个不同城市的时钟,并且这些时钟可以动态转动,每个时钟上都会标注出对应的24小时制时间,详细内容请阅读本文,希望能对你有所帮助...

HTML5 data-*自定义数据属性的示例代码

《HTML5data-*自定义数据属性的示例代码》HTML5的自定义数据属性(data-*)提供了一种标准化的方法在HTML元素上存储额外信息,可以通过JavaScript访问、修改和在CSS中使用... 目录引言基本概念使用自定义数据属性1. 在 html 中定义2. 通过 JavaScript 访问3.

Flutter监听当前页面可见与隐藏状态的代码详解

《Flutter监听当前页面可见与隐藏状态的代码详解》文章介绍了如何在Flutter中使用路由观察者来监听应用进入前台或后台状态以及页面的显示和隐藏,并通过代码示例讲解的非常详细,需要的朋友可以参考下... flutter 可以监听 app 进入前台还是后台状态,也可以监听当http://www.cppcn

Python使用PIL库将PNG图片转换为ICO图标的示例代码

《Python使用PIL库将PNG图片转换为ICO图标的示例代码》在软件开发和网站设计中,ICO图标是一种常用的图像格式,特别适用于应用程序图标、网页收藏夹图标等场景,本文将介绍如何使用Python的... 目录引言准备工作代码解析实践操作结果展示结语引言在软件开发和网站设计中,ICO图标是一种常用的图像

Java中有什么工具可以进行代码反编译详解

《Java中有什么工具可以进行代码反编译详解》:本文主要介绍Java中有什么工具可以进行代码反编译的相关资,料,包括JD-GUI、CFR、Procyon、Fernflower、Javap、Byte... 目录1.JD-GUI2.CFR3.Procyon Decompiler4.Fernflower5.Jav

javaScript在表单提交时获取表单数据的示例代码

《javaScript在表单提交时获取表单数据的示例代码》本文介绍了五种在JavaScript中获取表单数据的方法:使用FormData对象、手动提取表单数据、使用querySelector获取单个字... 方法 1:使用 FormData 对象FormData 是一个方便的内置对象,用于获取表单中的键值