每天一道Java题[1]

问题:char[]与String相比,有什么优胜的地方?

回答:

针对安全保密高的信息,char[]比String做得更好。因为String是不可变得,即使你修改原先的变量,实际上也是在内存中新建一个对象,原数据还是保留在内存中,等待回收。而char[]中的元素是可以更改的。这就意味着,如密码等保密信息用完之后,你可以马上修改它而不能痕迹。从而相对于String有更好的安全性。可从下面例子中看出,char[]变更内容后,仍是那个对象。而String已经不是原来的String了。

引用:

这个对比,在知乎中这个贴有非常好的回答,我把它引用在下面:

作者:知乎用户

链接:https://www.zhihu.com/question/36734157/answer/68767786

来源:知乎

著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

原答案

=====================

图里说的很清楚,虽然String加载密码之后可以把这个变量扔掉,但是字符串并不会马上被GC回收,一但进程在GC执行到这个字符串之前被dump,dump出的的转储中就会含有这个明文的字符串。

那如果我去"修改"这个字符串,比如把它赋一个新值,那么是不是就没有这个问题了?

答案是否定的,因为String本身是不可修改的,任何基于String的修改函数都是返回一个新的字符串,原有的还会在内存里。

对于char[]来说,你可以在抛弃它之前直接修改掉它里面的内容,密码就不会存在了。但是如果你什么也不做直接交给gc的话,也会存在上面一样的问题。

有的大神说这个是毫无根据的扯淡

这些所谓的大神应该没有什么安全方面的常识,这非常普遍。

FAQ

======================

1. 这种做法意义有多大?

如果没有及时清空而由GC来清除的话,暴露窗口大约是秒这个数量级,如果能够在计算HASH后立即清除,暴露窗口大约是微秒数量级。如此简单的设计就可以降低如此多的被攻击概率,性价比是非常高的。

2. 如何使用反射来修改String? 和修改char[] 相比,有何区别和风险?

通过reflection机制可以查看String的内部的内存成员,从而可以直接修改其中的数据区。但是这样的做法会有问题,内部化的String为了提高HASH速度,节省空间,值相同的字符串通常只有一个实例。

你自己的char[],修改它是没有副作用的。但是String里的char[],很可能是多个String所共享的,你改掉它就会殃及别的String。举个例子,有一个密码是"Password",而你密码框提示密码输入的文字也是"Password",改掉第一个"Password"会把后面那个也改掉。

3. 如果一点明文也不想出现,应该怎么做?

为了保证"全部处理流程均无明文密码",需要底层API在给你密码之前就做了HASH,并且这个HASH算法就是你想要的那种。最好还加盐。不过这只是在用户程序方面无明文,底层获取中会不会有明文就保证不了了。

4. 有没有绝对安全策略?

安全往往是相对于攻击成本而言的,攻击收益越高,黑客就越能接受攻击成本高的方案。因此,你采取的安全策略应该与这个攻击收益相匹配。对于极其敏感和宝贵的数据来源,就需要在安全方面上下很大功夫。目前来看,没有绝对的安全,只有相对的安全。

时间: 2024-12-23 05:58:45

每天一道Java题[1]的相关文章

每天一道Java题[11]

题目 synchronized怎么实现线程同步?请修改<每天一道Java题[10]>中的MyRunnableThread类以解决三个线程都获取到10的问题. 解答 方法一: 采用synchronized关键字包裹需要保证线程安全的代码块,来实现线程同步.语法格式为: Synchronized(expression){ //需同步的代码 } <每天一道Java题[10]>中的MyRunnableThread类修改为: package me.huangzijian; public cl

每天一道Java题[3]

问题 为什么在重写equals()方法的同时,必须重写hashCode()方法? 解答 在<每天一道Java题[2]>中,已经对hashCode()能否判断两个对象是否相等做出了解释.equals()方法与hashCode()方法的关系如下: 如果两个对象的hashCode()返回值不一样,则equals()返回的结果必为false. 如果两个对象的hashCode()返回值一样的时候,equals()返回的结果未知. 如果两个对象的equals()返回的结果为true,则两个对象的hashC

每天一道Java题[9]

题目 native关键字的作用是什么? 解答 首先,需了解JNI(Java Native Interface),它是连接Java平台与本地C代码的一个API. 其次,用native关键字声明的方法,是告诉JVM调用的方法是一个外部定义的方法,也就是本地C代码定义的一个方法. 总结来说,native关键字的具体作用是,用它声明的方法,并不需要Java代码自己实现.而是JVM通过JNI来加载本地系统C/C++的DLL,然后调用其中的方法来实现. 发散思维 1.说一下,我们平时用到的哪个方法,是用na

每天一道Java题[6]

题目 String字符串怎么转换为Date,Date又怎么转换成String字符串 解答 String->Date 主要用到类SimpleDateFormat及其抽象父类DateFormat中的方法parse(). 如下: String dateStr = "2017-05-18 16:00:00"; SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); Date date =

每天一道Java题[10]

题目 阐述创建线程最常用的两种方法及其对比. 解答 方法一:继承Thread类实现 步骤: 创建Thread类的子类,如MyThread. 重写Thread类的run()方法. 实例化MyThread类,对象名如myThread. 运用Thread类的start()方法启动线程,如myThread.start(). 方法二:实现Runnable接口 步骤: 创建一个类,如MyRunnableThread,实现Runnable接口. 创建MyRunnableThread类的对象. 实例化Threa

每天一道Java题[5]

题目 String.StringBuilder.StringBuffer有什么异同? 解答 相同点:String.StringBuilder.StringBuffer都可以用来存储字符串. 不同点: 1.String与StringBuilder.StringBuffer的不同点主要在于,String对象创建之后,是不可改变的,平时我们对同一个String变量赋值,实际上是创建了个新的对象.而后两者是可变的,这意思就是说,他们可以在同一内存地址上更改它的值,而无需创建新的对象. 2.StringB

每天一道Java题[8]

以下题目及解答属于个人见解,欢迎大家也分享和补充一下解答的内容,互相促进,共同进步! 题目 RESTful WebService与SOAP WebService有什么异同? 解答 SOAP是一个协议,而REST实际上是一个互联网软件的架构原则,并不是一个协议.它更像是对Http协议的设计初衷作诠释. SOAP发展到现在,相对REST来说,已经有相当的成熟度. SOAP WebService对于消息体.消息头等内容有统一的规范,通用性相对更强.而RESTful WebSerivce实际上只是一个设

每天一道Java题[7]

题目 什么是REST原则,请解释RESTful架构,以及其设计思想? 解答 REST,全称为Representation State Transfer,是一种互联网软件的架构原则.凡是满足REST原则的,我们都称它为RESTful架构. 对RESTful架构的理解,有以下几点: 资源,网络上的一个实体,或者是一个具体的信息.通常使用一个URI来表示一种资源. Representation,可理解为资源的表现层,资源的具体表现形式.在http请求的头信息Accept和Content-Type字段指

每天一道Java题[2]

问题 可以直接根据hashCode()方法产生的值判断两个对象是否相等吗? 解答 不能!根据Wikipedia(https://en.wikipedia.org/wiki/Java_hashCode())上对hashCode()方法的解释,它会根据这个对象内存储的数据及对象的一些特征来做散列,并返回一个有符号的32位哈希值.从这解释我们就可以看到,hashCode()方法返回的是一个散列值,而对于一个散列来说,不同的内容也是可能会出现相同的散列值的.所以即使两个对象的hashCode()返回的值