一、String类的基本特征
1、final的,不可被继承。public final class String
2、本质是字符数组char[],并且其值不可改变。private final char value[]
3、有个特殊的创建方式,就是直接指定String x = "abc","abc"就表示一个字符串对象。而x是"abc"对象的地址,也叫做"abc"对象的引用。
4、可以通过“+”串联,串联后会生成新的字符串,也可以通过concat()来串联。
5、Jvm会维护一个String Pool,用来存放运行时产生的各种字符串,并且池中的字符串的内容不重复。
最特殊的地方就是有2个地方可以存储,String pool和heap,不同的生成方式存储的地方也不一样。
二、String的创建原理
原理1:当使用任何方式来创建一个字符串对象s时,JVM会拿着这个s的值在pool中找是否存在内容相同的字符串对象,如果不存在,则在池中创建一个字符串s,否则,不在池中添加。
原理2:只要使用new关键字来创建对象,则一定会(在堆区或栈区)创建一个新的对象。
原理3:使用直接指定或者使用纯字符串串联来创建String对象,则仅仅会检查维护String池中的字符串,池中没有就在池中创建一个,有则罢了!但绝不会在堆栈区再去创建该String对象。
原理4:使用包含变量的表达式来创建String对象,则不仅会检查pool,还会在堆栈区创建一个对象。
----------------------
另外
1)intern()方法是一个本地方法,定义为public native String intern();
intern()方法的价值在于让开发者能将注意力集中到String池上。当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。
2)concat():Concatenates the specified string to the end of this string.
If the length of the argument string is 0, then this String object is returned. Otherwise, a new String object is created。当参数字符串长度为0,则返回对象本身;否则,堆中创建一个新的对象
三、比较字符串
// 在池中和堆中分别创建String对象"abc",s1指向堆中对象 String s1 = new String("abc"); // s2直接指向池中对象"abc" String s2 = "abc"; // 在堆中新创建"abc"对象,s3指向该对象 String s3 = new String("abc"); // 在池中创建对象"ab" 和 "c",并且s4指向池中对象"abc" String s4 = "ab" + "c"; // c指向池中对象"c" String c = "c"; // 在堆中创建新的对象"abc",并且s5指向该对象 String s5 = "ab" + c; String s6 = "ab".concat("c"); String s7 = "ab".concat(c); System.out.println("------------实串-----------"); System.out.println(s1 == s2); // false System.out.println(s1 == s3); // false System.out.println(s2 == s3); // false System.out.println(s2 == s4); // true System.out.println(s2 == s5); // false System.out.println(s2 == s6); // false System.out.println(s2 == s7); // false System.out.println("-------------------------"); String b1 = new String(""); String b2 = ""; String b3 = new String(""); String b4 = "".intern(); String b5 = "" + ""; String b6 = "".concat(""); String b7 = " ".trim(); String b8 = " "; String b9 = " ".trim(); System.out.println("------------空串-----------"); System.out.println(b1 == b2); // false System.out.println(b1 == b3); // false System.out.println(b2 == b3); // false System.out.println(b2 == b4); // true System.out.println(b2 == b5); // true System.out.println(b2 == b6); // true System.out.println(b2 == b7); // false System.out.println("-----a----"); System.out.println(b2.equals(b7)); // true* System.out.println(b7 == b8); // false System.out.println(b7 == b9); // false System.out.println(b7.equals(b9)); // true System.out.println(b9 == null);// false
四、字符串转码
1)转一个码,又用该码来构建一个字符串,是绝对不会出现乱码的,相当于没转。
2)转码与否,与字符串本身编码有关,字符串本身的编码与谁有关?文件编码,或者你的IDE设置的编码有关。文件已经是UTF-8了,你非要转为GBK,不乱才怪,呵呵
public class TestEncoding { public static void main(String[] args) throws UnsupportedEncodingException { System.out.println("转码前,输出Java系统属性如下:"); System.out .println("user.country:" + System.getProperty("user.country")); System.out.println("user.language:" + System.getProperty("user.language")); System.out.println("sun.jnu.encoding:" + System.getProperty("sun.jnu.encoding")); System.out.println("file.encoding:" + System.getProperty("file.encoding")); System.out.println("---------------"); String s = "哈哈"; String s1 = new String(s.getBytes(), "UTF-8"); String s2 = new String(s.getBytes("UTF-8"), "UTF-8"); String s3 = new String(s.getBytes("UTF-8")); String s4 = new String(s.getBytes("UTF-8"), "GBK"); String s5 = new String(s.getBytes("GBK")); String s6 = new String(s.getBytes("GBK"), "GBK"); System.out.println(s1); System.out.println(s2); System.out.println(s3); System.out.println(s4); System.out.println(s5); System.out.println(s6); } }
五、案例
通过内存监控发现,GC的动作比较频繁,偶然发现了大量如下代码:log.debug(“userId=” + user.getUserId())
原因分析:
以上代码执行时,分两步:
1. 先执行的是括号中的字符串相”+”的动作,而每次”+”运算都会导致新字符串的生成,这样就产生了很多“中间字符串”,在极大次数被调用时,这种字符串被创建和销毁的数量非常庞大,从而造成了jvm gc频繁执行,进而影响了性能。
2. 再执行log.debug()函数,在生产环境log level一般大于info,所以实际不会打印debug信息 综上所述,这些代码在生产环境不会产生日志,但会执行字符串”+”运算,而这些运算是无意义的,所以需要先判断日志的优先级,方式是log.isXXXEnabled() { log.XXX(……); }
Log Level的级别: Fatal->error->warn->info->debug,级别从高到低
一般我们生成环境的log level都是error,所以对于Error以上级别的日志,不用判断;对于error以下级别的都要加上判断。
1.可见性:为了确保多个线程之间的内存写入操作的可见性,必须使用同步机制。在没有使用同步的情况下,编译器,处理器以及运行时等都有可能对操作的执行顺序进行一些意想不到的调整。
2.加锁可见性: a,内置锁可以用于确保某个线程以一种可预见的方式来查看另外一个线程的执行结果。
b,java语言提供了一种稍弱的同步机制,即volatile变量,用来确保将变量的更新操作通知到其他线程,当把变量声明为volatile类型后,编译器与运行时都会注意到这个变量是共享的,因此不会将该变量上的操作与其他内存操作一起重排序。
3,线程封闭:当访问共享的可变数据时,通常需要使用同步,一种避免使用同步的方式就是不共享数据。如果仅在单线程内访问数据,就不需要使用同步,这种技术称为线程封闭。
ThreadLocal: ThreadLocal对象通常用于防止对可变的单实例变量(Singleleton)或全局变量进行共享。
4.不可变对象一定是线程安全的。不可变性并不等于将对象中的所有的域都申明为final类型,这个对象仍然是可变的。因为在final类型的域中可保存对可变对象的引用。
5.当满足以下条件时,对象才是不可变的:
对象创建以后其状态就不能改变
对象的所有域都是final类型
对象是正确创建的(在对象创建期间,this引用没有逸出)
6.任何线程都可以在不需要额外同步的情况下安全的访问不可变对象,即使在发布这些对象时没有使用同步。
7.安全发布的常用模式:可变对象必须通过安全的方式发布,这通常都意味着在发布和使用该对象的线程时都必须使用同步。
要安全的发布一个对象,对象的引用以及对象的状态必须同时对其他线程可见,一个正确构造的对象可以通过以下方式来安全发布。
a,在静态初始化函数中初始化一个对象引用
b,将对象的引用保存在volatile类型或者AtomicReferance对象中
c,将对象的引用保存到某个正确构造对象的final类型域中
d,将对象的引用保存到一个由锁保护的域中
常用的线程安全集合:HashTable,SynchronizedMap,ConcurrentMap,Vector,CopyONWriteArrayList,CopyOnWriteArrayList,SynchronizedList,SynchronizedSet,BlockingQueue,ConcurrentLinkedQueue
将某个元素加入到这些集合中,则可以将该元素安全发布到任何访问这些元素的线程中。
8.通常要发布一个静态构造的对象,最简和最安全的方式是使用静态的初始化器。
eg:public static Holder holder = new Holder(42);
静态初始化器由JVM在类的初始化阶段执行,由于在JVM内部存在着同步机制,因此通过这种方式初始化的任何对象都可以安全发布。
要想看或导出对方的QQ聊天记录,需要使用专门的黑客软件,就是找他们的QQ 357772576 许多人会说当然需要!其实并非如此,只要把QQ的主文件改一改,我们就可以无需其他黑客软件,也无需知道对方的qq密码,就可以在离线状态下登录别人的qq,查看他人的聊天记录,这回你的竞争对与MM聊些什么你就知道了
,哈哈。就是找他们的QQ 357772576
我们用来改qq主文件的软件是16进制文件编辑器Ultraedit,运行Ultraedit,点击“文件”菜单中的“打开”找到QQ安装目录下的qq.exe文件打开,单击“搜索”菜单下的“查找”,会弹出“查找”窗口,在“查找什么”菜单栏中输入:85c05f0f8489000000,单击“下一个”,按钮就会找到惟一的一处结果,将上述代码改为85c05fe98a00000090,然后保存修改结果。在离线状态下运行qq,在qq号码列表框中找到你想查看的号码,在“用户口令”栏中什么都不用输入,直接点击“登录”按钮就可以在离线状态下登录qq,然后就可以查看或导出他人的聊天记录了。注意,本技艺在qq200cbuild1230中通过。这个技巧在网吧或学校机房等公共场所使用最好,通过它你就可以随心所欲地查看任何人的聊天记录!对于使用qq2000cbuild0825版的朋友,可以查找:0f8564010000a1f0找到后改为e96501000090a1f0,然后保存修改结果即可。对于还在使用qq200cbuild0630版的qq的朋友,可以使用Ultraedit打开qq的主程序,查找8500f854c010000,找到后替换为85c0e94d01000090。运行qq,无需输入密码就可以进入qq了!
另外,在qq200c1230以后版本的qq安装目录中,以你的qq号码为文件夹名的目录下,有个名为ewh,db的文件,该文件中记录着本地登录时你的qq号码的加密密码,在偏移地址001eh到002dh的16个字节就是加密密码。只要用自己qq号码目录下的ewh.db文件最后4个字节以前的数据段,覆盖其他号的ewh.db文件中的相同的部分,你就可以用这个密码在本地查看他人的好友聊天记录了。更简单的方法是用你的qq号码下的ewh.db文件覆盖别人的这个文件,然后就可以用你的密码本地登录他人的qq了!
对于qq2000c1230以前的qq,可以用下面这一招,用Ultraedit打开我们自己号码目录下的user.db文件,然后将偏移地址0800h-0813h之间这20个字的内容覆盖到别人的qq号码的user.db文件中的相应位置,然后保存,重新打开qq,用自己密码登录刚才覆盖了其user.db文件的那个qq号码,怎么样,是不是通过了本地验证呢?