创建String字符串

String greeting = "JunSouth";

前言

Java虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。JVM主要管理堆和非堆两种类型内存
简单来说,非堆包含方法区、JVM内部处理或优化所需的内存(编译后的代码缓存)、每个类结构(如运行时常数池、字段和方法数据)以及方法和构造方法的代码。Java的堆是一个运行时数据区,类的对象从中分配空间。这些对象通过new、newarray等指令建立。因为它是在运行时动态分配内存的,Java的垃圾收集器会自动收走这些不再使用的数据。由于要在运行时动态分配内存,存取速度较慢。

栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放一些基本类型的变量数据(int, short, long, byte, float, double, boolean, char)

虚拟机必须为每个被装载的类型维护一个常量池。常量池就是该类型所用到常量的一个有序集合,包括直接常量(string,integer和 floating point常量)和对其他类型,字段和方法的符号引用。

对于String常量,它的值是在常量池中的。而JVM中的常量池在内存当中是以表的形式存在的,

对于String类型,有一张固定长度的CONSTANT_String_info表用来存储文字字符串值,该表只存储文字字符串值,不存储符号引用。在程序执行的时候,常量池会储存在Method Area,而不是堆中。常量池中保存着很多String对象; 并且可以被共享使用,因此它提高了效率。

常用String的方法

length()

求字符串的长度

indexOf()

求某个字符在字符串中的位置

charAt()

求一个字符串中某个位置的值

equals()

比较两个字符串是否相同

replace()

将字符串中的某些字符用别的字符替换掉。形如replace(“abc”,”ddd”);字符串中的abc将会被ddd替换掉

split()

根据给定正则表达式的匹配拆分此字符串。形如 String s = "The time is going quickly!"; str1=s.split(" ")

substring(int,int)

输出一个新的字符串,它是此字符串中的子串,形如substring(3,7)把字符串中的第四至六位输出

substring(int) 

从指定下标开始一直截取到字符串的最后

trim()

将字符串开头的空白(空格)和尾部的空白去掉

format()

使用指定的语言环境、格式字符串和参数返回一个格式化字符串

toLowerCase()

将字符串中所有的大写改变成小写

toUpperCase()

将字符串中所有的小写改变为大写

concat()

连接两个字符串

indexOf()

得到指定内容第一次出现的下标 

lastIndexOf()

得到指定内容最后一次出现的下标

valueOf() 

转换为字符串

tartsWith(String)endsWith(String)

判断一个字符串是否以指定的内容开头 boolean

判断一个字符串是否以指定的内容结尾

public static void main(String[] args) {
        String str1 = new String("abcdef");//初始化一个String对象
        System.out.println(str1.length());//输出字符串的长度
        System.out.println(str1.indexOf("c"));//输出指定字符在字符串中的位置
        System.out.println(str1.charAt(3));//输出指定位置的值
        String str2 = new String("JunSouth");
        System.out.println(str1.equals(str2));//比较两个字符串是否是相等的。
 
        String str3 = new String("abcdefghijk");
        System.out.println(str3);//输出str3的字符串
        System.out.println( str3.replace("def","fed"));//输出替换之后的字符串
 }
public static void main(String[] args) {
        //split()
        String[] str1;//声明一个空的数组
        String s = "The time is going quickly!";//声明一个字符串
        str1=s.split(" ");//根据给定正则表达式的匹配拆分字符串。
        for (int i=0;i<str1.length;i++){
            System.out.println(str1[i]);
        }
        str1 = s.split(" ",3);//根据匹配给定的正则表达式来拆分字符串。
        for (int i=0;i<str1.length;i++){
            System.out.println(str1[i]);
        }
        //substring()
        System.out.println(s.substring(4,7));//预计输出tim
        //trim()
        String ss = "  dssd   fkdj   ";
        System.out.println(ss.trim());//输出字符串前面的空白处
        //format
        int d = 2018;
        String sd = "Study Hard!";
        System.out.println(String.format("%d年加油,%s",d,sd));//将其他类型的数据和字符串按照指定的格式输出
        //toLowerCase
        //toUpperCase
        System.out.println(sd.toLowerCase());//将所有大写字母变为小写
        System.out.println(sd.toUpperCase());//将所有小写字母变为大写
}
public static void main(String[] args) {  
       /** 
        * 情景一:字符串池 
	* JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多String对象; 
	* 并且可以被共享使用,因此它提高了效率。 
	* 由于String类是final的,它的值一经创建就不可改变。 
	* 字符串池由String类维护,我们可以调用intern()方法来访问字符串池。  
	*/ 	
        //在字符串池创建了一个对象  
        String s1 = "abc";     
	//字符串pool已经存在对象“abc”(共享),所以创建0个对象,累计创建一个对象  
        String s2 = "abc";     		
        //true 指向同一个对象,
        System.out.println("s1 == s2 : "+(s1==s2));  		
        //true 值相等 
        System.out.println("s1.equals(s2) : " + (s1.equals(s2)));          
        
        /** 
         * 情景二:关于new String("")         
         */  		
	//创建了两个对象,一个存放在字符串池中,一个存在与堆区中;
        //还有一个对象引用s3存放在栈中
        String s3 = new String("abc"); 		
        //字符串池中已经存在“abc”对象,所以只在堆中创建了一个对象
        String s4 = new String("abc");		
        //false s3和s4栈区的地址不同,指向堆区的不同地址;
        System.out.println("s3 == s4 : "+(s3==s4));  		
        //true s3和s4的值相同
        System.out.println("s3.equals(s4) : "+(s3.equals(s4)));   
	//false 存放的地区多不同,一个栈区,一个堆区
        System.out.println("s1 == s3 : "+(s1==s3));  
	//true 值相同
        System.out.println("s1.equals(s3) : "+(s1.equals(s3)));   
	   	   
        /**
         * 情景三
         * 由于常量的值在编译的时候就被确定(优化)了。 
         * 在这里,"ab"和"cd"都是常量,因此变量str3的值在编译时就可以确定。 
         * 这行代码编译后的效果等同于: String str3 = "abcd"; 
         */
        String str1 = "ab" + "cd";  //1个对象  
        String str11 = "abcd";   
        System.out.println("str1 = str11 : "+ (str1 == str11));      

	/** 
         * 情景四:  
         * 局部变量str2,str3存储的是存储两个拘留字符串对象(intern字符串对象)的地址。 
         *  
         * 第三行代码原理(str2+str3): 
         * 运行期JVM首先会在堆中创建一个StringBuilder类, 
         * 同时用str2指向的拘留字符串对象完成初始化, 
         * 然后调用append方法完成对str3所指向的拘留字符串的合并, 
         * 接着调用StringBuilder的toString()方法在堆中创建一个String对象, 
         * 最后将刚生成的String对象的堆地址存放在局部变量str3中。 
         *  
         * 而str5存储的是字符串池中"abcd"所对应的拘留字符串对象的地址。 
         * str4与str5地址当然不一样了。 
         *  
         * 内存中实际上有五个字符串对象: 
         *       三个拘留字符串对象、一个String对象和一个StringBuilder对象。 
         */  
        String str2 = "ab";  //1个对象  
        String str3 = "cd";  //1个对象                                         
        String str4 = str2+str3;                                        
        String str5 = "abcd";    
	// false
        System.out.println("str4 = str5 : " + (str4==str5));   
       
	/** 
         * 情景五: 
         *  JAVA编译器对string + 基本类型/常量 是当成常量表达式直接求值来优化的。 
         *  运行期的两个string相加,会产生新的对象的,存储在堆(heap)中 
         */  
        String str6 = "b";  
        String str7 = "a" + str6;  
        String str67 = "ab";  
	//str6为变量,在运行期才会被解析。  
        System.out.println("str7 = str67 : "+ (str7 == str67));  
        
        final String str8 = "b";
        String str9 = "a" + str8;
        String str89 = "ab";
        //str8为常量变量,编译期会被优化        		
        System.out.println("str9 = str89 : "+ (str9 == str89));  
    }

 

数组

//声明数组
String [] arr;
int arr1[];
String[] array=new String[5];
int score[]=new int[3];
 
//初始化数组
//静态初始化
int arr2[]=new int[]{1,2,3,4,5};
String[] array1={"马超","马云","关羽","刘备","张飞"};
String[] array2=new String[]{"黄渤","张艺兴","孙红雷","小猪","牙哥","黄磊"};
int score[]=new int[3];
 
//动态初始化
for(int i=0;i<score.length;i++)
{
 score[i]=i+1; 
}
 
//查看数组长度
int length=array1.length;
System.out.println("length:"+array1.length);
 
//遍历数组
for (int i = 0; i < array1.length; i++) {
    System.out.println(array1[i]);
}
 
//int数组转成string数组
int[]  array3={1,2,3,4,5,6,7,8,9,0};
String arrStrings=Arrays.toString(array3);
System.out.println(arrStrings);
 
//从array中创建arraylist
ArrayList<String> arrayList=new ArrayList<String>(Arrays.asList(array1));
System.out.println(arrayList);
 
//从array中创建arraylist
ArrayList<String> arrayList=new ArrayList<String>(Arrays.asList(array1));
System.out.println(arrayList);
 
//数组中是否包含某一个值
String a="马超";
String[] array1={"马超","马云","关羽","刘备","张飞"};
if (Arrays.asList(array1).contains(a)) {
    System.out.println("马超在这里");
}
 
//将数组转成set集合
String[] array2=new String[]{"黄渤","张艺兴","孙红雷","小猪","牙哥","黄磊"};
Set<String> set=new HashSet<String>(Arrays.asList(array2));
System.out.println(set);
 
//将数组转成list集合
String[] array3=new String[]{"大黑","大白","大红","大紫","大蓝","大黄"};
//方法 1.
List<String> list=new ArrayList<String>();
for (int i = 0; i < array3.length; i++) {
    list.add(array3[i]);
}
String[] arrStrings2={"1","2","3"};
//方法 2.
List<String > list2=java.util.Arrays.asList(arrStrings2);
System.out.println(list2);
 
//Arrays.fill()填充数组
int[] arr3=new int[5];
//将数组全部填充10
Arrays.fill(arr3, 10);  
//遍历输出
for (int i = 0; i < arr3.length; i++) {
	System.out.println(arr3[i]);
}
 
//数组排序
//方法1
int[] arr4 = {3, 7, 2, 1, 9};
Arrays.sort(arr4);  //.sort(int[] a)  放入数组名字
for (int i = 0; i < arr4.length; i++) {
    System.out.println(arr4[i]);
}
//方法2.       
int[] arr5 = {3, 7, 2, 1, 9,3,45,7,8,8,3,2,65,34,5};
Arrays.sort(arr5, 1, 4);  //.sort(a, fromIndex, toIndex) 从第几个到第几个之间的进行排序
for (int i = 0; i < arr5.length; i++) {
    System.out.println(arr5[i]);
}
 
//数组拷贝(两个变量引用同一数组)。
//如果新数组过大,如果元素是数值型的会赋值0,如果元素是Boolean会赋值false。如果新数组过小,只拷贝前面的。
//方法1
int[] arr6 = {3, 7, 2, 1};
int[] arr7=Arrays.copyOf(arr6, 10);  //指定新数组的长度
//方法2        
int[] arr8=Arrays.copyOfRange(arr6, 1, 3); //只复制从索引[1]到索引[3]之间的元素(不包括索引[3]的元素)
for (int i = 0; i < arr8.length; i++) {
        System.out.println(arr8[i]);
}
//数组克隆
int [] kl1= {1,2,3,4,5,6,7,8,9,10};
int [] kl2= kl1.clone();
 
//比较两个数组
int[] arr9 = {1, 2, 3, 4,5,6,7,8,9,0};
boolean arr10=Arrays.equals(arr6, arr9);
System.out.println(arr10);
 
//去重复
int[] arr11 = {1, 2, 3, 4,5,6,7,8,9,0,3,2,4,5,6,7,4,32,2,1,1,4,6,3};
//利用set的特性
Set<Integer> set2=new HashSet<Integer>();
for (int i = 0; i < arr11.length; i++) {
    set2.add(arr11[i]);
}
System.out.println(set2);
int[] arr12 = new int[set2.size()];
int j=0;
for (Integer i:set2) {
    arr12[j++]=i;
}
System.out.println(Arrays.toString(arr12));
 
//查询数组中的最大值和最小值
int[] arr11 = {10, 2, 3, 4,5,6,7,8,9,0,3,2,4,5,6,7,4,32,2,1,1,4,6,3};
 //计算最大值
int max = arr11[0];
for (int i = 1; i < arr11.length; i++) {
     if (arr11[i] > max) {
         max = arr11[i];
    }
  }
  System.out.println("Max is " + max);
  //计算最小值
  int min = arr11[0];
  for (int i = 0; i < arr11.length; i++) {
    if (arr11[i]<min) {
        min = arr11[i];
    }
}
System.out.println("Min is " + min);

Iterator简介

迭代器是一种设计模式,可以遍历并选择序列中的对象,因为创建它的代价小,被称为"轻量级"对象。

主要方法

publicinterface Iterator<E> {
 boolean hasNext(); //判断是否存在下一个对象元素
 
 E next();         //指针下移,返回该指针所指向的元素
 
 void remove();   // remove 删除上次调用next方法返回时的元素 如果没有next的调用会报异常IllegalStateException
 
 forEachRemaining() //  Java8 增加的方法用到lambda遍历每个元素
 
}
  1. 当创建完成指向某个集合或者容器的Iterator对象是,这是的指针其实指向的是第一个元素的上方,即指向一个空元素。
  2. 当调用hasNext方法的时候,只是判断下一个元素的有无,并不移动指针。
  3. 当调用next方法的时候,向下移动指针,并且返回指针指向的元素,如果指针指向的内存中没有元素,会报异常。
  4. remove方法删除的元素是指针指向的元素。如果当前指针指向的内存中没有元素,那么会抛出异常。

使用方法

public static void main(String[] args) {
	List list=new ArrayList();
	Map map=new HashMap();
	for(int i=0;i<10;i++){
	list.add(new String("list"+i));
	   map.put(i, new String("map"+i));
	}
 
	Iterator iterList= list.iterator();//List接口实现了Iterable接口
	while(iterList.hasNext()){
	   String strList=(String)iterList.next();
	   System.out.println(strList.toString());
	}
 
	Iterator iterMap=map.entrySet().iterator();
	while(iterMap.hasNext()){
	   Map.Entry strMap=(Map.Entry)iterMap.next();
	   System.out.println(strMap.getValue());
        }
 
 
        ArrayList<String> aList=new ArrayList<String>();
	 aList.add("bbc");
	 aList.add("abc");
	 aList.add("ysc");
	 aList.add("saa");
	 System.out.println("移除前:"+aList);
	 Iterator<String> it=aList.iterator();
	 while(it.hasNext())
	 {
	    if("abc".equals(it.next())){
		it.remove();
	    }
	 }
	 System.out.println("移除后:"+aList);
}