ss

 

 

 

 

 

Java_Note

 

 

 

 

 

 

Index

 

第一章:编程基础 3-11

 

第二章:数组 11-31


 

第三章:面向对象程序开发

31-74


 

第四章:异常机制 74-89

 

第五章:多线程技术 89-122

 

第六章:常用类 API 122-139


 

第七章:集合框架(容器)+其他类对象使用

139-199


 

第八章:IO 

199-280


 

第九章:GUI 编程

280-284


 

第十章:网络编程 284-309

 

第十一章:反射机制 310-315

 

第十二章:正则表达式 315-322


 

第十三章:HTML 语言

322-335


 

第十四章:CSS 语言

335-341


 

第十五章:JavaScript 语言

341-375


 

第十六章:DOM

375-429


 

 

第一章:编程基础

 

 

1.Java 的特性和优势:简单性、面向对象、可移植性(write oncerun anywhere)、高性能、分

 

布式、动态性、多线程、安全性、健壮性。

 

2.Java 的运行机制,计算机高级编程语言类型有:编译型、解释型,Java 语言是这两种类型的结

 

合,利用编辑器编写 Java 源程序--源文件名:主类名.java,利用编译器(javac)将源程序编译

 

成字节码--字节码文件名:源文件名.class,再利用虚拟机(解释器,即 java)解释执行,运行过

 

程:载入、代码校验、解释执行。

 

3.Java 运行时环境 JRE 包含:Java 虚拟机、库函数、运行 Java 应用程序和 Applet 所必须的文件,

 

JDK(又称 SDK)包含:JRE 的超集、编译器、调试器等开发工具。

 

4.Java 运行环境的三项主要功能:加载代码:由 class loader 完成;代码校验:由 bytecode verifier

 

完成;执行代码:由 runtime interpreter 完成。

 

5.Java 发环境搭建:下载安装 JDK,新增系统环境变量 JAVA_HOME,设置就是 JDK 的安装路径,

 

修改系统环境变量 Path,在最前面加上%JAVA_HOME%\bin,并以分号;和原路径分隔,再在最前面

 

加上.表示当前目录的意思,完整的就是.;%JAVA_HOME%\bin;

 

6.注释:就是程序员为读者作的说明,是提高程序可读性的一种手段,Java 中有三种注释,即单

 

行注释(//)、多行注释(/**/)、文档注释,/**/不能被嵌套,并且应当知道注释不会出现在字节

 

码文件中,即注释不会影响到字节码文件的大小,Java 编译器编译时也会跳过注释语句。

 

7.标识符:用作给变量、类、方法命名,但应注意表示类名的标识符每个单词的首字母都要大写,

 

方法和变量的标识符用小写字母开头,后面的描述性词以大写开头,即从第二个单词起每个单词的

 

首字母要大写,并且 Java 的标识符必须以字母、下划线_、美元符$开头,不能以数字开头,标识

 

符除开头外的其他部分可以是字母、下划线_、美元符$、以及数字的任意组合,Java 标识符大小

 

写敏感,长度无限制,不能是 Java 中的关键字。注意:Java 不采用通常语言使用的 ASCII 字符集,

 

而是采用 unicode 这样的标准的国际字符集,所以这里的字母的含义:英文、汉字等等,但尽量

 

不要用汉字来定义标识符,更不能用关键字来作为变量名和方法名,但可以是关键字与其他单词的

 

 

 

 

 

组合形式。

 

8.变量:通过变量来操纵存储空间中的数据,变量就指代这个存储空间!空间位置是确定的,但是

 

里面放置什么值不确定,Java 是一种强类型的语言,每个变量都必须在它本身前面声明其类型才

 

能被使用并为其分配相应长度的存储单元,Java 变量是程序中最基本的存储单元,其要素包括变

 

量名、变量类型和作用域。注意:每个变量都有类型,类型可以是基本数据类型也可以是引用数据

 

类型,变量必须是合法的标识符,变量声明是一条完整的语句,因此每一个声明都必须以分号";"

 

结束。

 

9.变量可以分为:①局部变量,即定义在方法或语句块内部的变量,生命周期是从声明位置到"}"

 

止,并且局部变量在使用前必须先声明其类型和初始化(赋初值);②成员变量,又称实例变量,即

 

定义在方法外部、类的内部的变量,从属于对象,生命周期伴随对象始织,如果不自行初始化,他

 

会自动初始化成该类型的默认初始值(数值型变量初始化成 或 0.0,字符型变量的初始化值是

 

16 位的 0,布尔类型默认是 false);③静态变量,又称类变量,即被 static 修饰,用 static 定义,

 

从属于类,生命周期伴随类始织,从类加载到卸载,如果不自行初始化,他会自动初始化成该类型

 

的默认初始值(数值类型的初始化成 或 0.0,字符型的初始化成默认的 16 位的 0,布尔类型默

 

认是 false)。

 

10.常量:即初始化后不能再改变的值!

 

11.命名规则(规范):所有变量、方法、类名,都要见名知意;类成员变量要首字母小写和驼峰原

 

则;局部变量也要首字母小写和驼峰原则;常量的单词字母要全部大写,若有两个以上的单词组成,

 

就用下划线"_"进行连接,如 MAX_VALUE;类名要首字母大写和驼峰原则;方法名要首字母小写和驼

 

峰原则,如 run()showRun()

 

12.Java 的数据类型分为两大类,即基本数据类型和引用数据类型,在基本数据类型中有 类 

 

基本数据类型(逻辑型-boolean;文本型-char;数值型-byteshortintlongfloatdouble),

 

引用数据类型有类(class)、接口(interface)、数组(array),注意,引用数据类型的大小统一

 

 

 

4


 

 

 

 

为 个字节,记录的是其引用对象的地址!

 

13.Byte 类型(字节)、short字节)、int字节)、long字节)、float字节)、double

 

字节)。

 

14.字符型(个字节),单引号来表示字符常量,例如'A'是一个字符,它不"A"不同的,"A"

 

示含有一个字符的字符串;char 类型用来表示在 unicode 编码表中的字符;unicode 编码被设计

 

用来处理各种语言的所有文字,它占 个字节,可允许有 65536 个字符,ASCII 码占 个字节,

 

可允许有 128 个字符,是 unicode 编码表中前 128 个字符;unicode 具有从 到 65535 乊间的

 

编码,他们通常从'\u0000''\uFFFF'乊间的 16 进制值来表示(前缀 表示 unicode);Java 

 

言还允许使用转义字符'\'来将其后的字符转发为其它的含义,如\b 代表退格、\n 代表换行、\r 

 

表回车、\t 代表制表符(tab)、\"代表双引号、\'代表单引号、\\代表反斜杠。

 

方法,Java的方法类似于其他语言的函数,是一段用来完成特定功能的代码片段,声明格式:[修饰


 

1

修饰符2

. . .]

返回值类型

方法名(形式参数列表){


Java语句;

. . .

}

 

形式参数,即在方法被调用时用于接收外界输入的数据;实参,调用方法时实际传给方法的数据;

 

返回值,方法在执行完毕后返还给调用它的环境的数据;返回值类型,事先约定的返回值的数据类

 

型,如无返回值,必须给出返回值类型 void;调用方法的格式为对象名.方法名(实参列表);实

 

参的数目、数据类型和次序必须和所调用的方法声明的形参列表匹配;return 语句织止方法的运

 

行并指定要返回的数据;Java 进行方法调用传递参数时,要遵循的原则是,基本数据类型传递

 

的是该数据值本身,引用数据类型传递的是对对象的引用,而不是对象本身;Java 只有值传递!

 

16.简单的键盘输入和输出:

 

Scanner scanner=new Scanner(System.in);


//

//

//

//

//

将输入的第一行赋给string

String string=scanner.nextLine();

将输入单词到第一个空白符为止的字符串赋给string

String string=scanner.next();

将输入的数字赋给变量

 

 

 

 

 

 

 

5


 

 

 

int string=scanner.nextInt();

System.out.println(string);

 

17.for 循环和 while 循环的特点:1,for 和 while 可以互换;2,格式上的不同,在使用上有点小区

 

别,即当需要通过变量来对循环进行控制,该变量只作为循环增量存在时,区别就体现出来了,在

 

for 循环中该变量随着循环结束而从内存消亡,但在 while 循环中却可以继续存在并使用,也就是

 

还在内存中并未消亡。

 

18.无限循环最简单的形式:while(true){ }和 for( ; ; ){ }。

 

19.什么时候使用循环结构呢?当对某些代码执行很多次时,使用循环结构完成,当对一个条件进

 

行一次判断时,可以使用 if 语句,当对一个条件进行多次判断时,可以使用 while 语句,注意:

 

在使用循环时,一定要明确哪些语句需要参不循环,哪些不需要,循环通常情况下,需要定义条件,

 

需要控制次数。

 

20.If语句的三种格式:

 

if(条件表达式)

{

执行语句;

          }

 

 

if(条件表达式)

{

执行语句;

}

else//否则

{

执行语句;

}

 

 

if(条件表达式)

{

执行语句;

}

else if (条件表达式)

{

执行语句;

}

„„

else

{

 

6


 

 

 

执行语句;

}

 

 

21.三元运算符就是if else 语句的简写格式,简写格式什么时候用?当if else语句运算后,有一个

 

具体的结果时,可以简化写成三元运算符形式。

 

22.局部代码块可以定义局部变量的生命周期,如:

 

{ //局部代码块可以定义局部变量的生命周期。

int a = 3;

//a 的运算。

System.out.println(a+4);

}

变量a的作用域只是在声明位置起到"}"为止。

23.ifswitch的应用,if:1,对具体的值进行判断;2,对区间判断;3,对运算结果是boolean类型的

 

表达式进行判断;switch:1,对具体的值进行判断;2,值的个数通常是固定的,对于几个固定的值判

 

断,建议使用switch语句,因为switch语句会将具体的答案都加载进内存,效率相对高一点。

 

24.比较运算符,运算完的结果必须是true或者false,如:

System.out.println(3>2);//true

System.out.println(3==2);//false

 

25.逻辑运算符有什么用?用于连接两个boolean类型的表达式,&:不,|:或。

 

&:符号的运算特点是:

true & true = true;

true & false = false;

false & true = false;

false & false = false;

&:运算规律:

&运算的两边只有有一个是false,结果肯定是false

只有两边都为true,结果才是true

|:运算特点:

true | true = true;

true | false = true;

false | true = true;

false | false = false;

|:运算规律:

|运算的两边只要有一个是true,结果肯定是true

只有两边都为false,结果才是false

^:或,和或有点不一样,^:运算特点是:

 

7


 

 

 

true ^ true = false;

true ^ false = true;

false ^ true = true;

false ^ false = false;

^异或的运算规律:

^符号的两边结果如果相同,结果是false

两边的结果不同,结果是true

!:非运算,判断事物的另一面,案例:

!true = false

!false = true;

!!true = true;

 

两个或和两个不比较:

&&:

&运算的结果是一样的。但是运算过程有点小区别,

&:无论左边的运算结果是什么,右边都参与运算;

&&:当左边为false时,右边不参与运算。

||:

|运算的结果是一样的,但是运算过程有点小区别,

|:无论左边的运算结果是什么,右边都参与运算。

||:当左边为true时,右边不参与运算。

 

26.经典小面试题:

 

class OperateTest

{

public static void main(String[] args)

{

//最有效率的方式算出2乘以8等于几?


 

 

 

//

 

 

//

 

 

System.out.println(2<<3); //位运算

 

 

对两个整数变量的值进行互换 (不需要第三方变量)

int a = 3,b = 5;

System.out.println("a="+a+",b="+b);

 

 

/*

开发时,使用第三方变量的形式,因为阅读性强。

int c ;

c = a;

a = b;

b = c;

*/

 

 

//这种方式不要用,如果两个整数的数值过大,会超出int范围,会强制转换。数据会变化。

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

8


 

 

 

/*

a = a + b; //a = 3 + 5;a = 8;

b = a - b; //3+5-5 = 3;b = 3;

a = a - b; //3+5-3 = 5;a = 5;

*/

 

 

/*

面试的时候用。

a = a ^ b; //a = 3 ^ 5;

b = a ^ b; //b = (3^5)^5; b = 3;

a = a ^ b; //a = (3^5)^3; a = 5;

*/

System.out.println("a="+a+",b="+b);

}

}

 

27.或运算中,当一个数异或运算同一个数两次,结果还是这个数本身,如a^b^b=a。

 

28.逻辑结构常见练习:

 

package cn.com.b;

 

 

public class AppMain {

 

 

public static void main(String[] args) {

AppMain appMain=new AppMain();

appMain.addFactorial();

}

 

 

/**

* 用while循环分别计算100以内的奇数的和、偶数的和

*/

void caculate(){

int sumOdd=0;//奇数的和

int sumEven=0;//偶数的和

for(int i=0;i<100;i++){

if(i%2==0){

sumEven+=i;

}else{

sumOdd+=i;

}

}

System.out.println("100之内偶数的和:"+sumEven);

System.out.println("100之内奇数的和:"+sumOdd);

}

 

 

/* 用循环结构输出1000之内所有被5整除

 

9


 

 

 

* 的数,并且每行最多输出3

*

*/

 

 

void test2(){

int i=0;

int j=0;

while(i<1000){

if(i%5==0){

System.out.println(i+"\t");

j++;

if(j%3==0){

System.out.println("\n");

j=0;

}

}

i++;

}

}

 

 

/**

* 输出九九乘法表

*

*/

 

 

void multiplicationTable(){

for (int i = 0; i < 10; i++) { //i是一个乘数

for (int j = 0; j <=i; j++) { //j是另外一个乘数

System.out.print(j+"*"+i+"="+(i*j<10?(" "+i*j):i*j)+" ");

}

System.out.println();

}

}

 

 

/**

* 编程求:1+1+2+1+2+3+. . .+1+2+3+. . .+100

*/

 

 

void addAdd(){

int sum=0;//总和

for(int i=1;i<=100;i++){

int tempSum=0;//临时和

for(int j=1;j<=i;j++){

tempSum+=j;

}

sum+=tempSum;

}

 

10


 

 

 

System.out.println(sum);

}

 

 

/**

* 编程求:1+2+3+4+. . .+15

*

*/

 

 

private void addFactorial() {

long result=0;

for(int i=1;i<=15;i++){

int temp=1;

for(int j=1;j<=i;j++){

temp*=j;

}

result+=temp;

}

System.out.println(result);

}

 

 

}

 

第二章:数组

 

29.内存的划分:

 

1,寄存器。

2,本地方法区。

3,方法区。

4,栈内存。

存储的都是局部变量,而且变量所属的作用域一旦结束,该变量就自动释放。

5,堆内存。

存储是数组和对象(其实数组就是对象) 凡是new建立在堆中。

特点:

1,每一个实体都有首地址值。

2,堆内存中的每一个变量都有默认初始化值,根据类型的不同而不同。整数是0,小数0.0或者0.0f

boolean false char '\u0000'

3,垃圾回收机制。

 

30.数组中常见的两个异常:

 

System.out.println(arr[3]);//ArrayIndexOutOfBoundsException:

//当访问到数组中不存在的角标时,就会发生该异常。

arr = null;

System.out.println(arr[0]);//NullPointerException

//当引用型变量没有任何实体指向时,还在用其操作实体。就会发生该异常。

11


 

 

 

 

31.breakcontinue

 

break:跳出。

break作用的范围:要么是switch语句,要么是循环语句。

记住:当break语句单独存在时,下面不要定义其他语句,因为执行不到。

break跳出所在的当前循环。

如果出现了循环嵌套,break想要跳出指定的循环,可以通过标号来完成。

continue:继续。

作用的范围:循环结构。

continue:结束本次循环,继续下次循环。

如果continue单独存在时,下面不要有任何语句,因为执行不到。

 

32.九九乘法表:

 

/*

需求:

打印一个99乘法表。

 

 

思路:

1,定义一个类。

2,为了可以运行,该类中定义一个main函数。

3,将99惩罚表的代码定义在main函数中。

 

 

步骤:

1,用class关键字定义名称Demo99为的类。

2,定义main函数。

3

1*1=1

1*2=2 2*2=4

1*3=3 2*3=6 3*3=9

分析一下99乘法表。

*/

class Demo99

{

public static void main(String[] args)

{

/*

99乘法表的分析:

。。。

符合了大圈套小圈的思想。

就是要使用 for嵌套循环。

*/

//int z = 1;

for (int x = 1;x<=9 ;x++ )

{

 

 

12


 

 

 

for (int y=1; y<=x ; y++)//

{

System.out.print(y+"*"+x+"="+y*x+"\t");

}

System.out.println();//


 

 

//

 

z++;


}

 

 

}

}

 

33.输出一个由星*组成的正方形:

 

class ForForDemo

{

public static void main(String[] args)

{

//大圈套小圈思想。

/*

for(int x=0; x<3; x++)

{

for(int y=0; y<4; y++)

{

System.out.println("ok");

}

}

*/

 

 

/*

 

 

*****

*****

*****

*****

 

 

*/

for(int x=0; x<4; x++)//外循环控制的是行数

{

for(int y=0; y<5; y++)//内循环控制的是每一行的个数

{

System.out.print("*");

}

System.out.println();

}

 

 

 

 

13


 

 

 

 

}

}

 

34.根据"大圈套小圈"思想输出不同的图形:

 

class ForForTest

{

public static void main(String[] args)

{

 

 

/*

 

 

*****

****

***

**

*

 

 

*/

 

 

/*

int z = 5;

for (int x = 1; x<=5 ;x++ )//1-5 1-4 1-3//1-5 2-5 3-5

{

for (int y=1; y<=z ; y++ )

{

System.out.print("*");

}

System.out.println();

z--;

}

*/

/*

 

 

int z = 1;

for (int x = 1; x<=5 ;x++ )//1-5 1-4 1-3//1-5 2-5 3-5

{

for (int y=z; y<=5 ; y++ )

{

System.out.print("$");

}

System.out.println();

z++;

}

*/

for(int x=1; x<=5; x++)

 

14


 

 

 

{

for(int y=x; y<=5; y++)

{

System.out.print("*");

}

System.out.println();

}

 

 

/*

 

 

*

**

***

****

*****

*/

System.out.println("----------");

for (int x=1; x<=5 ;x++ )

{

for (int y=1;y<=x ;y++ )

{

System.out.print("*");

}

System.out.println();

}

 

 

 

System.out.println("----------");

/*

 

 

54321

5432

543

54

5

 

 

 

*/

for (int x=1; x<=5; x++ )

{

for (int y=5; y>=x ;y-- )

{

System.out.print(y);

}

System.out.println();

}

 

 

15


 

 

 

System.out.println("----------");

/*

1

22

333

4444

55555

 

 

*/

 

 

for (int x=1; x<=5 ; x++)

{

for (int y=1;y<=x ;y++ )

{

System.out.print(x);

}

System.out.println();

}

}

}

 

35.for嵌套循环"大圈套小圈"思想进一步运用:

 

class ForForTest2

{

public static void main(String[] args)

{

 

 

/*

 

 

* * * * *

-* * * *

--* * *

---* *

----*

 

 

*/

for(int x=1; x<=5; x++)

{

for(int y=1; y

{

System.out.print(" ");

}

 

 

for(int z=x; z<=5; z++)

{

System.out.print("* ");

 

16


 

 

 

}

 

 

System.out.println();

}

 

 

 

/*

 

 

*

* *

* * *

* * * *

* * * * *

 

 

*/

}

}

 

36.函数的定义格式:

 

注:什么是函数?

函数就是定义在类中的具有特定功能的一段独立小程序

函数也称为方法

 

 

修饰符 返回值类型 函数名(参数类型 形式参数1,参数类型 形式参数2...)

{

执行语句;

return 返回值;

}

 

 

特殊情况:

功能没有具体的返回值,这时return的后面直接用分号结束。

返回值类型怎么体现呢?因为没有具体值,所以不可以写具体的数据类型,在java中只能用一个关键字来

表示这种情况,关键字是:void

总结:没有具体返回值时,返回值类型用void来表示。

注意:如果返回值类型是void,那么函数中的return语句可以省略不写。

 

37.函数的重载:

 

1,同一个类,

2,同名。

3,参数个数不同or 参数类型不同、顺序不同。

4,函数重载和返回值类型无关。

5java是严谨性语言,如果函数出现的调用的不确定性,会编译失败。

38.数组时常用容器乊一,对数组操作最基本的动作就是存和叏,核心思想:就是对角标的操作。

 

39.获叏数组最值和选择、冒泡排序:

 

import java.util.*;

 

 

17


 

 

 

class ArrayDemo4

{

 

 

//遍历数组的功能。

public static void printArray(int[] arr)

{

System.out.print("[");

for(int x=0; x

{

if(x!=arr.length-1)

System.out.print(arr[x]+", ");

else

System.out.println(arr[x]+"]");

}

}

public static void main(String[] args)

{

int[] arr = {34,19,11,109,3,56};


 

 

//

//

 

 

 

 

 

//

//

//

 

int max = getMax_2(arr);

System.out.println("max="+max);

 

 

printArray(arr);

 

 

selectSort(arr);

bubbleSort(arr);

Arrays.sort(arr);

selectSort_2(arr);

 

 

printArray(arr);

}

 

 

public static void swap(int[] arr,int a,int b)

{

int temp = arr[a];

arr[a] = arr[b];

arr[b] = temp;

}

/*

冒泡排序。

*/

public static void bubbleSort(int[] arr)

{

for(int x=0; x

{

for(int y=0; y

{

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

18


 

 

 

if(arr[y]>arr[y+1])

{

swap(arr,y,y+1);

/*

int temp = arr[y];

arr[y] = arr[y+1];

arr[y+1] = temp;

*/

}

}

}

}

 

 

/*

选择排序。

*/

public static void selectSort(int[] arr)

{

for(int x=0; x

{

for(int y=x+1; y

{

if(arr[x]>arr[y])

{

swap(arr,x,y);

/*

int temp = arr[x];

arr[x] = arr[y];

arr[y] = temp;

*/

}

}

}

}

 

 

public static void selectSort_2(int[] arr)

{

for(int x=0; x

{

int num = arr[x];

int index = x;

for(int y=x+1; y

{

if(num>arr[y])

{

num = arr[y];

index = y;

 

19


 

 

 

}

}

if(index!=x)

swap(arr,x,index);

}

}

 

 

/*

获取数组中的最大值。

思路:

1,需要进行比较。并定义变量记录住每次比较后较大的值。

2,对数组中的元素进行遍历取出,和变量中记录的元素进行比较。

如果遍历到的元素大于变量中记录的元素,就用变量记录住该大的值。

3,遍历结果,该变量记录就是最大值。

 

 

定义一个功能来是实现。

明确一,结果。

是数组中的元素。int .

明确二,未知内容。

数组.

*/

public static int getMax(int[] arr)

{

//定义变量记录较大的值。

int maxElement = arr[0];//初始化为数组中的任意一个元素。

for(int x=1; x

{

if(arr[x]>maxElement)

maxElement = arr[x];

}

return maxElement;

}

public static int getMax_2(int[] arr)

{

//定义变量记录较大的值。

int maxIndex = 0;//初始化为数组中任意一个角标。

for(int x=1; x

{

if(arr[x]>arr[maxIndex])

maxIndex = x;

}

return arr[maxIndex];

}

}

 

40.二分法查找及普通查找方法运用:

 

20


 

 

 

import java.util.*;

/*

面试题:

给定一个有序的数组,如果往该数组中存储一个元素,并保证这个数组还是有序的,

那么个元素的存储的角标为如何获取。

{13,15,19,28,33,45,78,106};

*/

class ArrayDemo5

{

public static void main(String[] args)

{

//


//

int[] arr = {4,1,8,7,3,8,2};

int[] arr = {13,15,19,28,33,45,78,106};

int index = halfSearch_2(arr,5);

System.out.println("index="+index);

 

 

int index1 = Arrays.binarySearch(arr,5);//如果存在返回的具体的角标位置,不存在返

回的是 -插入点-1

System.out.println("index1="+index1);

}

/*

二分查找法。

 

*/


public

{

int

min

max

mid

static int halfSearch(int[] arr,int key)

 

 

max,min,mid;

= 0;

= arr.length-1;

= (max+min)/2;


 

 

while(arr[mid]!=key)

{

if(key>arr[mid])

min = mid + 1;

else if(key

max = mid - 1;

 

 

if(max

return -1;

 

 

mid = (max+min)/2;

}

return mid;

 

 

}

 

21


 

 

 

 

public

{

int

min

max

 

 

 

 

static int halfSearch_2(int[] arr,int key)

 

 

max,min,mid;

= 0;

= arr.length-1;


 

 

while(min<=max)

{

mid = (max+min)>>1;

 

 

if(key>arr[mid])

min = mid + 1;

else if(key

max = mid - 1;

else

return mid;

}

return -min-1;

}

 

 

 

/*

数组常见功能:查找。

*/

public static int getIndex(int[] arr,int key)

{

for(int x=0; x

{

if(arr[x]==key)

return x;

}

return -1;

}

}

 

41.将数组进行反转:

 

/*

给定一个数组,对其进行反转。

 

{3,1,6,5,8,2} -->

{2,8,5,6,1,3};

 

 

其实就是头尾元素的位置置换。

 

*/

 

22


 

 

 

 

class ArrayTest

{

public static void printArray(int[] arr)

{

System.out.print("[");

for(int x=0; x

{

if(x!=arr.length-1)

System.out.print(arr[x]+", ");

else

System.out.println(arr[x]+"]");

}

}

public static void main(String[] args)

{

int[] arr = {4,1,8,7,3,8,2};

printArray(arr);

reverseArray(arr);

printArray(arr);

}

 

 

public static void reverseArray(int[] arr)

{

for(int start=0,end=arr.length-1; start

{

swap(arr,start,end);

}

}

public static void swap(int[] arr,int a,int b)

{

int temp = arr[a];

arr[a] = arr[b];

arr[b] = temp;

}

}

 

42.数组中查表法的使用:

 

/*

获取一个整数的16进制表现形式。

*/

 

 

class ArrayTest2

{

public static void main(String[] args)

{

 

23


 

 

 

toHex_2(0);

}

 

 

// 0,1,2,3,4,5,6,7,8,9,A, B, C, D, E, F

// 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15

 

 

/*

什么时候使用数组呢?

如果数据出现了对应关系,而且对应关系的一方是有序的数字编号。并作为角标使用。

这时就必须要想到数组的使用。

 

 

就可以将这些数据存储到数组中。

根据运算的结果作为角标直接去查数组中对应的元素即可。

 

 

这种方式:称为查表法。

 

 

 

 

 

 

*/

 

 

public static void toHex_2(int num)

{

 

 

if(num==0)

{

System.out.println("0");

return ;

}

//定义一个对应关系表。

char[] chs = { '0','1','2','3',

'4','5','6','7',

'8','9','A','B',

'C','D','E','F'};

/*

一会查表会查到比较的数据。

数据一多,就先存储起来,在进行操作。

所以定义一个数组。 临时容器。

*/

char[] arr = new char[8];

int pos = arr.length;

 

 

while(num!=0)

{

int temp = num&15;

arr[--pos] = chs[temp];

num = num >>> 4;

 

24


 

 

 

}

 

 

System.out.println("pos="+pos);

for(int x=pos ;x

{

System.out.print(arr[x]);

}

 

 

}

 

 

public static void toHex_1(int num)

{

//定义一个对应关系表。

char[] chs = { '0','1','2','3',

'4','5','6','7',

'8','9','A','B',

'C','D','E','F'};

 

 

for(int x=0 ; x<8; x++)

{

int temp = num & 15;

System.out.print(chs[temp]);

num = num >>> 4;

}

 

 

}

 

 

public static void toHex(int num)

{

 

 

for(int x=0; x<8; x++)

{

int temp = num & 15;

if(temp>9)

System.out.print((char)(temp-10+'A'));

else

System.out.print(temp);

num = num >>> 4;

}

/*

int n1 = num & 15;

System.out.println("n1="+n1);

 

 

num = num >>> 4;

int n2 = num & 15;

System.out.println("n2="+n2);

*/

 

25


 

 

 

}

}

 

43.查表运用:

 

class ArrayTest3

{

public static void main(String[] args)

{


//

 

 

//

toHex(26);

toBinary(-6);

toOctal(26);

System.out.println(Integer.toBinaryString(-6));

}

 

 

//十进制-->十六进制。

public static void toHex(int num)

{

trans(num,15,4);

}

//十进制-->二进制。

public static void toBinary(int num)

{

trans(num,1,1);

}

//十进制-->八进制。

public static void toOctal(int num)

{

trans(num,7,3);

}

 

 

public static void trans(int num,int base,int offset)

{

 

 

if(num==0)

{

System.out.println("0");

return ;

}

//定义一个对应关系表。

char[] chs = { '0','1','2','3',

'4','5','6','7',

'8','9','A','B',

'C','D','E','F'};

/*

一会查表会查到比较的数据。

数据一多,就先存储起来,在进行操作。

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

26


 

 

 

所以定义一个数组。 临时容器。

*/

char[] arr = new char[32];

int pos = arr.length;

 

 

while(num!=0)

{

int temp = num & base;

arr[--pos] = chs[temp];

num = num >>> offset;

}

 

 

for(int x=pos ;x

{

System.out.print(arr[x]);

}

System.out.println();

 

 

}

 

 

}

 

44.查表运用(数组容器):

 

class ArrayTest4

{

public static void main(String[] args)

{

String week = getWeek(71);

System.out.println(week);

}

/*

使用查表法。

星期。

String s = "abc";

int x = 4;

*/

public static String getWeek(int num)

{

 

 

if(num>7 || num<1)

{

return "错误的星期";

}

String[] weeks = { "","星期一","星期二","星期三","星期四","星期五","星期六","星期日

"};

 

 

27


 

 

 

return weeks[num];

}

 

 

}

 

45.二维数组:

 

/*

二维数组定义的格式。

*/

 

 

class Array2Demo

{

public static void main(String[] args)

{


 

 

//

//

 

 

 

//

 

int[] arr = new int[3];

System.out.println(arr);//[I@1fb8ee3 @左边是实体的类型。 @右边是实体的哈希值。

 

 

 

int[][] arr = new int[3][2];//创建一个二维数组,该数组中有3个一维数组,每一个一维数


组中有2个元素。


//

System.out.println(arr);//直接打印二维数组。

[[I@c17164


//

//

素。 0

 

 

 

//

System.out.println(arr[0]);//直接打印二维数组中的角标0的一维数组。 [I@1fb8ee3

System.out.println(arr[0][0]);//直接打印二维数组中的角标0的一维数组中角标为0的元

 

 

 

 

 

int[][] arr = new int[3][];


//

System.out.println(arr);//直接打印二维数组。

[[I@c17164


//

//

System.out.println(arr[0]);//直接打印二维数组中的角标0的一维数组。null

System.out.println(arr[0][0]);//直接打印二维数组中的角标0的一维数组中角标为0的元


素。 NullPointerException


 

 

 

//

//

//

 

 

 

 

 

 

 

 

 

 

 

 

//

 

 

int[][] arr = new int[3][2];

System.out.println(arr.length);//打印二维数组的长度。其实就是一维数组的个数。

System.out.println(arr[1].length);//打印二维数组中角标为1一维数组的长度。

 

int sum = 0;

int[][] arr = { {3,1,7},{5,8,2,9},{4,1}};

 

 

for(int x=0; x

{

for(int y=0; y

{

System.out.print(arr[x][y]+",");

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

28


 

 

 

sum += arr[x][y];

 

 

}

}

System.out.println("sum="+sum);


 

 

 

//

//

//

 

 

甲:30 59 28 17

乙;37 60 22 19

int[] arr = { {30,59,28,17},{37,60,22,19}};


 

 

int[][][] arr = new int[3][2][4];

}

}

 

 

/*

int[] x,y[];

int[] x;

int[] y[];

 

 

a

x = y;

 

 

b

x = y[0];

 

 

c

x[0] = y[0];

 

 

d

x[0] = y[0][0];

 

 

e

x[0] = y;

*/

 

46.数组综述:

 

数组的定义:数组是相同类型数据的集合,描述的是相同类型的若干个数据按照一定的先后顺序排

 

列组合而成,其中每一个数据称作一个数组元素,每个数组元素可以通过一个下标来访问它们。

 

数组的四个基本特点:1.长度固定,一旦被创建它的长度就是不可改发的;2.其元素类型必须是相

 

同类型,不允许出现混合类型;3.数组中的元素可以是任何数据类型,包括基本数据类型和引用数

 

 

 

29


 

 

 

 

据类型;4.数组变量属于引用类型,数组也可以看做是对象,数组中的每个元素相当于该对象的成

 

员变量,数组本身就是对象,Java中对象是在堆中的,因此数组无论保存原始类型还是其他对象类

 

型,数组本身是在堆中的。

 

数组的下标合法区间是[0,length-1]

 

数组的拷贝:System.arrayCopy(源数组,从哪开始,目标数组,从哪开始贴,粘几个)。

 

数组排序:Arrays.sort(被排序的数组)。

 

二分法查找:Arrays.binarySearch(哪个数组,数组中的什么元素)。

 

填充:Arrays.filla, 2, 4, 100)。//将数组a24的索引的元素替换为100

 

获叏数组的长度:数组.length,如a.length,获叏数组a的元素个数;a[0].length,表示获叏二维

 

数组中第一个数组的长度。

 

数组的遍历:可以使用for循环或者for嵌套循环(对于二维数组),也可以使用增强for循环来对数

 

组进行遍历,增强for循环格式:for(变量类型 变量名:被遍历的数组)

 

第三章:面向对象程序开发

 

47.面向对象简述:

 

 

/*

java语言对现实生活中的事物进行描述。

通过类的形式来体现的。

 

 

怎么描述呢?

对于事物描述通常只关注两方面。

一个是属性,一个是行为。

 

 

只要明确该事物的属性和行为并定义在类中即可。

 

 

对象:其实就是该类事物实实在在存在的个体。

 

 

类与对象之间的关系?

类:事物的描述。

对象:该类事物的实例。在java中通过new来创建的。

 

 

30


 

 

 

*/

 

 

 

/*

 

 

描述小汽车

 

 

分析:

1,属性。

轮胎数。

颜色。

2,行为。

运行。

 

 

 

 

 

定义类其实就是在定义类中的成员。

成员:成员变量<-->属性,成员函数<-->行为。

 

 

 

成员变量和局部变量的区别:

 

 

1

成员变量定义在类中,整个类中都可以访问。

局部变量定义在函数,语句,局部代码块中,只在所属的区域有效。

 

 

2

成员变量存在于堆内存的对象中。

局部变量存在于栈内存的方法中。

 

 

3

成员变量随着对象的创建而存在,随着对象的消失而消失。

局部变量随着所属区域的执行而存在,随着所属区域的结束而释放。

 

 

4

成员变量都有默认初始化值。

局部变量没有默认初始化值。

 

 

 

*/

class Car

{

int num;

String color;

 

 

void run()

{

 

31


 

 

 

//int num = 10;

System.out.println(num+"..."+color);

}

 

 

}

 

 

 

class CarDemo

{

public static void main(String[] args)

{

//在计算机中创建一个car的实例。通过new关键字。


//

//

//

 

 

//

 

 

 

//

//

//

 

 

//

//

//

 

 

//

//

//

//

Car c = new Car();// c就是一个类类型的引用变量,指向了该类的对象。

c.num = 4;

c.color = "red";

 

 

c.run();//要使用对象中的内容可以通过 对象.成员 的形式来完成调用。

 

 

 

Car c1 = new Car();

c1.num = 4;

c1.color = "red";

 

 

Car c2 = new Car();

c2.num = 4;

c2.color = "red";

 

 

Car c1 = new Car();

Car c2 = new Car();

show(c1);

show(c2);

 

 

/*

匿名对象。没有名字的对象 。

new Car();//匿名对象。其实就是定义对象的简写格式。

 

Car c = new Car();

c.run();

 

 

new Car().run();

 

 

1,当对象对方法仅进行一次调用的时候,就可以简化成匿名对象。

 

new Car().num = 5;

new Car().color = "green";

new Car().run();

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

32


 

 

 

2,匿名对象可以作为实际参数进行传递。

 

*/


 

 

//

//

 

Car c1 = new Car();

show(c1);


show(new Car());

}

 

 

//汽车改装厂。

public static void show(Car c)//类类型的变量一定指向对象。要不就是null

{

c.num = 3;

c.color = "black";

System.out.println(c.num+"..."+c.color);

}

}

 

48.封装(private)使用:

 

 

/*

人:

属性:

年龄。

 

 

行为:

说话。

 

*/

 

 

/*

private:私有,是一个权限修饰符。用于修饰成员。

私有的内容只在本类中有效。

 

 

注意:私有仅仅是封装的一种体现而已。

 

*/

class Person

{

private /*私有*/int age;

 

public void setAge(int a)//setXxx getXxx

{

age = a;

}

public int getAge()

 

33


 

 

 

{

return age;

}

/*

public void haha(int a)

{

if(a>0 && a<130)

{

age = a;

speak();

}

else

System.out.println("错误的数据");

}

*/

void speak()

{

System.out.println("age="+age);

}

}

 

 

class PersonDemo

{

public static void main(String[] args)

{

Person p = new Person();


//

 

 

//

p.age = -20;

p.haha(-20);

p.speak();


}

 

 

public static void selectSort(int[] arr){}

 

 

private static void swap(int[] arr,int a,int b){}

}

 

49.基本数据类型和引用数据类型参数的传递:

 

//基本数据类型参数传递

class Demo

{

public static void main(String[] args)

{

 

 

int x = 3;

show(x);

System.out.println("x="+x);

 

34


 

 

 

}

public static void show(int x)

{

x = 4;

}

}

 

 

//引用数据类型参数传递

class Demo

{

int x = 3;

public static void main(String[] args)

{

Demo d = new Demo();

d.x = 9;

show(d);

System.out.println(d.x);

}

public static void show(Demo d)

{

d.x = 4;

}

}

 

50.构造函数:

 

 

class Person

{

private String name;

private int age;

 

 

//定义一个Person类的构造函数。

Person()//构造函数,而且是空参数的。

{

 

 

name = "baby";

age = 1;

System.out.println("person run");

 

 

}

 

 

//如果有的孩子一出生就有名字。

Person(String n)

{

name = n;

}

 

35


 

 

 

 

 

 

public void setName(String n)

{

name = n;

}

 

 

Person(String n,int a)

{

name = n;

age = a;

}

 

 

public void speak()

{

System.out.println(name+":"+age);

}

 

 

}

/*

//构造函数:构建创造对象时调用的函数。作用:可以给对象进行初始化。

创建对象都必须要通过构造函数初始化。

 

 

一个类中如果没有定义过构造函数,那么该类中会有一个默认的空参数构造函数。

如果在类中定义了指定的构造函数,那么类中的默认构造函数就没有了。

 

 

一般函数和构造函数什么区别呢?

 

 

构造函数:对象创建时,就会调用与之对应的构造函数,对对象进行初始化。

一般函数:对象创建后,需要函数功能时才调用。

 

 

构造函数:对象创建时,会调用只调用一次。

一般函数:对象创建后,可以被调用多次。

 

 

 

什么时候定义构造函数呢?

在描述事物时,该事物一存在就具备的一些内容,这些内容都定义在构造函数中。

构造函数可以有多个,用于对不同的对象进行针对性的初始化.

多个构造函数在类中是以重载的形式来体现的。

 

 

细节:

1,构造函数如果完成了set功能。set方法是否需要。

2,一般函数不能直接调用构造函数。

3,构造函数如果前面加了void就变成了一般函数。

4,构造函数中是有return语句的。

*/

 

 

36


 

 

 

class ConsDemo

{

public static void main(String[] args)

{

Person p = new Person();


//

p.speak();


Person p1 = new Person("旺财");

p1.setName("旺旺");

 

p1.speak();

 

 

Person p2 = new Person("小强",10);

p2.speak();

}

}

 

51.主函数解剖:

 

/*

public static void main(String[] args)

 

 

主函数特殊之处:

1,格式是固定的。

2,被jvm所识别和调用。

 

 

public:因为权限必须是最大的。

static:不需要对象的,直接用主函数所属类名调用即可。

void:主函数没有具体的返回值。

main:函数名,不是关键字,只是一个jvm识别的固定的名字。

String[] args:这是主函数的参数列表,是一个数组类型的参数,而且元素都是字符串类型。

 

*/

 

 

class MainDemo

{

public static void main(String[] args) //new String[0]

{

/**/


//

System.out.println(args);//[Ljava.lang.String;@c17164


System.out.println(args.length);

for(int x=0; x

System.out.println(args[x]);

 

 

}

}

 

 

 

 

37


 

 

 

 

52.静态代码块:

 

 

/*

静态代码块。

随着类的加载而执行。而且只执行一次。

 

 

作用:

用于给类进行初始化。

*/

class StaticCode

{

static int num ;

static

{

num = 10;


//

num *=3;


System.out.println("hahahah");

}

StaticCode(){}

 

 

static void show()

{

System.out.println(num);

}

}

 

 

class Person

{

private String name;

 

 

 

{ //构造代码块。可以给所有对象进行初始化的。

 

System.out.println("constructor code ");


//

cry();

}

 

 

static

{

System.out.println("static code");

}

 

 

Person()//是给对应的对象进行针对性的初始化。

{

name = "baby";

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

38


 

 

 

//

 

 

 

 

 

 

//

 

 

 

cry();

}

Person(String name)

{

this.name = name;

cry();


}

public void cry()

{

System.out.println("哇哇");

 

}

 

 

public void speak()

{

System.out.println("name:"+name);

}

 

 

static void show()

{

System.out.println("show run");

}

}

 

 

 

class StaticCodeDemo

{

static

{


//

 

 

 

 

 

 

//

//

 

 

//

//

//

//

//

//

 

 

 

//

//

System.out.println("a");

}

public static void main(String[] args)

{

 

 

Person p = null;

p.speak();

 

 

Person.show();

Person p1 = new Person();

p1.speak();

Person p2 = new Person("旺财");

p2.speak();

new Person();

 

 

 

new StaticCode().show();

new StaticCode().show();

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

39


 

 

 

//

//

 

 

 

StaticCode.show();

System.out.println("b");


}

}

 

53.static

 

class Person

{

String name;//成员变量,实例变量

static String country = "CN";//静态变量。类变量

public void show()

{

System.out.println(Person.country+":"+this.name);

}

}

 

 

/*

static的特点:

1static是一个修饰符,用于修饰成员。

2static修饰的成员被所有的对象所共享。

3static优先于对象存在,因为static的成员随着类的加载就已经存在了。

4static修饰的成员多了一种调用方式,就可以直接被类名所调用 。 类名.静态成员 。

5static修饰的数据是共享数据,对象中的存储的是特有数据。

 

 

成员变量和静态变量的区别?

1,两个变量的生命周期不同。

成员变量随着对象的创建而存在,随着对象的被回收而释放。

静态变量随着类的加载而存在,随着类的消失而消失。

 

 

2,调用方式不同。

成员变量只能被对象调用。

静态变量可以被对象调用,还可以被类名调用。

 

 

3,别名不同。

成员变量也称为实例变量。

静态变量称为类变量。

 

 

4,数据存储位置不同。

成员变量数据存储在堆内存的对象中,所以也叫对象的特有数据.

静态变量数据存储在方法区(共享数据区)的静态区,所以也叫对象的共享数据.

 

 

 

静态使用的注意事项:

1,静态方法只能访问静态成员。(非静态既可以访问静态,又可以访问非静态)

2,静态方法中不可以使用this或者super关键字。

3,主函数是静态的。

 

40


 

 

 

*/

 

 

class StaticDemo

{

int num = 4;

public static void main(String[] args)

{


//

//

//

//

//

//

Person p = new Person();

p.name = "小强";

p.show();

System.out.println(p.country);

System.out.println(Person.country);

Person.show();


new StaticDemo().show();

}

 

 

public void show()

{

System.out.println(num);

}

}

 

54.静态的使用:

 

/*

静态什么时候用?

1,静态变量。

当分析对象中所具备的成员变量的值都是相同的 。

这时这个成员就可以被静态修饰。

只要数据在对象中都是不同的,就是对象的特有数据,必须存储在对象中,是非静态的。

如果是相同的数据,对象不需要做修改,只需要使用即可,不需要存储在对象中,定义成静态的。

 

 

2,静态函数。

函数是否用静态修饰,就参考一点,就是该函数功能是否有访问到对象中的特有数据。

简单点说,从源代码看,该功能是否需要访问非静态的成员变量,如果需要,该功能就是非静态的。

如果不需要,就可以将该功能定义成静态的。当然,也可以定义成非静态,

但是非静态需要被对象调用,而仅创建对象调用非静态的

没有访问特有数据的方法,该对象的创建是没有意义。

*/

 

 

class Demo

{

int age ;

static int num = 9;

Demo(int age)

{

this.age = age;

 

41


 

 

 

}

 

 

public static void speak()

{

System.out.println(num);

}

public void show()

{

System.out.println(age);

}

 

 

}

 

 

class StaticDemo3

{

public static void main(String[] args)

{


//

//

 

 

 

//

Demo d = new Demo(30);

d.speak();

Demo.speak();

 

 

System.out.println("Hello World!");


}

}

 

55.this关键字:

 

/*

当成员变量和局部变量重名,可以用关键字this来区分。

 

 

this : 代表对象。代表哪个对象呢?当前对象。

this就是所在函数所属对象的引用。

简单说:哪个对象调用了this所在的函数,this就代表哪个对象。

 

 

this也可以用于在构造函数中调用其他构造函数。

注意:只能定义在构造函数的第一行。因为初始化动作要先执行。

*/

class Person

{

private String name;

private int age;

Person()

{

name = "baby";

age = 1;

System.out.println("person run");

}

 

42


 

 

 

Person(String name)

{

this();

this.name = name;

}

Person(String name,int age)

{

this.name = name;

this.age = age;

}

public void speak()

{

System.out.println(this.name+":"+this.age);

}

 

 

/*

判断是否是同龄人。

*/

public boolean compare(Person p)

{

/*

if(this.age==p.age)

return true;

else

return false;

*/

return this.age==p.age;

}

 

 

}

 

 

class ThisDemo

{

public static void main(String[] args)

{

 

 

Person p1 = new Person("aa",30);//

Person p2 = new Person("zz",12);

 

 

p2.compare(p1);


//

//

//

//

//

new Person();

Person p = new Person("旺财",30);

p.speak();

Person p1 = new Person("小强");

p1.speak();


}

}

 

43


 

 

 

 

56.建立数组工具(练习):

 

 

/**

建立一个用于操作数组的工具类,其中包含着常见的对数组操作的函数如:最值,排序等 。

@author 张三

@version V1.0

*/

 

 

public class ArrayTool

{

private ArrayTool(){}//该类中的方法都是静态的,所以该类是不需要的创建对象的。为了保证不让

//其他成创建该类对象

//可以将构造函数私有化。

 

/**

获取整型数组的最大值。

@param arr 接收一个元素为int类型的数组。

@return 该数组的最大的元素值

*/

public static int getMax(int[] arr)

{

int maxIndex = 0;

for(int x=1; x

{

if(arr[x]>arr[maxIndex])

maxIndex = x;//

}

 

 

return arr[maxIndex];

}

/**

对数组进行选择排序。

@param arr 接收一个元素为int类型的数组。

 

*/

public static void selectSort(int[] arr)

{

for(int x=0; x

{

for(int y=x+1; y

{

if(arr[x]>arr[y])

swap(arr,x,y);

}

}

 

44


 

 

 

}

 

 

/*

用于给数组进行元素的位置置换。

@param arr 接收一个元素为int类型的数组。

@param a

@param b

 

 

*/

private static void swap(int[] arr,int a,int b)

{

int temp = arr[a];

arr[a] = arr[b];

arr[b] = temp;

}

 

 

/**

获取指定的元素在指定数组中的索引.

@param arr 接收一个元素为int类型的数组。

@param key 要找的元素。

@return 返回该元素第一次出现的位置,如果不存在返回-1.

*/

public static int getIndex(int[] arr,int key)

{

for(int x=0; x

{

if(arr[x]==key)

return x;

}

return -1;

}

 

 

/**

int数组转换成字符串。格式是:[e1,e2,...]

@param arr 接收一个元素为int类型的数组。

@return 返回该数组的字符串表现形式。

*/

public static String arrayToString(int[] arr)

{

String str = "[";

 

 

for(int x=0; x

{

if(x!=arr.length-1)

str = str + arr[x]+", ";

else

str = str + arr[x]+"]";

 

45


 

 

 

}

return str;

}

}

 

57.继承:

 

关于成员变量(案例一):

 

/*

在子父类中,成员的特点体现。

1,成员变量。

2,成员函数。

3,构造函数。

*/

 

 

//1,成员变量。

 

/*

当本类的成员和局部变量同名用this区分。

当子父类中的成员变量同名用super区分父类。

 

 

this和super的用法很相似。

 

 

this:代表一个本类对象的引用。

super:代表一个父类空间。

*/

class Fu

{

private int num = 4;

 

 

public int getNum()

{

return num;

}

}

 

 

class Zi extends Fu

{

private int num = 5;

 

 

void show()

{

System.out.println(this.num+"....."+super.getNum());

}

}

 

 

46


 

 

 

class ExtendsDemo2

{

public static void main(String[] args)

{

Zi z = new Zi();

z.show();

}

}

 

 

关于成员函数(案例二):

 

 

//成员函数。

/*

当子父类中出现成员函数一模一样的情况,会运行子类的函数。

这种现象,称为覆盖操作。这时函数在子父类中的特性。

函数两个特性:

1,重载。同一个类中。overload

2,覆盖。子类中。覆盖也称为重写,覆写。override

 

 

覆盖注意事项:

1,子类方法覆盖父类方法时,子类权限必须要大于等于父类的权限。

2,静态只能覆盖静态,或被静态覆盖。

*/

class Fu

{

public static void show()

{

System.out.println("fu show run");

}

}

 

 

class Zi extends Fu

{

public static void show()

{

System.out.println("Zi show run");

}

}

 

 

class ExtendsDemo3

{

public static void main(String[] args)

{


//

//

Zi z = new Zi();

z.show();

NewPhone p = new NewPhone();

p.show();

 

 

 

 

 

 

47


 

 

 

p.call();

}

}

 

 

/*

什么时候使用覆盖操作?

当对一个类进行子类的扩展时,子类需要保留父类的功能声明,

但是要定义子类中该功能的特有内容时,就使用覆盖操作完成.

*/

class honeP

{

void call()

{}

void show()

{

System.out.println("number");

}

}

 

 

class NewPhone extends Phone

{

void show()

{

System.out.println("name");

System.out.println("pic");

super.show();

}

}

 

58.单例设计模式:

 

/*

设计模式:对问题行之有效的解决方式。其实它是一种思想。

 

 

1,单例设计模式。

解决的问题:就是可以保证一个类在内存中的对象唯一性。

 

 

必须对于多个程序使用同一个配置信息对象时,就需要保证该对象的唯一性。

 

 

如何保证对象唯一性呢?

1,不允许其他程序用new创建该类对象。

2,在该类创建一个本类实例。

3,对外提供一个方法让其他程序可以获取该对象。

 

 

步骤:

1,私有化该类构造函数。

2,通过new在本类中创建一个本类对象。

 

48


 

 

 

3,定义一个公有的方法,将创建的对象返回。

*/

 

 

//饿汉式

class Single //类一加载,对象就已经存在了。

{

private static Single s = new Single();

 

 

private Single(){}

 

 

public static Single getInstance()

{

return s;

}

}

 

 

//懒汉式


class Single2

//类加载进来,没有对象,只有调用了getInstance方法时,才会创建对象。


//延迟加载形式。

{

private static Single2 s = null;

 

 

private Single2(){}

 

 

public static Single2 getInstance()

{

if(s==null)

s = new Single2();

return s;

}

}

 

 

class SingleDemo

{

public static void main(String[] args)

{

Single s1 = Single.getInstance();

Single s2 = Single.getInstance();

 

 

System.out.println(s1==s2);


 

 

//

 

Single ss = Single.s;


 

 

//

//

 

Test

Test

Test

Test

 

t1

t2

t1

t2

 

=

=

=

=

 

new Test();

new Test();

Test.getInstance();

Test.getInstance();


 

49


 

 

 

t1.setNum(10);

t2.setNum(20);

System.out.println(t1.getNum());

System.out.println(t2.getNum());

}

}

 

 

class Test

{

private int num;

 

 

private static Test t = new Test();

private Test(){}

public static Test getInstance()

{

return t;

}

public void setNum(int num)

{

this.num = num;

}

public int getNum()

{

return num;

}

 

 

}

 

59.面向对象总结:

 

面向对象:

 

 

1,面向对象和面向过程思想。

 

 

面向对象强调的是对象实例。

面向过程强调的是动作。

对象将动作进行封装。

 

 

在问题领域中,我们先去找的都是涉及的对象,

然后分析对象的属性和行为。

 

 

 

2,面向对象的特点。

1,是一种常见思想。

2,复杂变简单。

3,执行者变指挥者。

举例:面试的例子。

 

50


 

 

 

 

 

 

3,类与对象的关系。

类:就是对事物的描述,其实类就是将对象共性的内容进行抽取。

对象:就是该类事物实实在在存在个体,在java中通过new来完成创建的,

堆内存的对象主要用于封装对象中的特有数据。

 

 

 

4,类中的成员:

成员变量:事物的属性,

成员函数:事物的行为。

 

 

成员变量和局部变量的区别?

答:

 

 

如何使用成员,只要建立该类对象,并通过 "对象.对象成员" 就可以完成调用

 

 

5,封装。

隐藏实现细节,并对外提供公共的访问方式。

 

 

函数或者类都是一个封装体。

 

 

特点:

1,隔离的变量。

2,便于使用。

3,提高复用。

4,提高安全性。

 

 

举例:机箱的例子

 

 

体现之一:私有。

 

 

私有在源代码中的使用就是在本类中有效。

 

 

通常会将成员变量xxx私有化,并对外提供对应的setXxx getXxx方法对其进行访问。

其实目的就是成员变量访问进行控制。 让成员的访问具备可控性,提高了程序的健壮性。

私有仅仅是封装的体现形式之一而已。

自定义一个Person类。动手。

 

 

6,构造函数。

写法特点:

1,函数名和类名相同。

2,不需要定义返回值类型

3,没有具体的返回值。但是有return语句,用于结束构造函数。

 

 

使用特点:

1,定义类中,有默认的空参数构造函数。如果已经自定义,那么默认就没有了。

 

51


 

 

 

2,构造函数在类有多个,是重载形式体现出来的。

 

 

构造函数的作用:

用于给对象进行针对性的初始化。

构造函数和一般函数的区别?

1,

2,

3,

什么时候使用构造函数呢?

当对象创建时就需要一些内容(数据和行为),那么这些内容都定义在构造函数中。

 

 

7,this关键字.

this:代表的是对象。哪个对象调用了this所在的函数,this就代表哪个对象。

 

 

用法体现

1,当成员变量和局部变量同名时,可以用this区别。

2,当构造函数中调用本类其他构造函数时,可以用this完成。 this(实际参数);

这种调用必须定义在构造函数的第一行。初始化动作先完成。

 

 

应用:只要在定义功能时,用到了本类对象,那么就使用this来表示。

 

 

8,static关键字:

 

 

特点:

1,修饰成员。,

2,随着类加载,随着类消失。

3,优先于对象。

4,用类名直接访问

 

 

使用注意事项:

1,静态方法只能访问静态,静态有访问局限性。

2,静态方法中不能有this super关键字。

3,主函数是静态的。

 

 

静态变量和成员变量的区别?

1,

2,

3,

4,

什么时候使用静态?

1,当成员变量的数据各个对象都相同时,可以用static修饰,让多个对象共享。

2,函数如果访问了特有数据(非静态成员变量),该函数是非静态的。

函数如果没有访问特有数据,那么该函数就可以静态修饰。

如果类中的功能都是静态的,那么该类创建对象是没有意义的,所以构造函数需要私有化。

 

 

 

9,代码块。

 

52


 

 

 

1,局部代码快。

对局部变量的生命周期进行控制。

2,构造代码块。

对所有对象进行初始化。

3,静态代码块。

对类进行初始化。

 

 

10,单例设计模式:

1,解决的问题:保证类在内存的对象唯一性。

2,思路:

3,步骤:

4,两种方式的区别?懒汉式(延迟加载方式),饿汉式

涉及的内存图要会画!必须的。

 

60.抽象类简述:

 

/*

抽象类:

抽象:笼统,模糊,看不懂!不具体。

 

 

 

特点:

1,方法只有声明没有实现时,该方法就是抽象方法,需要被abstract修饰。

抽象方法必须定义在抽象类中。该类必须也被abstract修饰。

2,抽象类不可以被实例化。为什么?因为调用抽象方法没意义。

3,抽象类必须有其子类覆盖了所有的抽象方法后,该子类才可以实例化。

否则,这个子类还是抽象类。

 

 

 

1,抽象类中有构造函数吗?

有,用于给子类对象进行初始化。

 

 

 

2,抽象类可以不定义抽象方法吗?

可以的。 但是很少见,目的就是不让该类创建对象。AWT的适配器对象就是这种类。

通常这个类中的方法有方法体,但是却没有内容。

 

abstract class Demo

{

void show1()

{}

 

 

 

void show2()

{}

}

 

 

53


 

 

 

 

 

3,抽象关键字不可以和那些关键字共存?

private 不行

static 不行

final 不行

 

 

 

4,抽象类和一般类的异同点。

相同点:

抽象类和一般类都是用来描述事物的,都在内部定了成员。

不同:

1,一般类有足够的信息描述事物。

抽象类描述事物的信息有可能不足。

2,一般类中不能定义抽象方法,只能定非抽象方法。

抽象类中可定义抽象方法,同时也可以定义非抽象方法。

3,一般类可以被实例化。

抽象类不可以被实例化。

 

 

 

5,抽象类一定是个父类吗?

是的。因为需要子类覆盖其方法后才可以对子类实例化。

*/

 

 

abstract class Demo

{

abstract /*抽象*/ void show();

 

}

 

 

/*

 

 

class DemoA extends Demo

{

void show()

{

System.out.println("demoa show");

}

}

class DemoB extends Demo

{

void show()

{

System.out.println("demob show");

}

}

*/

abstract class 犬科

 

54


 

 

 

{

abstract void 吼叫();

}

 

 

class 狗 extends 犬科

{

 

 

void 吼叫()

{

System.out.println("汪汪");

}

}

class 狼 extends 犬科

{

void 吼叫()

{

System.out.println("嗷嗷");

}

}

 

 

 

class AbstractDemo

{

public static void main(String[] args)

{

System.out.println("Hello World!");

}

}

 

61.继承&构造函数:

 

/*

子父类中的构造函数的特点。

 

 

在子类构造对象时,发现,访问子类构造函数时,父类也运行了。

为什么呢?

原因是:在子类的构造函数中第一行有一个默认的隐式语句。 super();

 

 

子类的实例化过程:子类中所有的构造函数默认都会访问父类中的空参数的构造函数。

 

 

为什么子类实例化的时候要访问父类中的构造函数呢?

那是因为子类继承了父类,获取到了父类中内容(属性),所以在使用父类内容之前,

要先看父类是如何对自己的内容进行初始化的。

 

 

所以子类在构造对象时,必须访问父类中的构造函数。

为什么完成这个必须的动作,就在子类的构造函数中加入了super()语句。

 

 

55


 

 

 

如果父类中没有定义空参数构造函数,那么子类的构造函数必须用super明确要调用

父类中哪个构造函数。同时子类构造函数中如果使用this调用了本类构造函数时,

那么super就没有了,因为superthis都只能定义第一行。所以只能有一个。

但是可以保证的是,子类中肯定会有其他的构造函数访问父类的构造函数。

 

 

注意:supre语句必须要定义在子类构造函数的第一行。因为父类的初始化动作要先完成。

*/

 

 

class Fu

{

int num ;

Fu()

{

num =10;

System.out.println("A fu run");

}

Fu(int x)

{

System.out.println("B fu run..."+x);

}

}

class Zi extends Fu

{

int num;

Zi()

{

//super();//调用的就是父类中的空参数的构造函数。

 

System.out.println("C zi run"+num);

}

Zi(int x)

{

this();

//super();


//

super(x);


System.out.println("D zi run "+x);

}

}

 

 

class ExtendsDemo4

{

public static void main(String[] args)

{

new Zi(6);

}

}

class Demo//extends Object

 

56


 

 

 

{

/*

 

 

Demo()

{

super();

return;

}

*/

}

 

62.一个对象实例化的过程(绅节):

 

class Fu

{

Fu()

{

super();

show();

return;

}

 

 

void show()

{

System.out.println("fu show");

}

}

class Zi extends Fu

{

int num = 8;

Zi()

{

super();

//-->通过super初始化父类内容时,子类的成员变量并未显示初始化。等super()父类初始化完毕

//后,

//才进行子类的成员变量显示初始化。

 

System.out.println("zi cons run...."+num);

return;

}

void show()

{

System.out.println("zi show..."+num);

}

}

class ExtendsDemo5

{

 

57


 

 

 

public static void main(String[] args)

{

Zi z = new Zi();

z.show();

}

}

 

 

/*

一个对象实例化过程:

Person p = new Person();

1JVM会读取指定的路径下的Person.class文件,并加载进内存,

并会先加载Person的父类(如果有直接的父类的情况下).

2,在堆内存中的开辟空间,分配地址。

3,并在对象空间中,对对象中的属性进行默认初始化。

4,调用对应的构造函数进行初始化。

5,在构造函数中,第一行会先到调用父类中构造函数进行初始化。

6,父类初始化完毕后,在对子类的属性进行显示初始化。

7,在进行子类构造函数的特定初始化。

8,初始化完毕后,将地址值赋值给引用变量.

*/

 

63.final关键字:

 

//继承弊端:打破了封装性。

/*

final关键字:

1final是一个修饰符,可以修饰类,方法,变量。

2final修饰的类不可以被继承。

3final修饰的方法不可以被覆盖。

4final修饰的变量是一个常量,只能赋值一次。

为什么要用final修饰变量。其实在程序如果一个数据是固定的,

那么直接使用这个数据就可以了,但是这样阅读性差,所以它该数据起个名称。

而且这个变量名称的值不能变化,所以加上final固定。

写法规范:常量所有字母都大写,多个单词,中间用_连接。

*/

class Fu

{

void method()

{

//调用了底层系统的资源。

}

}

class Zi extends Fu

{

public static final double MY_PI = 3.14;

static final int x = 7;

 

 

58


 

 

 

void method()

{


//

//

final int x = 9;

x = 9;


final int NUMBER = 9;

 

 

System.out.println(MY_PI);

}

}

 

 

class FinalDemo

{

public static void main(String[] args)

{

System.out.println("Hello World!");

}

}

 

64.接口(interface):

 

/*

abstract class AbsDemo

{

abstract void show1();

abstract void show2();

}

 

 

当一个抽象类中的方法都是抽象的时候,这时可以将该抽象类用

另一种形式定义和表示,就是接口 interface。

*/

 

 

//定义接口使用的关键字不是class,是interface.

 

/*

对于接口当中常见的成员:而且这些成员都有固定的修饰符。

1,全局常量: public static final

 

 

2,抽象方法。public abstract

 

 

由此得出结论,接口中的成员都是公共的权限.

*/

interface Demo

{

public static final int NUM = 4;

 

 

public abstract void show1();

public abstract void show2();

 

59


 

 

 

}

//类与类之间是继承关系,类与接口直接是实现关系。

/*

接口不可以实例化。

 

 

只能由实现了接口的子类并覆盖了接口中所有的抽象方法后,该子类才可以实例化。

否则,这个子类就是一个抽象类。

 

*/

 

 

class DemoImpl implements /*实现*/Demo

{

public void show1()

{}

 

 

public void show2()

{

 

 

}

}

 

 

 

/*

在java中不直接支持多继承,因为会出现调用的不确定性。

所以java将多继承机制进行改良,在java中变成了多实现。

 

 

一个类可以实现多个接口。

 

*/

 

 

interface A

{

public void show();

}

 

 

interface Z

{

public int add(int a,int b);

}

 

 

class Test implements A,Z//多实现

{

 

 

public int add(int a,int b)

{

return a+b+3;

}

 

60


 

 

 

/**/

public void show(){}

 

 

}

 

 

/*

一个类在继承另一个类的同时,还可以实现多个接口。

*/

class Q

{

public void method()

{}

}

 

 

abstract class Test2 extends Q implements A,Z

{

 

 

}

/*

接口的出现避免了单继承的局限性。

*/

 

 

interface CC

{

void show();

}

interface MM

{

void method();

}


 

 

interface QQ extends

 

CC,MM//接口与接口之间是继承关系,而且接口可以多继承。


{

void function();

}

 

 

class WW implements QQ

{

//覆盖3个方法。

public void show(){}

public void method(){}

public void function(){}

}

 

 

class InterfaceDemo

{

public static void main(String[] args)

 

61


 

 

 

{

 

 

Test t = new Test();

t.show();


 

 

//

//

//

//

 

DemoImpl d = new DemoImpl();

System.out.println(d.NUM);

System.out.println(DemoImpl.NUM);

System.out.println(Demo.NUM);


}

}

 

65.接口和抽象类异同点:

 

/*

抽象类和接口的异同点:

 

 

相同点:

都是不断向上抽取而来的。

 

 

不同点:

1,抽象类需要被继承,而且只能单继承。

接口需要被实现,而且可以多实现。

2,抽象类中可以定义抽象方法和非抽象方法,子类继承后,可以直接使用非抽象方法。

接口中只能定义抽象方法,必须由子类去实现。

3,抽象类的继承,是is a关系,在定义该体系的基本共性内容。

接口的实现是 like a 关系,在定义体系额外功能。

犬按功能分:有导盲犬,搜爆犬。

 

 

abstract class 犬

{

abstract void 吼叫();

}

 

 

//abstract class 导盲

interface 导盲

{

abstract void 导盲();

}

 

 

class 导盲犬 extends 犬 implements 导盲

{

public void 吼叫()

{

}

public void 导盲(){}

}

 

62


 

 

 

 

 

//在不同的问题领域中,有不同的分析方式。

学员:

学习。

抽烟学员

 

 

 

烟民。

*/

 

 

class InterfaceDemo

{

public static void main(String[] args)

{

System.out.println("Hello World!");

}

}

 

66.多态简述:

 

/*

对象的多态性。

 

 

class 动物

{}

 

 

class 猫 extends 动物

{}

 

 

class 狗 extends 动物

{}

 

 

 x = new 猫();

 

 

动物 x = new 猫();//一个对象,两种形态。

 

 

猫这类事物即具备者猫的形态,又具备着动物的形态。

这就是对象的多态性。

 

 

简单说:就是一个对象对应着不同类型.

 

 

多态在代码中的体现:

父类或者接口的引用指向其子类的对象。

 

 

多态的好处:

提高了代码的扩展性,前期定义的代码可以使用后期的内容。

 

 

63


 

 

 

多态的弊端:

前期定义的内容不能使用(调用)后期子类的特有内容。

 

 

多态的前提:

1,必须有关系,继承,实现。

2,要有覆盖。

*/

 

 

abstract class Animal

{

abstract void eat();

 

 

}

 

 

class Dog extends Animal

{

void eat()

{

System.out.println("啃骨头");

}

void lookHome()

{

System.out.println("看家");

}

}

 

 

class Cat extends Animal

{

void eat()

{

System.out.println("吃鱼");

}

void catchMouse()

{

System.out.println("抓老鼠");

}

}

 

 

class Pig extends Animal

{

void eat()

{

System.out.println("饲料");

}

void gongDi()

{

System.out.println("拱地");

 

64


 

 

 

}

}

 

 

class DuoTaiDemo

{

public static void main(String[] args)

{


 

 

//

//

//

 

 

 

 

 

 

 

 

 

//

 

 

 

 

 

//

//

//

 

 

//

//

//

 

 

 

 

 

//

 

 

//

 

 

//

//

 

Cat c = new Cat();

c.eat();

c.catchMouse();

 

 

Animal a = new Cat(); //自动类型提升,猫对象提升了动物类型。但是特有功能无法s访问。

//作用就是限制对特有功能的访问。

//专业讲:向上转型。将子类型隐藏。就不用使用子类的特有方法。

 

 

 

a.eat();

 

 

//如果还想用具体动物猫的特有功能。

//你可以将该对象进行向下转型。

Cat c = (Cat)a;//向下转型的目的是为了使用子类中的特有方法。

c.eat();

c.catchMouse();

 

 

注意:对于转型,自始自终都是子类对象在做着类型的变化。

Animal a1 = new Dog();

Cat c1 = (Cat)a1;//ClassCastException

/*

Cat c = new Cat();

 

 

Dog d = new Dog();

 

 

c.eat();

method(c);

method(d);

method(new Pig());

*/

 

 

method(new Dog());

 

 

}

 

 

public static void method(Animal a)//Animal a = new Dog();

{

a.eat();

 

 

65


 

 

 

if(a instanceof Cat)//instanceof:用于判断对象的具体类型。只能用于引用数据类型判断


//

//通常在向下转型前用于健壮性的判断。


 

{

Cat c = (Cat)a;

c.catchMouse();

}

else if(a instanceof Dog)

{

Dog d = (Dog)a;

d.lookHome();

}

else

{

 

 

}

 

 

}

/*

public static void method(Cat c)

{

c.eat();

}

public static void method(Dog d)

{

 

 

}

*/

}

 

67.多态时各组成的发化:

 

/*

多态时,成员的特点:

 

 

1,成员变量。

编译时:参考引用型变量所属的类中的是否有调用的成员变量,有,编译通过,没有,编译失败。

运行时:参考引用型变量所属的类中的是否有调用的成员变量,并运行该所属类中的成员变量。

简单说:编译和运行都参考等号的左边。哦了。

作为了解。

 

 

2,成员函数(非静态)

编译时:参考引用型变量所属的类中的是否有调用的函数。有,编译通过,没有,编译失败。

运行时:参考的是对象所属的类中是否有调用的函数。

简单说:编译看左边,运行看右边。

因为成员函数存在覆盖特性。

 

 

66


 

 

 

3,静态函数。

编译时:参考引用型变量所属的类中的是否有调用的静态方法。

运行时:参考引用型变量所属的类中的是否有调用的静态方法。

简单说,编译和运行都看左边。

其实对于静态方法,是不需要对象的。直接用类名调用即可。

*/

 

 

class Fu

{

// int num = 3;

void show()

{

System.out.println("fu show");

}

 

 

static void method()

{

System.out.println("fu static method");

}

}

 

 

class Zi extends Fu

{

// int num = 4;

void show()

{

System.out.println("zi show");

}

 

 

static void method()

{

System.out.println("zi static method");

}

}

 

 

class DuoTaiDemo3

{

public static void main(String[] args)

{

Fu.method();

Zi.method();

Fu f = new Zi();//


//

//

//

 

 

//

f.method();

f.show();

System.out.println(f.num);

 

 

Zi z = new Zi();

 

 

 

 

 

 

 

67


 

 

 

//

 

 

 

System.out.println(z.num);


}

}

 

68.内部类:

 

/*

内部类访问特点:

1,内部类可以直接访问外部类中的成员。

2,外部类要访问内部类,必须建立内部类的对象。

一般用于类的设计。

分析事物时,发现该事物描述中还有事物,而且这个事物还在访问被描述事物的内容。

这时就是还有的事物定义成内部类来描述。

 

 

内部类能直接访问外部类中成员,是因为内部类持有了外部类的引用,即外部类名.this

 

 

内部类也可以存放在局部位置上,但是内部类在局部位置上只能访问局部中被final修饰的局部变量。

*/

class Outer

{

private static int num = 31;

 

 

class Inner// 内部类。

{

void show()

{

System.out.println("show run..."+num);

}

/*static void function()//如果内部类中定义了静态成员,该内部类也必须是静态的。

{

System.out.println("function run ...."+num);

}

*/

}

 

 

public void method()

{

Inner in = new Inner();

in.show();

}

}

 

 

class InnerClassDemo

{

public static void main(String[] args)

{


//

Outer out = new Outer();

 

68


 

 

 

//

 

 

//

//

 

 

 

//

//

 

 

 

//

 

 

 

out.method();

//直接访问外部类中的内部类中的成员。

Outer.Inner in = new Outer().new Inner();

in.show();

 

 

//如果内部类是静态的。 相当于一个外部类

Outer.Inner in = new Outer.Inner();

in.show();

 

 

//如果内部类是静态的,成员是静态的。

Outer.Inner.function();


 

 

}

}

 

69.匿名内部类:

 

/*

匿名内部类,就是内部类的简写格式。

必须有前提:

内部类必须继承或者实现一个外部类或者接口。

 

 

匿名内部类:其实就是一个匿名子类对象。

格式:new 父类or接口(){ 子类内容}

*/

 

 

abstract class Demo

{

abstract void show();

}

 

 

class Outer

{

int num = 4;

/*

class Inner extends Demo

{

void show()

{

System.out.println("show ..."+num);

}

}

*/

public void method()

{

//new Inner().show();

new Demo()//匿名内部类。

 

69


 

 

 

{

void show()

{

System.out.println("show ........"+num);

}

}.show();

}

}

 

 

class InnerClassDemo4

{

public static void main(String[] args)

{

new Outer().method();

}

}

 

70.匿名内部类使用范例(两个):

 

范例(一):

interface Inter

{

void show1();

void show2();

}

 

 

class Outer

{

/*

class Inner implements Inter

{

public void show1()

{

}

public void show2()

{

 

 

}

}

*/

 

 

public void method()

{


//

//

//

Inner in = new Inner();

in.show1();

in.show2();

 

 

 

 

 

 

70


 

 

 

 

Inter in = new Inter()

{

public void show1()

{

 

 

}

public void show2()

{

 

 

}

 

 

};

in.show1();

in.show2();

}

}

 

 

 

/*

通常的使用场景之一:

当函数参数是接口类型时,而且接口中的方法不超过三个。

可以用匿名内部类作为实际参数进行传递

*/

class InnerClassDemo5

{

 

 

class Inner

{

}

public static void main(String[] args)

{

System.out.println("Hello World!");

/*

show(new Inter()

{

public void show1(){}

public void show2(){}

});

*/


 

 

//

 

new Inner();

}

 

 

public void method()

{

new Inner();

 

 

 

 

 

 

 

 

 

 

 

71


 

 

 

}

 

 

public static void show(Inter in)

{

in.show1();

in.show2();

}

}

 

 

范例(二):

 

class Outer

{

void method()

{

Object obj = new Object()

{

public void show()

{

System.out.println("show run");

}

 

 

};

obj.show();//因为匿名内部类这个子类对象被向上转型为了Object类型。

//这样就不能在使用子类特有的方法了。

}

}

 

 

class InnerClassDemo6

{

public static void main(String[] args)

{

new Outer().method();

}

}

 

 

注意:内部类的class字节码文件名字格式为Outer$Inner.class。

 

第四章:异常机制

 

71.异常简述:

 

/*

异常:是在运行时期发生的不正常情况。。

java中用类的形式对不正常情况进行了描述和封装对象。

描述不正常的情况的类,就称为异常类。

 

72


 

 

 

以前正常流程代码和问题处理代码相结合,

现在将正常流程代码和问题处理代码分离。提高阅读性.

其实异常就是java通过面向对象的思想将问题封装成了对象.

用异常类对其进行描述。

不同的问题用不同的类进行具体的描述。 比如角标越界、空指针等等。

问题很多,意味着描述的类也很多,

将其共性进行向上抽取,形成了异常体系。

 

 

最终问题(不正常情况)就分成了两大类。

Throwable:无论是error,还是异常,问题,问题发生就应该可以抛出,让调用者知道并处理。

//该体系的特点就在于Throwable及其所有的子类都具有可抛性。

可抛性到底指的是什么呢?怎么体现可抛性呢?

其实是通过两个关键字来体现的。

throws throw ,凡是可以被这两个关键字所操作的类和对象都具备可抛性.

|--1,一般不可处理的。Error

特点:是由jvm抛出的严重性的问题。

这种问题发生一般不针对性处理。直接修改程序

|--2,可以处理的。Exception

 

 

该体系的特点:

子类的后缀名都是用其父类名作为后缀,阅读性很强。

*/

 

 

class ExceptionDemo

{

public static void main(String[] args)

{

int[] arr = new int[1024*1024*800];//java.lang.OutOfMemoryError: Java heap

space


//

//

//

//

//

 

 

 

 

 

 

 

 

 

//

//

//

//

//

arr = null;

System.out.println(arr[3]);

 

 

 

sleep(-5);

}

 

 

public static void sleep2(int time)

{

if(time<0)

{

处理办法。

处理办法。

处理办法。

处理办法。

处理办法。

}

if(time>100000)

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

73


 

 

 

{


//

//

//

//

//

//

 

 

 

 

 

//

 

 

 

 

 

 

 

 

//

处理办法。

处理办法。

处理办法。

处理办法。

处理办法。

处理办法。

}

System.out.println("我睡。。。"+time);

 

sleep(-5);

}

public static void sleep(int time)

{

if(time<0)

{

抛出 new FuTime();//就代码着时间为负的情况,这个对象中会包含着问题的名称,信息,位


置等信息。

}

if(time>100000)

{


//

抛出 new BigTime();


}

System.out.println("我睡。。。"+time);

}

}

/*

class FuTime

{

}

class BigTime

{

}

*/

 

72.自定义异常:

 

/*

对于角标是整数不存在,可以用角标越界表示,

对于负数为角标的情况,准备用负数角标异常来表示。

 

 

负数角标这种异常在java中并没有定义过。

那就按照java异常的创建思想,面向对象,将负数角标进行自定义描述。并封装成对象。

 

 

这种自定义的问题描述成为自定义异常。

 

 

注意:如果让一个类称为异常类,必须要继承异常体系,因为只有称为异常体系的子类才有资格具备可抛性。

 

74


 

 

 

才可以被两个关键字所操作,throws throw

 

 

异常的分类:

1,编译时被检测异常:只要是Exception和其子类都是,除了特殊子类RuntimeException体系。

这种问题一旦出现,希望在编译时就进行检测,让这种问题有对应的处理方式。

这样的问题都可以针对性的处理。

 

 

2,编译时不检测异常(运行时异常):就是Exception中的RuntimeException和其子类。

这种问题的发生,无法让功能继续,运算无法进行,更多是因为调用者的原因导致的而或者引发了内部

状态的改变导致的。

那么这种问题一般不处理,直接编译通过,在运行时,让调用者调用时的程序强制停止,让调用者对代

码进行修正。

 

 

所以自定义异常时,要么继承Exception,要么继承RuntimeException

 

 

 

throws 和throw的区别:

1throws使用在函数上。

throw使用在函数内。

2throws抛出的是异常类,可以抛出多个,用逗号隔开。

throw抛出的是异常对象。

*/

 

 

class FuShuIndexException extends Exception

{

FuShuIndexException()

{}

 

 

 

FuShuIndexException(String msg)

{

super(msg);

}

}

 

 

class Demo

{


public

int

method(int[]

arr,int

index)//throws


NullPointerException//FuShuIndexException

{

 

 

if(arr==null)

throw new NullPointerException("数组的引用不能为空!");

 

if(index>=arr.length)

{

throw new ArrayIndexOutOfBoundsException("数组的角标越界啦,哥们,你是不是疯

 

75


 

 

 

了?:"+index);

}

if(index<0)

{

throw new FuShuIndexException("角标变成负数啦!!");

}

return arr[index];

}

}

 

 

class ExceptionDemo3

{

public static void main(String[] args) //throws FuShuIndexException

{

int[] arr = new int[3];

 

 

Demo d = new Demo();

int num = d.method(null,-30);

System.out.println("num="+num);

System.out.println("over");

}

 

 

}

 

 

/*

class Person

{

private String name;

Person(String name)

{

this.name = name;

}

public String getName()

{

return name;

}

}

class Student extends Person

{

Student(String name)

{

super(name);

}

}

*/

 

 

 

 

76


 

 

 

 

73.异常的捕捉处理:

 

/*

异常处理的捕捉形式:

这是可以对异常进行针对性处理的方式。

 

 

具体格式是:

 

try

{

//需要被检测异常的代码。

}

catch(异常类 变量)//该变量用于接收发生的异常对象

{

//处理异常的代码。

}

finally

{

//一定会被执行的代码。

}

 

 

异常处理的原则:

1,函数内容如果抛出需要检测的异常,那么函数上必须要声明。

否则必须在函数内用trycatch捕捉,否则编译失败。

 

 

2,如果调用到了声明异常的函数,要么trycatch要么throws,否则编译失败。

 

 

3,什么时候catch,什么时候throws 呢?

功能内容可以解决,用catch

解决不了,用throws告诉调用者,由调用者解决 。

 

 

4,一个功能如果抛出了多个异常,那么调用时,必须有对应多个catch进行针对性的处理。

内部又几个需要检测的异常,就抛几个异常,抛出几个,就catch几个。

*/

 

 

class FuShuIndexException extends Exception

{

FuShuIndexException()

{}

 

 

FuShuIndexException(String msg)

{

super(msg);

}

}

 

 

77


 

 

 

class Demo

{


public

int

method(int[]

arr,int

index)//throws


NullPointerException,FuShuIndexException

{

if(arr==null)

throw new NullPointerException("没有任何数组实体");

if(index<0)

throw new FuShuIndexException();

 

 

return arr[index];

}

}

 

 

class ExceptionDemo4

{

public static void main(String[] args)

{

int[] arr = new int[3];

Demo d = new Demo();

try

{

int num = d.method(null,-1);

System.out.println("num="+num);

 

 

}

 

 

catch(NullPointerException e)

{

System.out.println(e.toString());

}

catch (FuShuIndexException e)

{

System.out.println("message:"+e.getMessage());

System.out.println("string:"+e.toString());

 

 

e.printStackTrace();//jvm默认的异常处理机制就是调用异常对象的这个方法。

 

 

System.out.println("负数角标异常!!!!");

}

/*

catch(Exception e)//catch父类的catch放在最下面。

{

 

 

}

*/

System.out.println("over");

 

78


 

 

 

}

 

 

}

 

74.trycatchfinally的几种组合:

 

 

class Demo

{

public int show(int index)throws ArrayIndexOutOfBoundsException

{

 

 

if(index<0)

throw new ArrayIndexOutOfBoundsException("越界啦!!");

int[] arr = new int[3];

return arr[index];

}

}

 

 

class ExceptionDemo5

{

public static void main(String[] args)

{

Demo d = new Demo();

try

{

 

 

int num = d.show(-1);

System.out.println("num="+num);

}

catch (ArrayIndexOutOfBoundsException e)

{

System.out.println(e.toString());


 

 

//

//

 

return ;

System.exit(0);//退出jvm


}

finally//通常用于关闭(释放)资源。

{

System.out.println("finally");

}

 

 

System.out.println("over");

 

 

}

}

/*

 

79


 

 

 

 

 

连接数据库

查询。Exception

关闭连接。

 

*/

 

 

 

/*

try catch finally 代码块组合特点:

 

 

1

try catch finally

 

 

2,

try catch(多个)当没有必要资源需要释放时,可以不用定义finally

 

3,

try finally 异常无法直接catch处理,但是资源需要关闭。

 

 

 

void show()throws Exception

{

try

{

//开启资源。

throw new Exception();

}

finally

{

//关闭资源。

 

}

 

 

/*

catch(Exception e)

{

 

 

}

*/

}

*/

 

75.异常的注意事项:

 

异常的注意事项:

 

 

80


 

 

 

1,子类在覆盖父类方法时,父类的方法如果抛出了异常,

那么子类的方法只能抛出父类的异常或者该异常的子类。

 

 

2,如果父类抛出多个异常,那么子类只能抛出父类异常的子集。

 

 

简单说:子类覆盖父类只能抛出父类的异常或者子类或者子集。

注意:如果父类的方法没有抛出异常,那么子类覆盖时绝对不能抛,就只能try .

 

interface Inter

{

void function();

}

 

 

class D implements Inter

{

public void function()//throws Exception

{}

}

 

 

class A extends Exception

{

}

 

 

class B extends A

{

}

class C extends Exception

{

}

 

 

 

Exception

|--A

|--B

|--C

 

 

class Fu

{

void show()throws A

{}

}

 

 

class Test

{

void method(Fu f)//Fu f = new Zi();

{

try

 

81


 

 

 

{

f.show();

 

 

}

catch (A a)

{

}

}

}

 

 

class Zi extends Fu

{

void show()throws C

{

 

 

}

}

 

 

class

{

public static void main(String[] args)

{

Test t = new Test();

t.show(new Zi());

}

}

 

76.异常处理练习:

 

/*

毕老师用电脑上课。

问题领域中涉及两个对象。

毕老师,电脑。

分析其中的问题。

比如电脑蓝屏啦。冒烟啦。

*/

class LanPingException extends Exception

{

LanPingException(String msg)

{

super(msg);

}

}

 

 

class MaoYanException extends Exception

{

MaoYanException(String msg)

 

82


 

 

 

{

super(msg);

}

}

 

 

class NoPlanException extends Exception

{

NoPlanException(String msg)

{

super(msg);

}

}

 

 

class Computer

{

private int state = 2;

public void run()throws LanPingException,MaoYanException

{

if(state==1)

throw new LanPingException("电脑蓝屏啦!!");

if(state==2)

throw new MaoYanException("电脑冒烟啦!!");

 

 

System.out.println("电脑运行");

}

public void reset()

{

state = 0;

System.out.println("电脑重启");

}

}

 

 

class Teacher

{

private String name;

private Computer comp;

Teacher(String name)

{

this.name = name;

comp = new Computer();

}

 

 

public void prelect()throws NoPlanException

{

try

{

comp.run();

 

83


 

 

 

System.out.println(name+"讲课");

 

}

catch (LanPingException e)

{

System.out.println(e.toString());

comp.reset();

prelect();

}

catch (MaoYanException e)

{

System.out.println(e.toString());

test();

//可以对电脑进行维修。


//

throw e;


throw new NoPlanException("课时进度无法完成,原因:"+e.getMessage());

}

 

 

}

public void test()

{

System.out.println("大家练习");

}

}

class ExceptionTest

{

public static void main(String[] args)

{

Teacher t = new Teacher("毕老师");

try

{

t.prelect();

 

 

}

catch (NoPlanException e)

{

System.out.println(e.toString()+"......");

System.out.println("换人");

 

}

}

}

/*

 

 

class NoAddException extends Exception

{}

 

 

84


 

 

 

void addData(Data d)throws NoAddException

{

 

 

连接数据库

try

{

添加数据。出现异常 SQLException();

}

catch(SQLException e)

{

//处理代码。

 

throw new NoAddException();

}

fianlly

{

关闭数据库。

}

}

*/

 

77.类的全名=包名.类名:

 

package mypack;

 

 

//import packa.DemoA;//导入了packa包中的DemoA类。

//import packa.DemoAA;

//import packa.DemoAAA;

/*

import packa.*;//导入了packa包中所有的类。

import packa.abc.*;

 

 

packa\DemoA.class

packa\abc\DemoAbc.class

*/

 

 

//导包的原则:用到哪个类,就导入哪个类。

 

 

//import 干嘛用的啊?为了简化类名书写。

class PackageDemo

{

public static void main(String[] args)

{

 

 

//packa.DemoA d = new packa.DemoA();

DemoAbc d = new DemoA();

d.show();

 

85


 

 

 

 

packb.DemoB b = new packb.DemoB();

b.method();

System.out.println("Hello Package!");

}

}

/*

PackageDemo.java:8: 找不到符号

符号: 类 DemoA

位置: 类 mypack.PackageDemo

DemoA d = new DemoA();

^

PackageDemo.java:8: 找不到符号

符号: 类 DemoA

位置: 类 mypack.PackageDemo

DemoA d = new DemoA();

^

2 错误

 

 

原因:类名写错。

 

 

因为DemoA类已经有包的所属,所以必须要明确其报名。

记住:DemoA这个名词错的。 正确名称是报名.类名 packa.DemoA

 

--------------------------------------

PackageDemo.java:8: 软件包 packa 不存在

packa.DemoA d = new packa.DemoA();

^

PackageDemo.java:8: 软件包 packa 不存在

packa.DemoA d = new packa.DemoA();

^

2 错误

 

 

原因是在当前目录下没找到,因为packa存放在其他目录下,

应该配置一个classpath

-------------------------------------

PackageDemo.java:8: packa.DemoA 在 packa 中不是公共的;无法从外部软件包中对其进

行访问

packa.DemoA d = new packa.DemoA();

^

PackageDemo.java:8: packa.DemoA 在 packa 中不是公共的;无法从外部软件包中对其进

行访问

packa.DemoA d = new packa.DemoA();

^

2 错误

--------------------------------------

 

 

86


 

 

 

PackageDemo.java:9: show() 在 packa.DemoA 中不是公共的;无法从外部软件包中对其进

行访问

d.show();

^

1 错误

 

 

总结:

包与包之间的类进行访问,被访问的包中的类必须是public的,被访问的包中的类的方法也必须是public的。


 

public

protected

default

private


同一类中

同一包中

子类中

ok

ok

ok

ok

ok

ok

ok

ok

ok


不同包中

ok


*/

 

第五章:多线程技术

 

78.线程简述:

 

/*

进程:正在进行中的程序(直译)

线程:就是进程中一个负责程序执行的控制单元(执行路径)

一个进程中可以多执行路径,称之为多线程。

一个进程中至少要有一个线程。

开启多个线程是为了同时运行多部分代码。

每一个线程都有自己运行的内容。这个内容可以称为线程要执行的任务。

多线程好处:解决了多部分同时运行的问题。

多线程的弊端:线程太多回到效率的降低。

其实应用程序的执行都是cpu在做着快速的切换完成的,这个切换是随机的。

JVM启动时就启动了多个线程,至少有两个线程可以分析的出来。

1,执行main函数的线程,

该线程的任务代码都定义在main函数中。

 

 

2,负责垃圾回收的线程。

*/

class Demo extends Object

{

public void finalize()

{

System.out.println("demo ok");

}

}

 

 

class ThreadDemo

{

 

87


 

 

 

public static void main(String[] args)

{

 

 

new Demo();

new Demo();

new Demo();

System.gc();

System.out.println("Hello World!");

}

}

 

79.创建线程方法乊一即继承thread类:

 

/*

如何创建一个线程呢?

 

 

创建线程方式一:继承Thread类。

 

 

步骤:

1,定义一个类继承Thread类。

2,覆盖Thread类中的run方法。

3,直接创建Thread的子类对象创建线程。

4,调用start方法开启线程并调用线程的任务run方法执行。

 

 

可以通过ThreadgetName获取线程的名称 Thread-编号(0开始)

 

 

主线程的名字就是main

*/

class Demo extends Thread

{

private String name;

Demo(String name)

{

super(name);

//this.name = name;

}

public void run()

{

for(int x=0; x<10; x++)

{

//for(int y=-9999999; y<999999999; y++){}

 

 

System.out.println(name+"....x="+x+".....name="+Thread.currentThread().getNam

e());

}

}

}

 

88


 

 

 

 

class ThreadDemo2

{

public static void main(String[] args)

{

 

 

/*

创建线程的目的是为了开启一条执行路径,去运行指定的代码和其他代码实现同时运行。

 

 

而运行的指定代码就是这个执行路径的任务。

 

 

jvm创建的主线程的任务都定义在了主函数中。

 

 

而自定义的线程它的任务在哪儿呢?

Thread类用于描述线程,线程是需要任务的。所以Thread类也对任务的描述。

这个任务就通过Thread类中的run方法来体现。也就是说,run方法就是封装自定义线程运行任务的

函数。

 

 

run方法中定义就是线程要运行的任务代码。

 

 

开启线程是为了运行指定代码,所以只有继承Thread类,并复写run方法。

将运行的代码定义在run方法中即可。

*/

//


//

Thread t1 = new Thread();


 

 

Demo d1 = new Demo("旺财");

Demo d2 = new Demo("xiaoqiang");

d1.start();//开启线程,调用run方法。

 

d2.start();

System.out.println("over...."+Thread.currentThread().getName());

}

}

 

80.多线程典型案例乊银行存钱:

 

/*

需求:储户,两个,每个都到银行存钱每次存100,,共存三次。

*/

 

 

class Bank

{

private int sum;

// private Object obj = new Object();

public synchronized void add(int num)//同步函数

{

 

89


 

 

 

//

//

 

 

 

 

 

 

//

 

 

 

 

 

 

 

 

//

 

 

 

synchronized(obj)

{

sum = sum + num;

-->

try{Thread.sleep(10);}catch(InterruptedException e){}

System.out.println("sum="+sum);

}


}

}

 

 

class Cus implements Runnable

{

private Bank b = new Bank();

public void run()

{

for(int x=0; x<3; x++)

{

b.add(100);

}

}

}

 

 

class BankDemo

{

public static void main(String[] args)

{

Cus c = new Cus();

Thread t1 = new Thread(c);

Thread t2 = new Thread(c);

t1.start();

t2.start();

}

}

 

81.死锁常见情形乊同步嵌套:

 

/*

死锁:常见情景之一:同步的嵌套。

*/

class Ticket implements Runnable

{

private int num = 100;

Object obj = new Object();

boolean flag = true;

public void run()

{

 

 

90


 

 

 

 

if(flag)

while(true)

{

synchronized(obj)

{

show();

}

}

else

while(true)

this.show();

}

 

 

public synchronized void show()

{

 

 

synchronized(obj)

{

if(num>0)

{

try{Thread.sleep(10);}catch (InterruptedException e){}

 

 

 

System.out.println(Thread.currentThread().getName()+".....sale...."+num--);

}

}

}

}

 

 

class DeadLockDemo

{

public static void main(String[] args)

{

Ticket t = new Ticket();


//

System.out.println("t:"+t);


 

 

Thread t1 = new Thread(t);

Thread t2 = new Thread(t);

 

 

t1.start();

try{Thread.sleep(10);}catch(InterruptedException e){}

t.flag = false;

t2.start();

}

}

 

 

91


 

 

 

 

82.典型死锁案例(synchronized同步嵌套):

 

 

class Test implements Runnable

{

private boolean flag;

Test(boolean flag)

{

this.flag = flag;

}

 

 

public void run()

{

 

 

if(flag)

{

while(true)

synchronized(MyLock.locka)

{

System.out.println(Thread.currentThread().getName()+"..if

locka....");


synchronized(MyLock.lockb)

{


 

 

System.out.println(Thread.currentThread().getName()+"..if

lockb....");

}

}

}

else

{

while(true)

synchronized(MyLock.lockb)

{

System.out.println(Thread.currentThread().getName()+"..else

lockb....");

synchronized(MyLock.locka)

{

System.out.println(Thread.currentThread().getName()+"..else

locka....");

}

}

}

 

 

}

}

 

 

92


 

 

 

class MyLock

{

public static final Object locka = new Object();

public static final Object lockb = new Object();

}

 

 

class DeadLockTest

{

public static void main(String[] args)

{

Test a = new Test(true);

Test b = new Test(false);

 

 

Thread t1 = new Thread(a);

Thread t2 = new Thread(b);

t1.start();

t2.start();

}

}

 

83.多线程下的单例:

 

 

/*

多线程下的单例

*/

//饿汉式

class Single

{

private static final Single s = new Single();

private Single(){}

public static Single getInstance()

{

return s;

}

}

//懒汉式

加入同步为了解决多线程安全问题。

加入双重判断是为了解决效率问题。

class Single

{

private static Single s = null;

 

 

private Single(){}

 

 

public static Single getInstance()

{

 

93


 

 

 

if(s==null)

{

synchronized(Single.class)

{

if(s==null)


//

-->0 -->1


s = new Single();

}

}

return s;

}

}

class SingleDemo

{

public static void main(String[] args)

{

System.out.println("Hello World!");

}

}

 

84.静态同步函数的锁:

 

/*

静态的同步函数使用的锁是该函数所属字节码文件对象,

可以用 getClass方法获取,也可以用当前类名.class 表示。

*/

 

 

class Ticket implements Runnable

{

private static int num = 100;

// Object obj = new Object();

boolean flag = true;

public void run()

{


//

System.out.println("this:"+this.getClass());

 

 

if(flag)

while(true)

{

synchronized(Ticket.class)//(this.getClass())

{

if(num>0)

{

try{Thread.sleep(10);}catch (InterruptedException e){}

 

 

 

System.out.println(Thread.currentThread().getName()+".....obj...."+num--);

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

94


 

 

 

}

}

}

else

while(true)

this.show();

}

 

 

public static synchronized void show()

{

if(num>0)

{

try{Thread.sleep(10);}catch (InterruptedException e){}

 

 

System.out.println(Thread.currentThread().getName()+".....function...."+num--

);

}

}

}

 

 

class StaticSynFunctionLockDemo

{

public static void main(String[] args)

{

Ticket t = new Ticket();


 

 

//

//

//

//

 

Class clazz = t.getClass();

 

 

Class clazz = Ticket.class;

System.out.println("t:"+t.getClass());


 

 

Thread t1 = new Thread(t);

Thread t2 = new Thread(t);

t1.start();

try{Thread.sleep(10);}catch(InterruptedException e){}

t.flag = false;

t2.start();

}

}

 

85.同步函数&同步代码块的锁:

 

 

 

 

/*

同步函数的使用的锁是this

同步函数和同步代码块的区别:

 

95


 

 

 

同步函数的锁是固定的this

同步代码块的锁是任意的对象。

建议使用同步代码块。

*/

class Ticket implements Runnable

{

private int num = 100;

// Object obj = new Object();

boolean flag = true;

public void run()

{


//

System.out.println("this:"+this);


 

 

if(flag)

while(true)

{

synchronized(this)

{

if(num>0)

{

try{Thread.sleep(10);}catch (InterruptedException e){}

 

 

 

System.out.println(Thread.currentThread().getName()+".....obj...."+num--);

}

}

}

else

while(true)

this.show();

}

 

 

public synchronized void show()

{

if(num>0)

{

try{Thread.sleep(10);}catch (InterruptedException e){}

 

 

 

System.out.println(Thread.currentThread().getName()+".....function...."+num--

);

}

}

}

 

 

class SynFunctionLockDemo

{

 

96


 

 

 

public static void main(String[] args)

{

Ticket t = new Ticket();


//

System.out.println("t:"+t);


 

 

Thread t1 = new Thread(t);

Thread t2 = new Thread(t);

 

 

t1.start();

try{Thread.sleep(10);}catch(InterruptedException e){}

t.flag = false;

t2.start();

}

}

 

86.创建线程方法乊二即实现Runnable接口:

 

/*

创建线程的第一种方式:继承Thread类。

创建线程的第二种方式:实现Runnable接口。

1,定义类实现Runnable接口。

2,覆盖接口中的run方法,将线程的任务代码封装到run方法中。

3,通过Thread类创建线程对象,并将Runnable接口的子类对象作为Thread类的构造函数的参数进行传递。

为什么?因为线程的任务都封装在Runnable接口子类对象的run方法中。

所以要在线程对象创建时就必须明确要运行的任务。

4,调用线程对象的start方法开启线程。

 

 

实现Runnable接口的好处:

1,将线程的任务从线程的子类中分离出来,进行了单独的封装。

按照面向对象的思想将任务的封装成对象。

2,避免了java单继承的局限性。

所以,创建线程的第二种方式较为常用。

*/

 

 

class Demo implements Runnable//extends Fu //准备扩展Demo类的功能,让其中的内容可以作为

线程的任务执行。

//通过接口的形式完成。

{

public void run()

{

show();

}

public void show()

{

for(int x=0; x<20; x++)

{

System.out.println(Thread.currentThread().getName()+"....."+x);

 

97


 

 

 

}

}

}

 

 

class ThreadDemo

{

public static void main(String[] args)

{

Demo d = new Demo();

Thread t1 = new Thread(d);

Thread t2 = new Thread(d);

t1.start();

t2.start();


 

 

//

//

//

//

 

Demo d1 = new Demo();

Demo d2 = new Demo();

d1.start();

d2.start();


}

}

/*

class Thread

{

private Runnable r;

Thread()

{

 

 

}

Thread(Runnable r)

{

this.r = r;

}

 

 

public void run()

{

if(r!=null)

r.run();

}

 

 

public void start()

{

run();

}

}

class ThreadImpl implements Runnable

{

public void run()

 

98


 

 

 

{

System.out.println("runnable run");

}

}

ThreadImpl i = new ThreadImpl();

Thread t = new Thread(i);

t.start();

 

 

class SubThread extends Thread

{

public void run()

{

System.out.println("hahah");

}

}

//SubThread s = new SubThread();

//s.start();

*/

 

87.多线程典型案例乊卖票:

 

/*

需求:卖票。

*/

 

 

/*

线程安全问题产生的原因:

1,多个线程在操作共享的数据。

2,操作共享数据的线程代码有多条。

当一个线程在执行操作共享数据的多条代码过程中,其他线程参与了运算。

就会导致线程安全问题的产生。

 

 

解决思路;

就是将多条操作共享数据的线程代码封装起来,当有线程在执行这些代码的时候,

其他线程时不可以参与运算的。

必须要当前线程把这些代码都执行完毕后,其他线程才可以参与运算。

java中,用同步代码块就可以解决这个问题。

 

 

同步代码块的格式:

synchronized(对象)

{

需要被同步的代码 ;

}

同步的好处:解决了线程的安全问题。

同步的弊端:相对降低了效率,因为同步外的线程的都会判断同步锁。

同步的前提:同步中必须有多个线程并使用同一个锁。

*/

 

99


 

 

 

 

class Ticket implements Runnable//extends Thread

{

private int num = 100;

 

 

Object obj = new Object();

public void run()

{

while(true)

{

synchronized(obj)

{

if(num>0)

{

try{Thread.sleep(10);}catch (InterruptedException e){}

 

 

 

System.out.println(Thread.currentThread().getName()+".....sale...."+num--);

}

}

}

}

}

 

 

class TicketDemo

{

public static void main(String[] args)

{

 

 

Ticket t = new Ticket();//创建一个线程任务对象。


 

Thread

Thread

Thread

Thread

t1

t2

t3

t4

=

=

=

=

new

new

new

new

Thread(t);

Thread(t);

Thread(t);

Thread(t);


 

 

t1.start();

t2.start();

t3.start();

t4.start();

/*


Ticket t1 =

new

Ticket();


//

//

//

Ticket t2 =

Ticket t3 =

Ticket t4 =

new

new

new

Ticket();

Ticket();

Ticket();


 

 

t1.start();

 

100


 

 

 

t1.start();//一个线程不能开启两次,会抛出无效线程状态异常

t1.start();

t1.start();

*/

}

}

 

88.线程中的join()方法:

 

class Demo implements Runnable

{

public void run()

{

for(int x=0; x<50; x++)

{

System.out.println(Thread.currentThread().toString()+"....."+x);

Thread.yield();

}

}

}

 

 

class JoinDemo

{

public static void main(String[] args) throws Exception

{

Demo d = new Demo();

 

 

Thread t1 = new Thread(d);

Thread t2 = new Thread(d);

 

 

t1.start();

t2.start();


//

 

 

//

 

 

 

 

 

//

t2.setPriority(Thread.MAX_PRIORITY);

 

 

t1.join();//t1线程要申请加入进来,运行。临时加入一个线程运算时可以使用join方法。

 

for(int x=0; x<50; x++)

{

System.out.println(Thread.currentThread()+"....."+x);


}

}

}

 

89.线程乊(多)生产者、(多)消费者模式:

 

/*

生产者,消费者。

 

101


 

 

 

多生产者,多消费者的问题。

if判断标记,只有一次,会导致不该运行的线程运行了。出现了数据错误的情况。

while判断标记,解决了线程获取执行权后,是否要运行!

 

 

notify:只能唤醒一个线程,如果本方唤醒了本方,没有意义。而且while判断标记+notify会导致死锁。

notifyAll解决了本方线程一定会唤醒对方线程的问题。

*/

 

 

class Resource

{

private String name;

private int count = 1;

private boolean flag = false;

public synchronized void set(String name)//

{

while(flag)


try{ this.wait();}catch(InterruptedException e){}//

 

 

this.name = name + count;//烤鸭1 烤鸭2 烤鸭3

count++;//2 3 4

t1

t0


System.out.println(Thread.currentThread().getName()+"...

..."+this.name);//生产烤鸭1 生产烤鸭2 生产烤鸭3

flag = true;

notifyAll();

}

 

 

public synchronized void out()// t3

{

while(!flag)

try{ this.wait();}catch(InterruptedException e){}

-4//t2 t3

System.out.println(Thread.currentThread().getName()+"...

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 


........"+this.name);//消费烤鸭1

flag = false;

notifyAll();

}

}

 

 

class Producer implements Runnable

{

private Resource r;

Producer(Resource r)

{

this.r = r;

}

public void run()

{

while(true)

 

102


 

 

 

{

r.set("烤鸭");

}

}

}

 

 

class Consumer implements Runnable

{

private Resource r;

Consumer(Resource r)

{

this.r = r;

}

public void run()

{

while(true)

{

r.out();

}

}

}

 

 

class ProducerConsumerDemo

{

public static void main(String[] args)

{

Resource r = new Resource();

Producer pro = new Producer(r);

Consumer con = new Consumer(r);


 

 

Thread t0 =

Thread t1 =

Thread t2 =

Thread t3 =

 

new

new

new

new

 

Thread(pro);

Thread(pro);

Thread(con);

Thread(con);


t0.start();

t1.start();

t2.start();

t3.start();

 

 

}

}

注意:if判断标记与notify()方法对应,while标记判断跟notifyAll()方法对应,这样才不会导致死

锁,否则任意一个搭配错的话都会导致死锁。

 

90.jdk1.5乊后新的锁解决生产者和消费者问题:

 

/*

 

103


 

 

 

jdk1.5以后将同步和锁封装成了对象。

并将操作锁的隐式方式定义到了该对象中,

将隐式动作变成了显示动作。

 

 

Lock接口: 出现替代了同步代码块或者同步函数。将同步的隐式锁操作变成现实锁操作。

同时更为灵活。可以一个锁上加上多组监视器。

lock():获取锁。

unlock():释放锁,通常需要定义finally代码块中。

Condition接口:出现替代了Object中的wait notify notifyAll方法。

将这些监视器方法单独进行了封装,变成Condition监视器对象。

可以任意锁进行组合。

await();

signal();

signalAll();

*/

import java.util.concurrent.locks.*;

 

 

class Resource

{

private String name;

private int count = 1;

private boolean flag = false;

 

 

// 创建一个锁对象。

Lock lock = new ReentrantLock();

 

 

//通过已有的锁获取该锁上的监视器对象。

// Condition con = lock.newCondition();

 

 

//通过已有的锁获取两组监视器,一组监视生产者,一组监视消费者。

Condition producer_con = lock.newCondition();

Condition consumer_con = lock.newCondition();

 

 

 

public void set(String name)// t0 t1

{

lock.lock();

try

{

while(flag)


//

try{lock.wait();}catch(InterruptedException e){}// t1 t0


try{producer_con.await();}catch(InterruptedException e){}// t1

 

 

this.name = name + count;//烤鸭1 烤鸭2 烤鸭3

count++;//2 3 4

t0


System.out.println(Thread.currentThread().getName()+"...


5.0..."+this.name);//生产烤鸭1 生产烤鸭2 生产烤鸭3

 

104


 

 

 

flag = true;


//

//

notifyAll();

con.signalAll();

consumer_con.signal();

}

finally

{

lock.unlock();

}

 

 

}

 

 

public void out()// t2 t3

{

lock.lock();

try

{

while(!flag)


//

try{this.wait();}catch(InterruptedException e){}

//t2 t3


try{cousumer_con.await();}catch(InterruptedException e){}

//t2 t3


System.out.println(Thread.currentThread().getName()+"...

者.5.0......."+this.name);//消费烤鸭1

flag = false;

//

notifyAll();

//

con.signalAll();


producer_con.signal();

}

finally

{

lock.unlock();

}

 

 

}

}

 

 

class Producer implements Runnable

{

private Resource r;

Producer(Resource r)

{

this.r = r;

}

public void run()

{

while(true)

{

r.set("烤鸭");

 

105


 

 

 

}

}

}

 

 

class Consumer implements Runnable

{

private Resource r;

Consumer(Resource r)

{

this.r = r;

}

public void run()

{

while(true)

{

r.out();

}

}

}

 

 

class ProducerConsumerDemo2

{

public static void main(String[] args)

{

Resource r = new Resource();

Producer pro = new Producer(r);

Consumer con = new Consumer(r);


 

 

Thread t0 =

Thread t1 =

Thread t2 =

Thread t3 =

 

new

new

new

new

 

Thread(pro);

Thread(pro);

Thread(con);

Thread(con);


t0.start();

t1.start();

t2.start();

t3.start();

 

 

}

}

 

91.线程间通讯:

 

/*

线程间通讯:

多个线程在处理同一资源,但是任务却不同。

*/

 

 

106


 

 

 

//资源

class Resource

{

String name;

String sex;

}

 

 

//输入

class Input implements Runnable

{

Resource r ;

// Object obj = new Object();

Input(Resource r)

{

this.r = r;

}

public void run()

{

int x = 0;

while(true)

{

synchronized(r)

{

if(x==0)

{

r.name = "mike";

r.sex = "nan";

}

else

{

r.name = "丽丽";

r.sex = "女女女女女女";

}

}

x = (x+1)%2;

 

 

}

}

}

//输出

class Output implements Runnable

{

 

 

Resource r;

// Object obj = new Object();

Output(Resource r)

{

 

107


 

 

 

this.r = r;

}

 

 

public void run()

{

while(true)

{

synchronized(r)

{

System.out.println(r.name+"....."+r.sex);

}

}

}

}

 

 

class ResourceDemo

{

public static void main(String[] args)

{

//创建资源。

Resource r = new Resource();

//创建任务。

Input in = new Input(r);

Output out = new Output(r);

//创建线程,执行路径。

Thread t1 = new Thread(in);

Thread t2 = new Thread(out);

//开启线程

t1.start();

t2.start();

}

}

 

92.线程中等待/唤醒机制:

 

/*

等待/唤醒机制。

 

 

涉及的方法:

1wait(): 让线程处于冻结状态,被wait的线程会被存储到线程池中。

2notify():唤醒线程池中一个线程(任意).

3notifyAll():唤醒线程池中的所有线程。

 

 

这些方法都必须定义在同步中。

因为这些方法是用于操作线程状态的方法。

必须要明确到底操作的是哪个锁上的线程。

 

 

108


 

 

 

为什么操作线程的方法wait notify notifyAll定义在了Object类中?

因为这些方法是监视器的方法。监视器其实就是锁。

锁可以是任意的对象,任意的对象调用的方式一定定义在Object类中。

*/

 

 

//资源

class Resource

{

String name;

String sex;

boolean flag = false;

}

 

 

//输入

class Input implements Runnable

{

Resource r ;

// Object obj = new Object();

Input(Resource r)

{

this.r = r;

}

public void run()

{

int x = 0;

while(true)

{

synchronized(r)

{

if(r.flag)

try{r.wait();}catch(InterruptedException e){}

if(x==0)

{

r.name = "mike";

r.sex = "nan";

}

else

{

r.name = "丽丽";

r.sex = "女女女女女女";

}

r.flag = true;

r.notify();

}

x = (x+1)%2;

 

 

}

 

109


 

 

 

}

}

//输出

class Output implements Runnable

{

 

 

Resource r;

// Object obj = new Object();

Output(Resource r)

{

this.r = r;

}

 

 

public void run()

{

while(true)

{

synchronized(r)

{

if(!r.flag)

try{r.wait();}catch(InterruptedException e){}

System.out.println(r.name+"....."+r.sex);

r.flag = false;

r.notify();

}

}

}

}

 

 

class ResourceDemo2

{

public static void main(String[] args)

{

//创建资源。

Resource r = new Resource();

//创建任务。

Input in = new Input(r);

Output out = new Output(r);

//创建线程,执行路径。

Thread t1 = new Thread(in);

Thread t2 = new Thread(out);

//开启线程

t1.start();

t2.start();

}

}

 

 

110


 

 

 

以上代码也可以使用synchronized同步函数实现为:

 

class Resource

{

private String name;

private String sex;

private boolean flag = false;

 

 

public synchronized void set(String name,String sex)

{

if(flag)

try{ this.wait();}catch(InterruptedException e){}

this.name = name;

this.sex = sex;

flag = true;

this.notify();

}

 

 

public synchronized void out()

{

if(!flag)

try{ this.wait();}catch(InterruptedException e){}

System.out.println(name+"...+...."+sex);

flag = false;

notify();

}

}

 

 

//输入

class Input implements Runnable

{

Resource r ;

// Object obj = new Object();

Input(Resource r)

{

this.r = r;

}

public void run()

{

int x = 0;

while(true)

{

if(x==0)

{

r.set("mike","nan");

}

else

 

111


 

 

 

{

r.set("丽丽","女女女女女女");

}

x = (x+1)%2;

}

}

}

//输出

class Output implements Runnable

{

 

 

Resource r;

// Object obj = new Object();

Output(Resource r)

{

this.r = r;

}

 

 

public void run()

{

while(true)

{

r.out();

}

}

}

 

 

class ResourceDemo3

{

public static void main(String[] args)

{

//创建资源。

Resource r = new Resource();

//创建任务。

Input in = new Input(r);

Output out = new Output(r);

//创建线程,执行路径。

Thread t1 = new Thread(in);

Thread t2 = new Thread(out);

//开启线程

t1.start();

t2.start();

}

}

 

93.停止线程的方法:

 

112


 

 

 

/*

停止线程:

1stop方法。

 

 

2run方法结束。

 

 

怎么控制线程的任务结束呢?

任务中都会有循环结构,只要控制住循环就可以结束任务。

 

 

控制循环通常就用定义标记来完成。

 

 

但是如果线程处于了冻结状态,无法读取标记。如何结束呢?

 

 

可以使用interrupt()方法将线程从冻结状态强制恢复到运行状态中来,让线程具备cpu的执行资格。

 

 

当时强制动作会发生了InterruptedException,记得要处理

*/

class StopThread implements Runnable

{

private boolean flag = true;

public synchronized void run()

{

while(flag)

{

try

{

wait();//t0 t1

}

catch (InterruptedException e)

{

System.out.println(Thread.currentThread().getName()+"....."+e);

flag = false;

}

 

 

System.out.println(Thread.currentThread().getName()+"......++++");

}

}

public void setFlag()

{

flag = false;

}

}

 

 

class StopThreadDemo

{

public static void main(String[] args)

{

 

113


 

 

 

StopThread st = new StopThread();

 

 

Thread t1 = new Thread(st);

Thread t2 = new Thread(st);

 

 

t1.start();

t2.setDaemon(true);

t2.start();

 

 

int num = 1;

for(;;)

{

if(++num==50)

{


//

 

 

//

st.setFlag();

t1.interrupt();

t2.interrupt();


break;

}

System.out.println("main...."+num);

}

 

 

System.out.println("over");

}

}

 

94.线程小练习(面试):

 

/*class Test implements Runnable

{

public void run(Thread t)

{}

}*/

//如果错误 错误发生在哪一行?错误在第一行,应该被abstract修饰

 

class ThreadTest

{

public static void main(String[] args)

{

 

 

new Thread(new Runnable()

{

public void run()

{

System.out.println("runnable run");

}

})

 

114


 

 

 

{

public void run()

{

System.out.println("subThread run");

}

}.start();

 

 

/*

new Thread()

{

public void run()

{

for(int x=0; x<50; x++)

{

System.out.println(Thread.currentThread().getName()+"....x="+x);

}

 

 

}

}.start();

 

 

for(int x=0; x<50; x++)

{

System.out.println(Thread.currentThread().getName()+"....y="+x);

}

Runnable r = new Runnable()

{

public void run()

{

for(int x=0; x<50; x++)

{

System.out.println(Thread.currentThread().getName()+"....z="+x);

}

}

};

new Thread(r).start();

 

 

*/

 

 

}

}

 

95.waitsleep的区别:

 

 

wait 和 sleep 区别?

1,wait可以指定时间也可以不指定。

sleep必须指定时间。

 

115


 

 

 

 

 

2,在同步中时,对cpu的执行权和锁的处理不同。

wait:释放执行权,释放锁。

sleep:释放执行权,不释放锁。

 

class Demo

{

 

 

void show()

{

synchronized(this)//

{

 

 

wait();//t0 t1 t2

 

 

}

}

void method()

{

synchronized(this)//t4

{

 

 

//wait();

 

 

notifyAll();

 

 

}//t4

}

}

 

 

class

{

public static void main(String[] args)

{

System.out.println("Hello World!");

}

}

 

96.多线程技术总结:

 

package src;

 

 

/*

多线程总结:

 

 

1,进程和线程的概念。

|--进程:

 

116


 

 

 

|--线程:

 

 

2jvm中的多线程体现。

|--主线程,垃圾回收线程,自定义线程。以及他们运行的代码的位置。

 

 

3,什么时候使用多线程,多线程的好处是什么?创建线程的目的?

|--当需要多部分代码同时执行的时候,可以使用。

 

 

4,创建线程的两种方式。★★★★★

|--继承Thread

|--步骤

|--实现Runnable

|--步骤

|--两种方式的区别?

 

 

5,线程的5种状态。

对于执行资格和执行权在状态中的具体特点。

|--被创建:

|--运行:

|--冻结:

|--临时阻塞:

|--消亡:

 

 

6,线程的安全问题。★★★★★

|--安全问题的原因:

|--解决的思想:

|--解决的体现:synchronized

|--同步的前提:但是加上同步还出现安全问题,就需要用前提来思考。

|--同步的两种表现方法和区别:

|--同步的好处和弊端:

|--单例的懒汉式。

|--死锁。

 

 

7,线程间的通信。等待/唤醒机制。

|--概念:多个线程,不同任务,处理同一资源。

|--等待唤醒机制。使用了锁上的 wait notify notifyAll. ★★★★★


|--生产者/消费者的问题。并多生产和多消费的问题。

while判断标记。用notifyAll唤醒对方。 


★★★★

|--JDK1.5以后出现了更好的方案,★★★

Lock接口替代了synchronized

Condition接口替代了Object中的监视方法,并将监视器方法封装成了Condition

和以前不同的是,以前一个锁上只能有一组监视器方法。现在,一个Lock锁上可以多组监视器方法对

象。

可以实现一组负责生产者,一组负责消费者。

|--waitsleep的区别。★★★★★

 

 

8,停止线程的方式。

 

117


 

 

 

|--原理:

|--表现:--中断。

 

 

9,线程常见的一些方法。

|--setDaemon()

|--join();

|--优先级

|--yield();

|--在开发时,可以使用匿名内部类来完成局部的路径开辟。

*/

 

97.线程查漏综述:

 

程序:源程序和字节码文件被称为"程序"Program),是一个静态的概念。

 

进程:执行中的程序叨做进程(Process),是一个动态的概念,进程是程序的一次动态执行过程,

 

占用特定的地址空间,每个进程由三部分组成即cpudatacode,每个进程都是独立的,保有

 

自己的cpu时间,代码和数据,即使同一份程序产生好几个线程,它们乊间还是拥有自己的三样东

 

西,缺点是内存的浪费和cpu的负担。

 

织止线程的典型方法:

 

package cn.com.b;

/**

* 终止线程的典型方法

*/

public class TestThreadCiycle implements Runnable {

 

 

String name;

boolean live=true;

 

 

public TestThreadCiycle(String name) {

super();

this.name = name;

}

 

 

@Override

public void run() {

int i=0;

while(live){

System.out.println(name+(i++));

}

}

 

118


 

 

 

 

public void terminate(){

live=false;

}

 

 

public static void main(String[] args) {

TestThreadCiycle ttc=new TestThreadCiycle("线程A");

Thread t1=new Thread(ttc);//新生状态

t1.start();//就绪状态

 

for (int i = 0; i < 1000; i++) {

System.out.println(i);

}

ttc.terminate();

System.out.println("ttc stop!");

}

 

 

 

 

 

}

 

第六章:常用类API

 

98.String类:

 

(一)String类的构造函数

package cn.itcast.p1.string.demo;

 

 

public class StringConstructorDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

/*

* 将字节数组或者字符数组转成字符串可以通过String类的构造函数完成。

*/

stringConstructorDemo2();

}

 

 

private static void stringConstructorDemo2() {

char[] arr = { 'w','a','p','q','x'};

String s = new String(arr,1,3);

 

 

System.out.println("s="+s);

 

119


 

 

 

}

 

 

public static void stringConstructorDemo() {

String s = new String();//等效于String s = ""; 不等效String s = null;

 

byte[] arr = {97,66,67,68};

 

 

String s1 = new String(arr);

System.out.println("s1="+s1);

}

 

 

}

 

 

(二)String类的特点

package cn.itcast.p1.string.demo;

 

 

public class StringDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

/*

* String类的特点:

* 字符串对象一旦被初始化就不会被改变。

*

*/

stringDemo2();

}

 

 

public static void stringDemo2() {

 

 

String s = "abc";//创建一个字符串对象在常量池中。

String s1 = new String("abc");//创建两个对象一个new一个字符串对象在堆内存中。

 

System.out.println(s==s1);//false

 

 

System.out.println(s.equals(s1));

//string类中的equals复写Object中的equals建立了string类自己的判断字符串对象是否相同

的依据。

//其实就是比较字符串内容。


 

//

//

 

 

 

 

 

}

System.out.println("s="+s);

System.out.println("s1="+s1);

 

 

 

 

 

 

 

120


 

 

 

/**

* 演示字符串定义的第一种方式,并明确字符串常量池的特点.

* 池中没有就建立,池中有,直接用。

*/

private static void stringDemo1() {

String s = "abc";//"abc"存储在字符串常量池中。


//

 

 

 

//

s = "nba";

String s1 = "abc";

System.out.println(s==s1);//true?

System.out.println("s="+s);


}

}

 

99.String类中的方法及使用:

 

package cn.itcast.p1.string.demo;

 

 

public class StringMethodDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

/*

* 按照面向对象的思想对字符串进行功能分类。

* "abcd"

*

* 1,获取:

* 1.1 获取字符串中字符的个数(长度).


*

int length();


* 1.2 根据位置获取字符。


*

char charAt(int index);


* 1.3 根据字符获取在字符串中的第一次出现的位置.


*

*

*

*

*

*

*

*

*

int indexOf(int ch)

int indexOf(int ch,int fromIndex):从指定位置进行ch的查找第一次出现位置

int indexOf(String str);

int indexOf(String str,int fromIndex);

根据字符串获取在字符串中的第一次出现的位置.

int lastIndexOf(int ch)

int lastIndexOf(int ch,int fromIndex):从指定位置进行ch的查找第一次出现位置

int lastIndexOf(String str);

int lastIndexOf(String str,int fromIndex);


* 1.4 获取字符串中一部分字符串。也叫子串.


*

*

String substring(int beginIndex, int endIndex)//包含begin 不包含end 。

String substring(int beginIndex);


*

 

121


 

 

 

*

* 2,转换。


*

*

*

*

*

*

*

*

*

*

*

*

*

*

*

*

2.1 将字符串变成字符串数组(字符串的切割)

String[] split(String regex):涉及到正则表达式.

2.2 将字符串变成字符数组。

char[] toCharArray();

2.3 将字符串变成字节数组。

byte[] getBytes();

2.4 将字符串中的字母转成大小写。

String toUpperCase():大写

String toLowerCase():小写

2.5 将字符串中的内容进行替换

String replace(char oldch,char newch);

String replace(String s1,String s2);

2.6 将字符串两端的空格去除。

String trim();

2.7 将字符串进行连接 。

String concat(string);


*

* 3,判断


*

*

*

*

*

*

*

*

3.1 两个字符串内容是否相同啊?

boolean equals(Object obj);

boolean equalsIgnoreCase(string str);忽略大写比较字符串内容。

3.2 字符串中是否包含指定字符串?

boolean contains(string str);

3.3 字符串是否以指定字符串开头。是否以指定字符串结尾。

boolean startsWith(string);

boolean endsWith(string);


*

* 4,比较。

*

*/

stringMethodDemo_4();


 

 

//

//

 

 

//

//

 

System.out.println("abc".concat("kk"));

System.out.println("abc"+"kk");

 

 

System.out.println(String.valueOf(4)+1);

System.out.println(""+4+1);

 

 

}

 

 

private static void stringMethodDemo_4() {

 

 

System.out.println("abc".compareTo("aqz"));

}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

122


 

 

 

private static void stringMethodDemo_3() {

String s = "abc";

System.out.println(s.equals("ABC".toLowerCase()));

System.out.println(s.equalsIgnoreCase("ABC"));

 

 

System.out.println(s.contains("cc"));

 

 

String str = "ArrayDemo.java";

 

 

System.out.println(str.startsWith("Array"));

System.out.println(str.endsWith(".java"));

System.out.println(str.contains("Demo"));

}

 

 

private static void stringMethodDemo_2() {

 

 

String s = "张三,李四,王五";

String[] arr = s.split(",");

 

 

for (int i = 0; i < arr.length; i++) {

System.out.println(arr[i]);

}

 

 

char[] chs = s.toCharArray();

 

 

for (int i = 0; i < chs.length; i++) {

System.out.println(chs[i]);

}

s = "ab";

byte[] bytes = s.getBytes();

for (int i = 0; i < bytes.length; i++) {

System.out.println(bytes[i]);

}

 

 

System.out.println("Abc".toUpperCase());

 

 

String s1 = "java";

String s2 = s1.replace('q', 'z');

System.out.println(s1==s2);//true


 

 

System.out.println("-"+"

 

ab c

 

".trim()+"-");


 

 

}

 

 

private static void stringMethodDemo_1() {

 

 

String s = "abcdae";

 

123


 

 

 

 

 

System.out.println("length:"+s.length());//6

 

 

System.out.println("char:"+s.charAt(2));//c//StringIndexOutOfBoundsException

System.out.println("index:"+s.indexOf('k'));//0//-1 我们可以根据-1,来判断该字

符或者字符串是否存在。

System.out.println("lastIndex:"+s.lastIndexOf('a'));//4

 

 

System.out.println("substring:"+s.substring(2,4));

}

 

 

}

 

100.String类精华练习题:

 

习题1:

package cn.itcast.p1.string.test;

 

 

/*

* 1,给定一个字符串数组。按照字典顺序进行从小到大的排序。

* {"nba","abc","cba","zz","qq","haha"}

*

* 思路:

* 1,对数组排序。可以用选择,冒泡都行。

* 2,for嵌套和比较以及换位。

* 3,问题:以前排的是整数,比较用的比较运算符,可是现在是字符串对象。


*

字符串对象怎么比较呢?爽了,对象中提供了用于字符串对象比较的功能。


*/

public class StringTest_1 {

/**

* @param args

*/

public static void main(String[] args) {

 

 

String[] arr = {  "nba", "abc", "cba", "zz", "qq", "haha" };

 

 

printArray(arr);

 

 

sortString(arr);

 

 

printArray(arr);

 

 

}

 

 

public static void sortString(String[] arr) {

 

 

for (int i = 0; i < arr.length - 1; i++) {

 

124


 

 

 

for (int j = i + 1; j < arr.length; j++) {

 

 

if(arr[i].compareTo(arr[j])>0)//字符串比较用compareTo方法

swap(arr,i,j);

}

}

}

 

 

private static void swap(String[] arr, int i, int j) {

String temp = arr[i];

arr[i] = arr[j];

arr[j] = temp;

}

 

 

public static void printArray(String[] arr) {

System.out.print("[");

for (int i = 0; i < arr.length; i++) {

if (i != arr.length - 1)

System.out.print(arr[i] + ", ");

else

System.out.println(arr[i] + "]");

}

}

 

 

}

 

 

习题2:

package cn.itcast.p1.string.test;

/*

* 2,一个子串在整串中出现的次数。

* "nbaernbatynbauinbaopnba        "

* 思路:

* 1,要找的子串是否存在,如果存在获取其出现的位置。这个可以使用indexOf完成。

* 2,如果找到了,那么就记录出现的位置并在剩余的字符串中继续查找该子串,

* 而剩余字符串的起始位是出现位置+子串的长度.

* 3,以此类推,通过循环完成查找,如果找不到就是-1,并对 每次找到用计数器记录。

*/

public class StringTest_2 {

 

 

public static void main(String[] args) {

 

 

String str = "nbaernbatnbaynbauinbaopnba";

String key = "nba";

 

 

int count = getKeyStringCount_2(str,key);

System.out.println("count="+count);

 

 

125


 

 

 

}

 

 

public static int getKeyStringCount_2(String str, String key) {

 

 

int count = 0;

int index = 0;

 

 

while((index = str.indexOf(key,index))!=-1){

 

 

index = index + key.length();

count++;

}

return count;

}

 

 

/**

* 获取子串在整串中出现的次数。

* @param str

* @param key

* @return

*/

public static int getKeyStringCount(String str, String key) {

//1,定义计数器。

int count = 0;

//2,定义变量记录key出现的位置。

int index = 0;

while((index = str.indexOf(key))!=-1){

str = str.substring(index+key.length());

count++;

}

return count;

}

}

 

 

习题3:

package cn.itcast.p1.string.test;

/*

* 3,两个字符串中最大相同的子串。

* "qwerabcdtyuiop"

* "xcabcdvbn"

*

* 思路:

* 1,既然取得是最大子串,先看短的那个字符串是否在长的那个字符串中。

* 如果存在,短的那个字符串就是最大子串。

* 2,如果不是呢,那么就将短的那个子串进行长度递减的方式去子串,去长串中判断是否存在。

* 如果存在就已找到,就不用在找了。

*/

 

126


 

 

 

public class StringTest_3 {

/**

* @param args

*/

public static void main(String[] args) {

 

 

String s1 = "qwerabcdtyuiop";

String s2 = "xcabcdvbn";

 

 

String s = getMaxSubstring(s2, s1);

System.out.println("s=" + s);

}

/**

* 获取最大子串

* @param s1

* @param s2

* @return

*/

public static String getMaxSubstring(String s1, String s2) {

 

 

String max = null,min = null;

max = (s1.length()>s2.length())?s1:s2;

 

 

min = max.equals(s1)?s2:s1;

 

 

System.out.println("max="+max);

System.out.println("min="+min);

 

 

for (int i = 0; i < min.length(); i++) {

 

 

for(int a = 0,b = min.length()-i; b != min.length()+1; a++,b++){

 

 

String sub = min.substring(a, b);


//

System.out.println(sub);


if(max.contains(sub))

return sub;

}

}

 

 

return null;

}

}

 

 

习题4:

package cn.itcast.p1.string.test;

/*

* 4,模拟一个trim功能一致的方法。去除字符串两端的空白

 

127


 

 

 

* 思路:

* 1,定义两个变量。

* 一个变量作为从头开始判断字符串空格的角标。不断++

* 一个变量作为从尾开始判断字符串空格的角标。不断--

* 2,判断到不是空格为止,取头尾之间的字符串即可。

*/

public class StringTest_4 {

 

 

/**

* @param args

*/

public static void main(String[] args) {


 

 

String s = "

 

ab

 

c

 

";


s = myTrim(s);

System.out.println("-" + s + "-");

}

 

 

public static String myTrim(String s) {

 

 

int start = 0, end = s.length() - 1;

 

 

while (start <= end && s.charAt(start) == ' ') {

start++;

}

while (start <= end && s.charAt(end) == ' ') {

end--;

}

return s.substring(start, end + 1);

}

 

 

}

 

101.StringBuffer及其方法使用:

 

package cn.itcast.p2.stringbuffer.demo;

 

 

public class StringBufferDemo {

/**

* @param args

*/

public static void main(String[] args) {

/*


*

*

*

*

StringBuffer:就是字符串缓冲区。

用于存储数据的容器。

特点:

1,长度的可变的。

 

 

 

 

 

 

128


 

 

 

* 2,可以存储不同类型数据。

* 3,最终要转成字符串进行使用。

* 4,可以对字符串进行修改。

*

* 既然是一个容器对象。应该具备什么功能呢?

* 1,添加:


*

*

StringBuffer append(data);

StringBuffer insert(index,data);


* 2,删除:


*

*

StringBuffer delete(start,end):包含头,不包含尾。

StringBuffer deleteCharAt(int index):删除指定位置的元素


* 3,查找:


*

*

*

char charAt(index);

int indexOf(string);

int lastIndexOf(string);


* 4,修改:


*

*

StringBuffer replace(start,end,string);

void setCharAt(index,char);


*

* 增删改查 C(create)U(update)R(read)D(delete)

*/

bufferMethodDemo_2();

}

 

 

private static void bufferMethodDemo_2() {

StringBuffer sb = new StringBuffer("abce");


 

 

//

 

 

 

//

//

 

 

//

//

 

 

//

 

 

//

//

 

sb.delete(1, 3);//ae

 

 

//清空缓冲区。

sb.delete(0,sb.length());

sb = new StringBuffer();

 

 

sb.replace(1, 3, "nba");

sb.setCharAt(2, 'q');

 

 

sb.setLength(10);

 

 

System.out.println("sb:"+sb);

System.out.println("len:"+sb.length());

 

 

System.out.println(sb.reverse());

}

 

 

private static void bufferMethodDemo_1() {

StringBuffer sb = new StringBuffer("abce");

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

129


 

 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

 

 

sb.append("xixi");

sb.insert(2, "qq");

 

 

System.out.println(sb.toString());

}

 

 

public static void bufferMethodDemo(){

//创建缓冲区对象。

StringBuffer sb = new StringBuffer();

 

 

sb.append(4).append(false);//.append("haha");

sb.insert(1, "haha");

sb.append(true);


System.out.println(sb);

}

}

 

102.StringBuilder使用:

 

package cn.itcast.p2.stringbuffer.demo;

 

 

public class StringBuilderDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

 

/*

* jdk1.5以后出现了功能和StringBuffer一模一样的对象。就是StringBuilder

*

* 不同的是:

* StringBuffer是线程同步的。通常用于多线程。

* StringBuilder是线程不同步的。通常用于单线程。 它的出现提高效率。

*

* jdk升级:

* 1,简化书写。

* 2,提高效率。

* 3,增加安全性。

*/

}

}

 

 

习题:

package cn.itcast.p2.stringbuffer.test;

 

 

130


 

 

 

public class StringBuilderTest {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

int[] arr = {3,1,5,3,8};

String s = arrayToString_2(arr);

 

 

System.out.println(s);

}

public static String arrayToString_2(int[] arr){

 

 

StringBuilder sb = new StringBuilder();

sb.append("[");

for (int i = 0; i < arr.length; i++) {

if(i!=arr.length-1)

sb.append(arr[i]+", ");

else

sb.append(arr[i]+"]");

}

return sb.toString();

}

/**

* 将一个int数组变成字符串。

*/

public static String arrayToString(int[] arr){

 

 

String str = "[";

for (int i = 0; i < arr.length; i++) {

if(i!=arr.length-1)

str+=arr[i]+", ";

else

str+=arr[i]+"]";

}

 

 

return str;

}

 

 

}


 

 

103.

包装类及方法使用:


 

package cn.itcast.p2.wrapper.demo;

 

 

public class WrapperDemo {

 

131


 

 

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

/*


*

*

*

*

*

基本数据类型对象包装类。

为了方便操作基本数据类型值,将其封装成了对象,在对象中定义了属性和行为丰富了该数据的操

作。

用于描述该对象的类就称为基本数据类型对象包装类。


*

*

*

*

*

byte

short

int

long

float

Byte

Short

Integer

Long

Float


*

double Double


*

char

Character


*

booleanBoolean


*

* 该包装对象主要用基本类型和字符串之间的转换。

*

* 基本类型--->字符串


*

*

*

1,基本类型数值+""

2,String类中的静态方法valueOf(基本类型数值);

3,Integer的静态方法valueO(基本类型数值);


*

* 字符串--->基本类型


*

1,使用包装类中的静态方法

xxx parseXxx("xxx类型的字符串");*****


*

*

*

*

*

*

*

*/

int parseInt("intstring");

long parseLong("longstring");

boolean parseBoolean("booleanstring");

只有Character没有parse方法

2,如果字符串被Integer进行对象的封装。

可使用另一个非静态的方法,intValue();

将一个Integer对象转成基本数据类型值。


 

 

//

 

 

//

 

 

//

//

 

 

//

 

System.out.println(Integer.MAX_VALUE);

 

 

System.out.println(Integer.toBinaryString(-6));

 

 

int num = 4;

Integer i = new Integer(5);

 

 

int x = Integer.parseInt("123");

 

 

 

 

 

 

 

 

 

 

 

 

 

 

132


 

 

 

//

 

 

//

//

 

 

 

System.out.println(Integer.parseInt("123")+1);

 

 

Integer i = new Integer("123");

System.out.println(i.intValue());

 

 

/*

* 整数具备不同的进制体现。

*

* 十进制-->其他进制。


*

*

*

toBinaryString

toOctalString

toHexString


*

* 其他进制-->十进制。


*

*

*/

parseInt("string",radix)


 

 

//

 

 

 

 

 

 

//

//

//

 

 

 

 

 

 

 

 

 

 

 

 

//

 

十进制-->其他进制。

System.out.println(Integer.toBinaryString(60));

System.out.println(Integer.toOctalString(60));

System.out.println(Integer.toHexString(60));

 

 

System.out.println(Integer.toString(60,16));

其他进制-->十进制。

System.out.println(Integer.parseInt("3c",16));

 

 

Integer a = new Integer("89");

Integer b = new Integer(300);

 

 

System.out.println(a==b);

 

 

System.out.println(a.equals(b));

 

 

System.out.println(3>3);


System.out.println(a.compareTo(b));

 

 

}

 

 

}


 

 

104.

自动装箱:


 

package cn.itcast.p2.wrapper.demo;

 

 

public class WrapperDemo2 {

 

 

133


 

 

 

public static void main(String[] args) {

 

 

int num = 4;

num = num + 5;

 

 

Integer i = 4;//i = new Integer(4);自动装箱 简化书写。

i = i + 6;// i = new Integer(i.intValue() + 6); //i.intValue() 自动拆箱


 

//

show(55);//


 

 

Integer a = new Integer(128);

Integer b = new Integer(128);

 

 

System.out.println(a==b);//false

System.out.println(a.equals(b));//true

 

 

Integer x = 129;//jdk1.5以后,自动装箱,如果装箱的是一个字节,那么该数据会被共享不会

重新开辟空间。

Integer y = 129;

System.out.println(x==y);//

System.out.println(x.equals(y));//true

 

 

}

 

 

public static void show(Object a){ //Object a = new Integer(55);

System.out.println("a="+a);

}

 

 

}


 

 

105.

包装类练习题:


 

package cn.itcast.p2.wrapper.test;

 

 

import java.util.Arrays;

 

 

/*

* 对一个字符串中的数值进行从小到大的排序。

*

* "20 78 9 -7 88 36 29"

*


*

*

*

*

*

*

思路:

1,排序,我很熟。可是我只熟int

2,如何获取到这个字符串中的这些需要排序的数值?

发现这个字符串中其实都是空格来对数值进行分隔的。

所以就想到用字符串对象的切割方法将大串变成多个小串。

3,数值最终变成小字符串,怎么变成一个int数呢?

 

 

 

 

 

 

 

 

 

134


 

 

 

* 字符串-->基本类型 可以使用包装类。

*

*

*/

 

 

public class WrapperTest {

 

 

private static final String SPACE_SEPARATOR = " ";

/**

* @param args

*/

public static void main(String[] args) {

 

 

String numStr = "20 78 9 -7 88 36 29";

 

 

System.out.println(numStr);

numStr = sortStringNumber(numStr);

System.out.println(numStr);

 

 

}

/**

*

* @param numStr

* @return

*/

public static String sortStringNumber(String numStr) {

 

 

//1,将字符串变成字符串数组。

String[] str_arr = stringToArray(numStr);

 

 

//2,将字符串数组变成int数组。

 

int[] num_arr = toIntArray(str_arr);

 

 

//3,int数组排序。

mySortArray(num_arr);

 

 

//4,将排序后的int数组变成字符串。

 

String temp = arrayToString(num_arr);

 

 

return temp;

}

 

 

public static String arrayToString(int[] num_arr) {

 

 

StringBuilder sb = new StringBuilder();

 

135


 

 

 

for(int x = 0; xlength; x++){

if(x!=num_arr.length-1)

sb.append(num_arr[x]+SPACE_SEPARATOR);

else

sb.append(num_arr[x]);

}

 

 

return sb.toString();

}

 

 

public static void mySortArray(int[] num_arr) {

 

 

Arrays.sort(num_arr);

}

 

 

public static int[] toIntArray(String[] str_arr) {

 

 

int[] arr = new int[str_arr.length];

 

 

for (int i = 0; i < arr.length; i++) {

arr[i] = Integer.parseInt(str_arr[i]);

}

 

 

return arr;

}

 

 

/**

* @param numStr

*/

public static String[] stringToArray(String numStr) {

String[] str_arr = numStr.split(SPACE_SEPARATOR);

 

 

 

return str_arr;

}

 

 

}

 

第七章:集合框架(容器)


 

 

106.

集合常用方法演示:


 

package cn.itcast.p3.collection.demo;

 

 

import java.util.ArrayList;

import java.util.Collection;

 

136


 

 

 

 

public class CollectionDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

Collection coll = new ArrayList();


 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

 

 

//

//

 

 

 

//

//

 

show(coll);

 

 

Collection c1 = new ArrayList();

Collection c2 = new ArrayList();

show(c1,c2);

 

 

}

 

 

public static void show(Collection c1,Collection c2){

 

 

//给c1添加元素。

c1.add("abc1");

c1.add("abc2");

c1.add("abc3");

c1.add("abc4");

 

 

//给c2添加元素。

c2.add("abc1");

c2.add("abc2");

c2.add("abc3");

c2.add("abc4");

c2.add("abc5");

 

 

System.out.println("c1:"+c1);

System.out.println("c2:"+c2);

 

 

//演示addAll

 

 

c1.addAll(c2);//将c2中的元素添加到c1中。

 

 

//演示removeAll

boolean b = c1.removeAll(c2);//将两个集合中的相同元素从调用removeAll的集合中删除。

System.out.println("removeAll:"+b);

 

 

//演示containsAll

boolean b = c1.containsAll(c2);

System.out.println("containsAll:"+b);

 

 

137


 

 

 

//演示retainAll

boolean b = c1.retainAll(c2);//取交集,保留和指定的集合相同的元素,而删除不同的元素。

//和removeAll功能相反 。

System.out.println("retainAll:"+b);

System.out.println("c1:"+c1);

 

 

}

 

 

public static void show(Collection coll){

 

 

//1,添加元素。add.

coll.add("abc1");

coll.add("abc2");

coll.add("abc3");

System.out.println(coll);

 

 

//2,删除元素。remove


//

 

 

 

//

coll.remove("abc2");//会改变集合的长度

 

 

//清空集合.

coll.clear();


 

 

System.out.println(coll.contains("abc3"));

 

 

System.out.println(coll);

 

 

}

 

 

}


 

 

107.

Iterator迭代器的使用:


 

package cn.itcast.p3.collection.demo;

 

 

import java.util.ArrayList;

import java.util.Collection;

import java.util.Iterator;

 

 

public class IteratorDemo {

/**

* @param args

*/

public static void main(String[] args) {

 

 

Collection coll = new ArrayList();

coll.add("abc1");

coll.add("abc2");

 

138


 

 

 

coll.add("abc3");

coll.add("abc4");


 

 

//

 

System.out.println(coll);


 

 

//使用了Collection中的iterator()方法。 调用集合中的迭代器方法,是为了获取集合中的迭代

器对象。


//

//

//

//

 

 

 

 

 

 

 

 

//

//

//

//

//

 

 

 

 

 

}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

}

Iterator it = coll.iterator();

while(it.hasNext()){

System.out.println(it.next());

}

 

 

for(Iterator it = coll.iterator(); it.hasNext(); ){

System.out.println(it.next());

}

 

 

System.out.println(it.next());

System.out.println(it.next());

System.out.println(it.next());

System.out.println(it.next());

System.out.println(it.next());//java.util.NoSuchElementException


 

 

108.

ArrayList


 

演示 1:

package cn.itcast.p4.list.demo;

 

 

import java.util.ArrayList;

import java.util.List;

 

 

public class ListDemo {

/**

* @param args

*/

public static void main(String[] args) {

List list = new ArrayList();

show(list);

}

 

 

public static void show(List list) {

 

 

//添加元素

list.add("abc1");

 

139


 

 

 

list.add("abc2");

list.add("abc3");

System.out.println(list);

 

 

//插入元素。


//

 

 

 

//

 

 

 

//

 

 

 

//

 

 

 

//

list.add(1,"abc9");

 

 

//删除元素。

System.out.println("remove:"+list.remove(2));

 

 

//修改元素。

System.out.println("set:"+list.set(1, "abc8"));

 

 

//获取元素。

System.out.println("get:"+list.get(0));

 

 

//获取子列表。

System.out.println("sublist:"+list.subList(1, 2));


 

 

System.out.println(list);

 

 

}

 

 

}

 

 

演示 2:

package cn.itcast.p4.list.demo;


 

 

import

import

import

import

 

java.util.ArrayList;

java.util.Iterator;

java.util.List;

java.util.ListIterator;


 

 

 

public class ListDemo2 {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

List list = new ArrayList();


//

show(list);

 

 

list.add("abc1");

list.add("abc2");

list.add("abc3");

 

 

 

 

 

 

 

140


 

 

 

 

System.out.println("list:"+list);

ListIterator it = list.listIterator();//获取列表迭代器对象

//它可以实现在迭代过程中完成对元素的增删改查。

//注意:只有list集合具备该迭代功能.

 

while(it.hasNext()){

 

 

Object obj = it.next();

 

 

if(obj.equals("abc2")){

it.set("abc9");

}

}


//

//

System.out.println("hasNext:"+it.hasNext());

System.out.println("hasPrevious:"+it.hasPrevious());

 

 

while(it.hasPrevious()){

System.out.println("previous:"+it.previous());

}

System.out.println("list:"+list);

 

 

/*Iterator it = list.iterator();

while(it.hasNext()){

 

 

Object obj = it.next();//java.util.ConcurrentModificationException

//在迭代器过程中,不要使用集合操作元素,容易出现异常。

//可以使用Iterator接口的子接口ListIterator来完成在迭代中对元素进行更

//多的操作。

 

if(obj.equals("abc2")){

list.add("abc9");

}

else

System.out.println("next:"+obj);

}

System.out.println(list);

*/

 

 

}

 

 

public static void show(List list) {

 

 

list.add("abc1");

list.add("abc2");

list.add("abc3");

list.add("abc4");

 

141


 

 

 

 

Iterator it = list.iterator();

while(it.hasNext()){

System.out.println("next:"+it.next());

}

 

 

//list特有的取出元素的方式之一。

 

for(int x=0; x

System.out.println("get:"+list.get(x));

}

}

 

 

}


 

 

109.

Vector演示:


 

package cn.itcast.p1.vector.demo;

 

 

import java.util.Enumeration;

import java.util.Iterator;

import java.util.Vector;

 

 

public class VectorDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

Vector v = new Vector();

 

 

v.addElement("abc1");

v.addElement("abc2");

v.addElement("abc3");

v.addElement("abc4");

 

 

Enumeration en = v.elements();

while(en.hasMoreElements()){

System.out.println("nextelment:"+en.nextElement());

}

 

 

Iterator it = v.iterator();

 

 

while(it.hasNext()){

System.out.println("next:"+it.next());

}

 

142


 

 

 

 

}

 

 

}


 

 

110.

LinkedList演示:


 

package cn.itcast.p2.linkedlist.demo;

 

 

import java.util.Iterator;

import java.util.LinkedList;

 

 

public class LinkedListDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

LinkedList link = new LinkedList();

 

 

link.addFirst("abc1");

link.addFirst("abc2");

link.addFirst("abc3");

link.addFirst("abc4");


//

//

//

 

 

//

//

 

 

 

 

 

 

 

 

 

//

//

//

//

 

 

}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

}

System.out.println(link);

System.out.println(link.getFirst());//获取第一个但不删除。

System.out.println(link.getFirst());

 

 

System.out.println(link.removeFirst());//获取元素但是会删除。

System.out.println(link.removeFirst());

 

 

while(!link.isEmpty()){

System.out.println(link.removeLast());

}

 

 

System.out.println(link);

Iterator it = link.iterator();

while(it.hasNext()){

System.out.println(it.next());

}


 

 

111.

队列的添加元素功能(先进先出):


 

package cn.itcast.p2.linkedlist.test;

 

143


 

 

 

 

import java.util.LinkedList;

 

 

public class DuiLie {

private LinkedList link;

 

 

public DuiLie() {

link = new LinkedList();

}

 

 

/**

* 队列的添加元素的功能。

*/

public void myAdd(Object obj) {

link.addLast(obj);

}

 

 

public Object myGet() {

return link.removeFirst();

}

 

 

public boolean isNull() {

return link.isEmpty();

}

}

 

 

习题:

package cn.itcast.p2.linkedlist.test;

 

 

import java.util.ArrayList;

 

 

/*

* 请使用LinkedList来模拟一个堆栈或者队列数据结构。

*

* 堆栈:先进后出 First In Last Out FILO

*

* 队列:先进先出 First In First Out FIFO

*

* 我们应该描述这样一个容器,给使用提供一个容器对象完成这两种结构中的一种。

*/

public class LinkedTest {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

144


 

 

 

DuiLie dl = new DuiLie();

 

 

dl.myAdd("abc1");

dl.myAdd("abc2");

dl.myAdd("abc3");

dl.myAdd("abc4");

 

 

while(!dl.isNull()){

System.out.println(dl.myGet());

}

 

 

}

 

 

}

 

 

注意:解体的核心思想还是运用已有的、已知具备功能的方法来作为自己定义方法的内容来实现模拟的功能。


 

 

112.

ArrayList习题演示:


 

习题1:

package cn.itcast.p3.arraylist.test;

 

 

import java.util.ArrayList;

import java.util.Iterator;

 

 

import cn.itcast.p.bean.Person;

 

 

public class ArrayListTest {

 

 

/**

* @param args

*/

public static void main(String[] args) {

Person p1 = new Person("lisi1",21);

 

 

ArrayList al = new ArrayList();

al.add(p1);

al.add(new Person("lisi2",22));

al.add(new Person("lisi3",23));

al.add(new Person("lisi4",24));

 

 

Iterator it = al.iterator();

while(it.hasNext()){


//

System.out.println(((Person)   it.next()).getName()+"::"+((Person)


it.next()).getAge());

Person p = (Person) it.next();

System.out.println(p.getName()+"--"+p.getAge());

 

145


 

 

 

}


//

al.add(5);//al.add(new Integer(5));


}

 

 

 

}

习题2:

package cn.itcast.p3.arraylist.test;

 

 

import java.util.ArrayList;

import java.util.Iterator;

 

 

import cn.itcast.p.bean.Person;

 

 

/*

* 定义功能去除ArrayList中的重复元素。

*/

 

 

public class ArrayListTest2 {

 

 

public static void main(String[] args) {


 

 

//

 

 

//

 

demo();

 

 

singleDemo();

 

 

ArrayList al = new ArrayList();

al.add(new Person("lisi1",21));

al.add(new Person("lisi2",22));

al.add(new Person("lisi3",23));

al.add(new Person("lisi4",24));

al.add(new Person("lisi2",22));

al.add(new Person("lisi3",23));

System.out.println(al);

 

 

al = getSingleElement(al);

 

 

System.out.println(al.remove(new Person("lisi2",22)));

System.out.println(al);

 

 

}

 

 

public static void singleDemo() {

ArrayList al = new ArrayList();

al.add("abc1");

al.add("abc2");

al.add("abc2");

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

146


 

 

 

al.add("abc1");

al.add("abc");

System.out.println(al);

 

 

al = getSingleElement(al);

 

 

System.out.println(al);

 

 

}

 

 

public static ArrayList getSingleElement(ArrayList al) {

 

 

//1,定义一个临时容器。

ArrayList temp = new ArrayList();

 

 

//2,迭代al集合。

Iterator it = al.iterator();

 

 

while(it.hasNext()){

Object obj = it.next();

 

 

//3,判断被迭代到的元素是否在临时容器存在。

if(!temp.contains(obj)){

temp.add(obj);

}

}

 

 

return temp;

}

 

 

public static void demo() {


 

 

//

 

 

 

}

 

 

}

 

al.add(5);//al.add(new Integer(5));


 

 

113.

HashSet演示:


 

演示1:

package cn.itcast.p4.hashset.demo;

 

 

import java.util.HashSet;

import java.util.Iterator;

 

 

public class HashSetDemo {

 

 

147


 

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

HashSet hs = new HashSet();

 

 

hs.add("hehe");


//

hs.add("heihei");


hs.add("hahah");

hs.add("xixii");

hs.add("hehe");

 

 

Iterator it = hs.iterator();

 

 

while(it.hasNext()){

System.out.println(it.next());

}

}

 

 

}

演示2:

package cn.itcast.p4.hashset.test;

 

 

import java.util.HashSet;

import java.util.Iterator;

 

 

import cn.itcast.p.bean.Person;

/*

* 往hashSet集合中存储Person对象。如果姓名和年龄相同,视为同一个人。视为相同元素。

*/

public class HashSetTest {

 

 

public static void main(String[] args) {

 

 

HashSet hs = new HashSet();

/*

* HashSet集合数据结构是哈希表,所以存储元素的时候,

* 使用的元素的hashCode方法来确定位置,如果位置相同,在通过元素的equals来确定是否相同。

*

*/

hs.add(new Person("lisi4",24));

hs.add(new Person("lisi7",27));

hs.add(new Person("lisi1",21));

hs.add(new Person("lisi9",29));

hs.add(new Person("lisi7",27));

 

 

148


 

 

 

Iterator it = hs.iterator();

 

 

while(it.hasNext()){

Person p = (Person)it.next();

System.out.println(p);


//

 

 

 

 

 

}

 

 

}

 

}

System.out.println(p.getName()+"...."+p.getAge());


 

 

114.

LinkedHashSet使用:


 

package cn.itcast.p4.hashset.demo;

 

 

import java.util.HashSet;

import java.util.Iterator;

import java.util.LinkedHashSet;

 

 

public class LinkedHashSetDemo {

 

 

public static void main(String[] args) {

 

 

HashSet hs = new LinkedHashSet();

 

 

hs.add("hahah");

hs.add("hehe");

hs.add("heihei");

hs.add("xixii");


//

hs.add("hehe");


 

 

Iterator it = hs.iterator();

 

 

while(it.hasNext()){

System.out.println(it.next());

}

}

 

 

}


 

 

115.

TreeSet和比较方法(实现Comparator接口)使用


 

的习题:

 

习题1:

/**

 

149


 

 

 

* 创建了一个根据Person类的name进行排序的比较器。

*/

class ComparatorByName implements Comparator {

 

 

@Override

public int compare(Object o1, Object o2) {

 

 

Person p1 = (Person)o1;

Person p2 = (Person)o2;

 

 

int temp = p1.getName().compareTo(p2.getName());

 

 

return temp==0?p1.getAge()-p2.getAge(): temp;


//

return 1;//有序。


}

 

 

}

 

 

 

public class TreeSetDemo {

 

 

public static void main(String[] args) {

 

 

TreeSet ts = new TreeSet(new ComparatorByName());

 

 

/*

* 以Person对象年龄进行从小到大的排序。

*

*/

ts.add(new Person("zhangsan",28));

ts.add(new Person("lisi",21));

ts.add(new Person("zhouqi",29));

ts.add(new Person("zhaoliu",25));

ts.add(new Person("wangu",24));

 

 

Iterator it = ts.iterator();

 

 

while(it.hasNext()){

Person p = (Person)it.next();

 

 

System.out.println(p.getName()+":"+p.getAge());

}

 

 

}

 

 

public static void demo1() {

TreeSet ts = new TreeSet();

 

150


 

 

 

 

ts.add("abc");

ts.add("zaa");

ts.add("aa");

ts.add("nba");

ts.add("cba");

 

 

Iterator it = ts.iterator();

 

 

while(it.hasNext()){

System.out.println(it.next());

}

}

 

 

}

习题2:

/*

定义一个按长度作比较的比较器

*/

class ComparatorByLength implements Comparator {

 

 

@Override

public int compare(Object o1, Object o2) {

 

 

String s1 = (String)o1;

String s2 = (String)o2;

 

 

int temp = s1.length()-s2.length();

 

 

return temp==0? s1.compareTo(s2): temp;

}

 

 

}

 

 

/*

* 对字符串进行长度排序。

*

* "20 18 -1 89 2 67"

*/

 

 

public class TreeSetTest {

 

 

public static void main(String[] args) {

 

 

TreeSet ts = new TreeSet(new ComparatorByLength());

 

 

ts.add("aaaaa");

 

151


 

 

 

ts.add("zz");

ts.add("nbaq");

ts.add("cba");

ts.add("abc");

 

 

Iterator it = ts.iterator();

 

 

while(it.hasNext()){

System.out.println(it.next());

}

}

 

 

}


 

 

 

116.

 

比 较 的 另 一 种 方 法 即 让 对 象 实 现 Comparable 


 

口,即让对象本身具备可比较性并重写compareTo()

 

方法 , 实际 上就 是 改发 return 表 达 式, 但 不如 实现

 

Comparator接口进行比较更为灵活,案例:

 

package cn.itcast.p.bean;

 

 

public class Person /*extends Object*/ implements Comparable {

 

 

private String name;

private int age;

 

 

public Person() {

super();

 

 

}

public Person(String name, int age) {

super();

this.name = name;

this.age = age;

}

 

 

@Override

public int hashCode() {


//

System.out.println(this+".......hashCode");

 

 

 

152


 

 

 

return name.hashCode()+age*27;


//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

//

return 100;

}

@Override

public boolean equals(Object obj) {

 

 

if(this == obj)

return true;

if(!(obj instanceof Person))

throw new ClassCastException("类型错误");

 

System.out.println(this+"....equals....."+obj);

Person p = (Person)obj;

 

 

return this.name.equals(p.name) && this.age == p.age;

}

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

public int getAge() {

return age;

}

public void setAge(int age) {

this.age = age;

}

public String toString(){

return name+":"+age;

}

@Override

public int compareTo(Object o) {

 

 

Person p = (Person)o;

 

 

int temp = this.age-p.age;

return temp==0?this.name.compareTo(p.name):temp;

 

 

int temp = this.name.compareTo(p.name);

return temp==0?this.age-p.age:temp;

 

 

/*

if(this.age>p.age)

return 1;

if(this.age

return -1;

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

153


 

 

 

 

else{

 

 

return this.name.compareTo(p.name);

}

*/

 

 

}

 

 

}


 

 

 

117.

 

集合(ListSet)部分总结:


 

集合类的由来:

对象用于封装特有数据,对象多了需要存储,如果对象的个数不确定。

就使用集合容器进行存储。

 

 

集合特点:

1,用于存储对象的容器。

2,集合的长度是可变的。

3,集合中不可以存储基本数据类型值。

 

 

 

集合容器因为内部的数据结构不同,有多种具体容器。

不断的向上抽取,就形成了集合框架。

 

 

框架的顶层Collection接口:

 

 

Collection的常见方法:

 

 

1,添加。

boolean add(Object obj):

boolean addAll(Collection coll):

 

 

 

2,删除。

boolean remove(object obj):

boolean removeAll(Collection coll);

void clear();

 

 

3,判断:

boolean contains(object obj):

boolean containsAll(Colllection coll);

boolean isEmpty():判断集合中是否有元素。

 

 

4,获取:

 

154


 

 

 

int size():

Iterator iterator():取出元素的方式:迭代器。

该对象必须依赖于具体容器,因为每一个容器的数据结构都不同。

所以该迭代器对象是在容器中进行内部实现的。

对于使用容器者而言,具体的实现不重要,只要通过容器获取到该实现的迭代器的对象即可,

也就是iterator方法。

 

 

Iterator接口就是对所有的Collection容器进行元素取出的公共接口。

其实就是抓娃娃游戏机中的夹子!

 

 

 

5,其他:

boolean retainAll(Collection coll);取交集。

Object[] toArray():将集合转成数组。

 

 

 

--------------------------------------------------------------------------------

Collection

|--List:有序(存入和取出的顺序一致),元素都有索引(角标),元素可以重复。

|--Set:元素不能重复,无序。

 

 

 

List:特有的常见方法:有一个共性特点就是都可以操作角标。

 

 

1,添加

void add(index,element);

void add(index,collection);

 

 

 

2,删除;

Object remove(index):

 

 

 

3,修改:

Object set(index,element);

 

 

 

4,获取:

Object get(index);

int indexOf(object);

int lastIndexOf(object);

List subList(from,to);

 

 

 

list集合是可以完成对元素的增删改查。

 

 

 

 

155


 

 

 

List:

|--Vector:内部是数组数据结构,是同步的。增删,查询都很慢!

|--ArrayList:内部是数组数据结构,是不同步的。替代了Vector。查询的速度快。

|--LinkedList:内部是链表数据结构,是不同步的。增删元素的速度很快。

 

 

 

LinkedList:

 

 

addFirst();

addLast():

jdk1.6

offerFirst();

offetLast();

 

 

 

getFirst();.//获取但不移除,如果链表为空,抛出NoSuchElementException.

getLast();

jdk1.6

peekFirst();//获取但不移除,如果链表为空,返回null.

peekLast():

 

 

removeFirst();//获取并移除,如果链表为空,抛出NoSuchElementException.

removeLast();

jdk1.6

pollFirst();//获取并移除,如果链表为空,返回null.

pollLast();

 

 

 

作业:

1,自己去查文档演示Vector中的elements()方法。

 

 

2,LinkedList中的,addFirst addLast getFirst,getLast removeFirst removeLast。

 

 

3,既然集合是存储对象的,请定义ArryaList集合,并存储Person对象。如new Person("lisi",20);

并取出。将姓名和年龄打印出来。

 

--------------------------------------------------------------------------------

 

 

Set:元素不可以重复,是无序。

Set接口中的方法和Collection一致。

|--HashSet: 内部数据结构是哈希表 ,是不同步的。

如何保证该集合的元素唯一性呢?

是通过对象的hashCode和equals方法来完成对象唯一性的。

如果对象的hashCode值不同,那么不用判断equals方法,就直接存储到哈希表中。

如果对象的hashCode值相同,那么要再次判断对象的equals方法是否为true。

如果为true,视为相同元素,不存。如果为false,那么视为不同元素,就进行存储。

 

 

156


 

 

 

记住:如果元素要存储到HashSet集合中,必须覆盖hashCode方法和equals方法。

一般情况下,如果定义的类会产生很多对象,比如人,学生,书,通常都需要覆盖equals,hashCode

方法。

建立对象判断是否相同的依据。

 

 

 

|--TreeSet:可以对Set集合中的元素进行排序。是不同步的。

判断元素唯一性的方式:就是根据比较方法的返回结果是否是0,是0,就是相同元素,不存。

 

 

TreeSet对元素进行排序的方式一:

让元素自身具备比较功能,元就需要实现Comparable接口。覆盖compareTo方法。

 

 

如果不要按照对象中具备的自然顺序进行排序。如果对象中不具备自然顺序。怎么办?

可以使用TreeSet集合第二种排序方式二:

让集合自身具备比较功能,定义一个类实现Comparator接口,覆盖compare方法。

将该类对象作为参数传递给TreeSet集合的构造函数。

 

 

 

if(this.hashCode()== obj.hashCode() && this.equals(obj))

 

 

哈希表确定元素是否相同

1,判断的是两个元素的哈希值是否相同。

如果相同,在判断两个对象的内容是否相同。

 

 

2,判断哈希值相同,其实判断的是对象的hashCode的方法。判断内容相同,用的是equals方法。

 

 

注意:如果哈希值不同,是不需要判断equals。


 

 

118.

泛型简述:


 

package cn.itcast.p4.generic.define.demo;

 

 

/*

public class Tool {

 

 

private Object object;

 

 

public Object getObject() {

return object;

}

 

 

public void setObject(Object object) {

this.object = object;

}

 

 

}

*/

 

157


 

 

 

//jdk1.5后,使用泛型来接收类中要操作的引用数据类型。

//泛型类。什么时候用?当类中的操作的引用数据类型不确定的时候,就使用泛型来表示。

 

public class Tool{

private QQ q;

 

 

public QQ getObject() {

return q;

}

 

 

public void setObject(QQ object) {

this.q = object;

}

 

 

 

/**

* 将泛型定义在方法上。

* @param str

*/

public  void show(W str){

System.out.println("show : "+str.toString());

}

public void print(QQ str){

System.out.println("print : "+str);

}

 

 

/**

* 当方法静态时,不能访问类上定义的泛型。如果静态方法使用泛型,

* 只能将泛型定义在方法上。

* @param obj

*/

public static  void method(Y obj){

System.out.println("method:"+obj);

}

}


 

 

119.

泛型演示:


 

示例1:

package cn.itcast.p5.generic.advance.demo;

 

 

import java.util.ArrayList;

import java.util.Collection;

import java.util.Iterator;

 

 

import cn.itcast.p2.bean.Person;

import cn.itcast.p2.bean.Student;

 

158


 

 

 

import cn.itcast.p2.bean.Worker;

 

 

public class GenericAdvanceDemo2 {

/**

* @param args

*/

public static void main(String[] args) {

 

 

ArrayList al = new ArrayList();

 

 

al.add(new Person("abc",30));

al.add(new Person("abc4",34));

 

 

ArrayList al2 = new ArrayList();

 

 

al2.add(new Student("stu1",11));

al2.add(new Student("stu2",22));

ArrayList al3 = new ArrayList();

 

 

al3.add("stu3331");

al3.add("stu33332");

 

 

printCollection(al2);

printCollection(al);

}

/**

* 迭代并打印集合中元素。

*

* 可以对类型进行限定:

* ? extends E:接收E类型或者E的子类型对象。上限!

*

* ? super E :接收E类型或者E的父类型。下限!

* @param al

*/


/*public static void printCollection(Collection

extends

Person>

al)


{//Collection al = new ArrayList()

Iterator it = al.iterator();

 

 

while(it.hasNext()){


//

//

//

 

 

 

 

 

 

 

 

}

T str = it.next();

System.out.println(str);

System.out.println(it.next().toString());

Person p = it.next();

 

 

System.out.println(p.getName()+":"+p.getAge());

 

 

 

 

 

 

 

 

 

 

 

 

159


 

 

 

}*/

 

 

public static void printCollection(Collection al){

Iterator it = al.iterator();

 

 

while(it.hasNext()){

 

 

System.out.println(it.next());

}

}

 

 

}

示例2(存储用上限):

package cn.itcast.p5.generic.advance.demo;

 

 

import java.util.ArrayList;

 

 

import cn.itcast.p2.bean.Person;

import cn.itcast.p2.bean.Student;

import cn.itcast.p2.bean.Worker;

 

 

public class GenericAdvanceDemo3 {

public static void main(String[] args) {

 

 

ArrayList al1 = new ArrayList();

 

 

al1.add(new Person("abc", 30));

al1.add(new Person("abc4", 34));

 

 

ArrayList al2 = new ArrayList();

 

 

al2.add(new Student("stu1", 11));

al2.add(new Student("stu2", 22));

 

 

ArrayList al3 = new ArrayList();

 

 

al3.add(new Worker("stu1", 11));

al3.add(new Worker("stu2", 22));

 

 

ArrayList al4 = new ArrayList();

al4.add("abcdeef");

// al1.addAll(al4);//错误,类型不匹配。

 

al1.addAll(al2);

al1.addAll(al3);

 

 

System.out.println(al1.size());

 

160


 

 

 

 

// printCollection(al2);

// printCollection(al);

}

 

 

}

/*

* 一般在存储元素的时候都是用上限,因为这样取出都是按照上限类型来运算的,不会出现类型安全隐患。

*/

class MyCollection {

public void add(E e) {

 

 

}

 

 

public void addAll(MyCollection e) {

 

 

}

}

示例3(取出元素用下限):

package cn.itcast.p5.generic.advance.demo;

 

 

import java.util.Comparator;

import java.util.Iterator;

import java.util.TreeSet;

 

 

import cn.itcast.p2.bean.Person;

import cn.itcast.p2.bean.Student;

import cn.itcast.p2.bean.Worker;

 

 

public class GenericAdvanceDemo4 {

 

 

public static void main(String[] args) {

 

 

TreeSet al1 = new TreeSet(new CompByName());

 

 

al1.add(new Person("abc4",34));

al1.add(new Person("abc1",30));

al1.add(new Person("abc2",38));

 

 

TreeSet al2 = new TreeSet(new CompByName());

 

 

al2.add(new Student("stu1",11));

al2.add(new Student("stu7",20));

al2.add(new Student("stu2",22));

 

 

TreeSet al3 = new TreeSet();

 

 

161


 

 

 

al3.add(new Worker("stu1",11));

al3.add(new Worker("stu2",22));

 

 

TreeSet al4 = new TreeSet();

al4.add("abcdeef");


//

 

//

//

 

 

//

al1.addAll(al4);//错误,类型不匹配。

 

al1.addAll(al2);

al1.addAll(al3);

 

 

System.out.println(al1.size());


 

 

Iterator it = al2.iterator();

while(it.hasNext()){

System.out.println(it.next());

}

 

 

}

}

/*

* class TreeSet

* {


*

Tree(Comparator comp);


* }

*

* 什么时候用下限呢?通常对集合中的元素进行取出操作时,可以是用下限。

*

*/

class CompByName implements Comparator{

 

 

@Override

public int compare(Person o1, Person o2) {

 

 

int temp = o1.getName().compareTo(o2.getName());

 

 

return temp==0? o1.getAge()-o2.getAge():temp;

}

 

 

}

 

 

class CompByStuName implements Comparator{

 

 

@Override

public int compare(Student o1, Student o2) {

 

 

int temp = o1.getName().compareTo(o2.getName());

 

 

162


 

 

 

return temp==0? o1.getAge()-o2.getAge():temp;

}

 

 

}

 

 

class CompByWorkerName implements Comparator{

 

 

@Override

public int compare(Worker o1, Worker o2) {

 

 

int temp = o1.getName().compareTo(o2.getName());

 

 

return temp==0? o1.getAge()-o2.getAge():temp;

}

 

 

}


 

 

120.

Map简述:


 

package cn.itcast.p6.map.demo;


 

 

import

import

import

import

import

 

java.util.Collection;

java.util.HashMap;

java.util.Iterator;

java.util.Map;

java.util.Set;


 

 

public class MapDemo {

 

 

public static void main(String[] args) {

 

 

Map map = new HashMap();

method_2(map);

}

 

 

public static void method_2(Map map){

 

 

map.put(8,"zhaoliu");

map.put(2,"zhaoliu");

map.put(2,"zhaoliu");

map.put(6,"wangcai");

 

 

Collection values = map.values();

 

 

Iterator it2 = values.iterator();

while(it2.hasNext()){

System.out.println(it2.next());

 

163


 

 

 

}

 

 

/*

* 通过Map转成set就可以迭代。

* 找到了另一个方法。entrySet。

* 该方法将键和值的映射关系作为对象存储到了Set集合中,而这个映射关系的类型就是Map.Entry

* 类型(结婚证)

*/

Set> entrySet = map.entrySet();

 

 

Iterator> it = entrySet.iterator();

 

 

while(it.hasNext()){

Map.Entry me = it.next();

Integer key = me.getKey();

String value = me.getValue();

System.out.println(key+"::::"+value);

 

 

}

 

 

//取出map中的所有元素。

//原理,通过keySet方法获取map中所有的键所在的Set集合,在通过Set的迭代器获取到每一个键,

//在对每一个键通过map集合的get方法获取其对应的值即可。

/*

Set keySet = map.keySet();

Iterator it = keySet.iterator();

 

 

while(it.hasNext()){

Integer key = it.next();

String value = map.get(key);

System.out.println(key+":"+value);

 

 

}

*/

 

 

}

 

 

public static void method(Map map){//学号和姓名

 

 

 

// 添加元素。

System.out.println(map.put(8, "wangcai"));//null

System.out.println(map.put(8, "xiaoqiang"));//wangcai 存相同键,值会覆盖。

map.put(2,"zhangsan");

map.put(7,"zhaoliu");

 

 

 

 

164


 

 

 

//删除。


//

 

 

 

//

System.out.println("remove:"+map.remove(2));

 

 

//判断。

System.out.println("containskey:"+map.containsKey(7));


 

 

//获取。

System.out.println("get:"+map.get(6));

 

 

System.out.println(map);

 

 

Outer.Inner.show();

}

 

 

}

 

 

interface MyMap{

public static interface MyEntry{//内部接口

void get();

}

}

 

 

class MyDemo implements MyMap.MyEntry{

public void get(){}

}

 

 

class Outer{

static class Inner{

static void show(){}

}

}


 

 

121.

HashMap演示:


 

package cn.itcast.p7.hashmap.demo;

 

 

import java.util.HashMap;

import java.util.Iterator;

import java.util.Set;

 

 

import cn.itcast.p2.bean.Student;

 

 

public class HashMapDemo {

 

 

/**

* @param args

*/

 

165


 

 

 

public static void main(String[] args) {

 

 

/*

* 将学生对象和学生的归属地通过键与值存储到map集合中。

*/

HashMap hm = new HashMap();


hm.put(new

hm.put(new

hm.put(new

hm.put(new

hm.put(new

Student("lisi",38),"北京");

Student("zhaoliu",24),"上海");

Student("xiaoqiang",31),"沈阳");

Student("wangcai",28),"大连");

Student("zhaoliu",24),"铁岭");


 

//

//

Set keySet = hm.keySet();

Iterator it = keySet.iterator();


 

 

Iterator it = hm.keySet().iterator();

 

 

while(it.hasNext()){

Student key = it.next();

String value = hm.get(key);

System.out.println(key.getName()+":"+key.getAge()+"---"+value);

}

 

 

}

 

 

}


 

 

122.

TreeMap演示:


 

package cn.itcast.p8.treemap.demo;

 

 

import java.util.Iterator;

import java.util.Map;

import java.util.TreeMap;

 

 

import cn.itcast.p2.bean.Student;

import cn.itcast.p3.comparator.ComparatorByName;

 

 

public class TreeMapDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {


 

 

TreeMap

 

tm

 

=

 

new

 

TreeMap(new


ComparatorByName());

 

166


 

 

 

 

 

tm.put(new

tm.put(new

tm.put(new

tm.put(new

tm.put(new

 

 

 

 

 

Student("lisi",38),"北京");

Student("zhaoliu",24),"上海");

Student("xiaoqiang",31),"沈阳");

Student("wangcai",28),"大连");

Student("zhaoliu",24),"铁岭");


 

Iterator> it = tm.entrySet().iterator();

 

 

while(it.hasNext()){

Map.Entry me = it.next();

Student key = me.getKey();

String value = me.getValue();

 

 

System.out.println(key.getName()+":"+key.getAge()+"---"+value);

}

 

 

}

 

 

}


 

 

123.

Map集合不Collection集合的区别:


 

1,

Map中一次存储是键值对。

Collection中一次存储是单个元素。

2,

Map的存储使用的put方法。

Collection存储使用的是add方法。

3,

Map的取出,是讲Map转成Set,在使用迭代器取出。

Collection取出,使用就是迭代器。

4,

如果对象很多,必须使用容器存储。

如果元素存在着映射关系,可以优先考虑使用Map存储或者用数组,

如果没有映射关系,可以使用Collection存储。


 

 

124.

LinkedHashMap演示:


 

package cn.itcast.p1.map.demo;


 

 

import

import

import

import

import

 

java.io.File;

java.util.HashMap;

java.util.Iterator;

java.util.LinkedHashMap;

java.util.Map;

 

 

 

 

 

 

 

 

 

 

 

167


 

 

 

public class LinkedHashMapDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

File f= null;

HashMap hm = new LinkedHashMap();


 

 

hm.put(7,

hm.put(3,

hm.put(1,

hm.put(5,

 

"zhouqi");

"zhangsan");

"qianyi");

"wangwu");


 

 

Iterator> it = hm.entrySet().iterator();

 

 

while(it.hasNext()){

Map.Entry me = it.next();

 

 

Integer key = me.getKey();

String value = me.getValue();

 

 

System.out.println(key+":"+value);

}

}

 

 

}

 

125.Map精华习题:

 

习题1:

package cn.itcast.p1.map.test;

 

 

import java.util.Iterator;

import java.util.Map;

import java.util.TreeMap;

 

 

/*


*

*

*

*

*

*

*

*

练习:

"fdgavcbsacdfs" 获取该字符串中,每一个字母出现的次数。

要求打印结果是:a(2)b(1)...;

思路:

对于结果的分析发现,字母和次数之间存在着映射的关系。而且这种关系很多。

很多就需要存储,能存储映射关系的容器有数组和Map集合。

关系一方式有序编号吗?没有!

那就是使用Map集合。 又发现可以保证唯一性的一方具备着顺序如 a b c ...

 

 

 

 

 

 

 

 

 

 

 

 

168


 

 

 

* 所以可以使用TreeMap集合。

*

* 这个集合最终应该存储的是字母和次数的对应关系。

*

* 1,因为操作的是字符串中的字母,所以先将字符串变成字符数组。

* 2,遍历字符数组,用每一个字母作为键去查Map集合这个表。

* 如果该字母键不存在,就将该字母作为键 1作为值存储到map集合中。

* 如果该字母键存在,就将该字母键对应值取出并+1,在将该字母和+1后的值存储到map集合中,

* 键相同值会覆盖。这样就记录住了该字母的次数.

* 3,遍历结束,map集合就记录所有字母的出现的次数。oy.

*/

 

 

public class MapTest {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

String str = "fdg+avAdc bs5dDa9c-dfs";

 

 

String s = getCharCount(str);

 

 

System.out.println(s);

 

 

}

 

 

public static String getCharCount(String str) {

 

 

//将字符串变成字符数组

char[] chs = str.toCharArray();

 

 

//定义map集合表。

Map map = new TreeMap();

 

 

for (int i = 0; i < chs.length; i++) {

 

 

if(!(chs[i]>='a' && chs[i]<='z' || chs[i]>='A' && chs[i]<='Z'))


//

if(!(Character.toLowerCase(chs[i])>='a'

&&


Character.toLowerCase(chs[i])<='z'))

continue;

 

 

//将数组中的字母作为键去查map表。

Integer value = map.get(chs[i]);

 

 

int count = 1;

 

 

169


 

 

 

//判断值是否为null.

if(value!=null){

count = value+1;

}


//

count++;


map.put(chs[i], count);

/*

if(value==null){

map.put(chs[i], 1);

}else{

map.put(chs[i], value+1);

}

*/

}

return mapToString(map);

}

 

 

private static String mapToString(Map map) {

 

 

StringBuilder sb = new StringBuilder();

 

 

Iterator it = map.keySet().iterator();

 

 

while(it.hasNext()){

Character key = it.next();

Integer value = map.get(key);

 

 

sb.append(key+"("+value+")");

}

 

 

return sb.toString();

}

 

 

}

习题2:

package cn.itcast.p1.map.test;

 

 

import java.util.HashMap;

import java.util.Map;

 

 

public class MapTest2 {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

170


 

 

 

/*

* Map在有映射关系时,可以优先考虑。

*

* 在查表法中的应用较为多见。

*/

 

 

String week = getWeek(1);

System.out.println(week);

 

 

System.out.println(getWeekByMap(week));

}

public static String getWeekByMap(String week){

 

 

Map map = new HashMap();

 

 

map.put("星期一","Mon");

map.put("星期二","Tus");

map.put("星期三","Wes");

map.put("星期日","Sun");

map.put("星期天","Sun");

 

return map.get(week);

}

 

 

public static String getWeek(int week){

 

 

if(week<1 || week>7)

throw new RuntimeException("没有对应的星期,请您重新输入");

 

 

String[] weeks = {"","星期一","星期二"};

 

return weeks[week];

}

}


 

 

126.

Collections集合框架工具类简述及使用:


 

package cn.itcast.p2.toolclass.collections.demo;


 

 

import

import

import

import

import

 

java.util.ArrayList;

java.util.Collections;

java.util.Comparator;

java.util.List;

java.util.TreeSet;


 

 

import cn.itcast.p2.comparator.ComparatorByLength;

 

 

171


 

 

 

public class CollectionsDemo {

 

 

public static void main(String[] args) {

/*

* Collections:是集合框架的工具类。

* 里面的方法都是静态的。

*/

 

 

demo_4();

}

 

 

public static void demo_4() {

List list = new ArrayList();

 

 

list.add("abcde");

list.add("cba");

list.add("zhangsan");

list.add("zhaoliu");

list.add("xiaoqiang");

 

 

System.out.println(list);


//

 

 

//

Collections.replaceAll(list, "cba", "nba"); // set(indexOf("cba"),"nba");

Collections.shuffle(list);

Collections.fill(list, "cc");

System.out.println(list);

}

 

 

public static void demo_3() {

/*

TreeSet ts = new TreeSet(new Comparator(){

 

 

@Override

public int compare(String o1, String o2) {

 

 

int temp = o2.compareTo(o1);

return temp;

}

 

 

});

*/


TreeSet ts

=

new

TreeSet(Collections.reverseOrder(new


ComparatorByLength()));

 

 

ts.add("abc");

ts.add("hahaha");

ts.add("zzz");

ts.add("aa");

 

172


 

 

 

ts.add("cba");

 

 

System.out.println(ts);

 

 

}

 

 

public static void demo_2(){

List list = new ArrayList();

 

 

list.add("abcde");

list.add("cba");

list.add("aa");

list.add("zzz");

list.add("cba");

list.add("nbaa");


//

 

 

 

//

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

//

//

Collections.sort(list);

System.out.println(list);

 

 

int index = Collections.binarySearch(list, "cba");

System.out.println("index="+index);

 

 

//获取最大值。

String max = Collections.max(list,new ComparatorByLength());

System.out.println("max="+max);

}

 

 

public static void demo_1(){

 

 

List list = new ArrayList();

 

 

list.add("abcde");

list.add("cba");

list.add("aa");

list.add("zzz");

list.add("cba");

list.add("nbaa");

System.out.println(list);

 

 

//对list集合进行指定顺序的排序。

Collections.sort(list);

mySort(list);

mySort(list,new ComparatorByLength());

Collections.sort(list,new ComparatorByLength());

System.out.println(list);

}

public static  void mySort(List list,Comparator comp){

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

173


 

 

 

for (int i = 0; i < list.size()-1; i++) {

 

 

for (int j = i+1; j < list.size(); j++) {

 

 

if(comp.compare(list.get(i), list.get(j))>0){


 

 

//

//

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

//

//

 

T temp = list.get(i);

list.set(i, list.get(j));

list.set(j, temp);

Collections.swap(list, i, j);

}

}

}

}

 

 

public static > void mySort(List list){

 

 

for (int i = 0; i < list.size()-1; i++) {

 

 

for (int j = i+1; j < list.size(); j++) {

 

 

if(list.get(i).compareTo(list.get(j))>0){

 

 

T temp = list.get(i);

list.set(i, list.get(j));

list.set(j, temp);


Collections.swap(list, i, j);

}

}

}

}

}


 

 

127.

Arrays数组工具类简述及使用:


 

package cn.itcast.p3.toolclass.arrays.demo;

 

 

import java.util.Arrays;

import java.util.List;

 

 

//数组转成集合。

public class ArraysDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

174


 

 

 

/*

* Arrays:集合框架的工具类。里面的方法都是静态的。

*/


 

 

//

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

int[] arr = {3,1,5,6,3,6};

System.out.println(Arrays.toString(arr));

 

 

demo_2();

}

 

 

public static void demo_2() {

 

 

/*

* 如果数组中的元素是对象,那么转成集合时,直接将数组中的元素作为集合中的元素进行集合存储。

*

* 如果数组中的元素是基本类型数值,那么会将该数组作为集合中的元素进行存储。

*/

int[] arr = {31,11,51,61};

 

 

List list = Arrays.asList(arr);

 

 

System.out.println(list);

}

 

 

public static void demo_1() {

/*

* 重点:List asList(数组)将数组转成集合。

*

* 好处:其实可以使用集合的方法操作数组中的元素。

* 注意:数组的长度是固定的,所以对于集合的增删方法是不可以使用的

* 否则会发生UnsupportedOperationException

*/

String[] arr = {"abc","haha","xixi"};

 

 

boolean b = myContains(arr, "xixi");

System.out.println("contains:"+b);

 

 

List list = Arrays.asList(arr);

boolean b1 = list.contains("xixi");

System.out.println("list contaisn:="+b1);

list.add("hiahia");//UnsupportedOperationException

 

 

System.out.println(list);

}

 

 

public static boolean myContains(String[] arr,String key){

for (int i = 0; i < arr.length; i++) {

 

175


 

 

 

if(arr[i].equals(key))

return true;

}

return false;

}

 

 

//toString的经典实现。

public static String myToString(int[] a){

int iMax = a.length - 1;

if (iMax == -1)

return "[]";

 

 

StringBuilder b = new StringBuilder();

b.append('[');

for (int i = 0; ; i++) {//中间省略条件判断,提高了效率。

b.append(a[i]);

if (i == iMax)

return b.append(']').toString();

b.append(", ");

}

}

}


 

 

128.

集合转数组的toArray()方法使用:


 

package cn.itcast.p3.toolclass.arrays.demo;

 

 

import java.util.ArrayList;

import java.util.Arrays;

import java.util.List;

 

 

public class ToArray {

public static void main(String[] args) {

 

 

/*

* 集合转成数组呢?

*

* 使用的就是Collection接口中的toArray方法。

*

* 集合转成数组:可以对集合中的元素操作的方法进行限定,但不允许对其进行增删。

*/

 

 

List list = new ArrayList();

list.add("abc1");

list.add("abc2");

list.add("abc3");

 

 

176


 

 

 

/*

* toArray方法需要传入一个指定类型的数组。

* 长度该如何定义呢?

* 如果长度小于集合的size,那么该方法会创建一个同类型并和集合相同size的数组。

* 如果长度大于集合的size,那么该方法就会使用指定的数组,存储集合中的元素,其他位置默认为

* null。

* 所以建议,最后长度就指定为,集合的size。

*/

String[] arr = list.toArray(new String[list.size()]);

 

 

System.out.println(Arrays.toString(arr));

 

 

}

}


 

 

129.

增强for循环使用:


 

package cn.itcast.p4.news.demo;


 

 

import

import

import

import

 

java.util.ArrayList;

java.util.HashMap;

java.util.List;

java.util.Map;


 

 

public class ForEachDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

/*


*

*

foreach语句:

格式:


*

*

*

*

for(类型 变量

{

 

 

}

:Collection集合|数组)


*

* 传统for和高级for的区别?

* 传统for可以完成对语句执行很多次,因为可以定义控制循环的增量和条件。

*

* 高级for是一种简化形式。

* 它必须有被遍历的目标。该目标要是数组,要么是Collection单列集合。

*

* 对数数组的遍历如果仅仅是获取数组中的元素,可以使用高级for。

* 如果要对数组的角标进行操作建议使用传统for。

 

177


 

 

 

*

*/

 

 

List list =new ArrayList();

 

 

list.add("abc1");

list.add("abc2");

list.add("abc3");


 

 

for(String s : list){

 

//简化书写。


System.out.println(s);

}

 

 

int[] arr = {3,1,5,7,4};

 

 

for(int i : arr){

System.out.println(i);

}

 

 

//可以使用高级for遍历map集合吗?不能直接用,但是可以将map转成单列的set,就可以用了。

Map map = new HashMap();

 

 

map.put(3,"zhagsan");

map.put(1,"wangyi");

map.put(7,"wagnwu");

map.put(4,"zhagsansan");

 

 

for(Integer key : map.keySet()){

String value = map.get(key);

System.out.println(key+"::"+value);

}

 

 

for(Map.Entry me : map.entrySet()){

Integer key = me.getKey();

String value = me.getValue();

 

 

System.out.println(key+":"+value);

}


 

 

//

//

//

//

 

Iterator it = list.iterator();

while(it.hasNext()){

System.out.println(it.next());

}


}

 

 

}

 

 

178


 

 

 

 

 

130.

 

 

 

 

函数的可发参数:


 

package cn.itcast.p4.news.demo;

 

 

public class ParamterDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {


 

 

//

//

//

//

 

 

 

//

//

//

//

//

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

//

 

int sum = add(4,5);

System.out.println("sum="+sum);

int sum1 = add(4,5,6);

System.out.println("sum1="+sum1);

 

 

 

int[] arr = {5,1,4,7,3};

int sum = add(arr);

System.out.println("sum="+sum);

int[] arr1 = {5,1,4,7,3,9,8,7,6};

int sum1 = add(arr1);

System.out.println("sum1="+sum1);

 

 

int sum = newAdd(5,1,4,7,3);

System.out.println("sum="+sum);

int sum1 = newAdd(5,1,2,7,3,9,8,7,6);

System.out.println("sum1="+sum1);

 

 

}

/*

* 函数的可变参数。

* 其实就是一个数组,但是接收的是数组的元素。

* 自动将这些元素封装成数组。简化了调用者的书写。

* 注意:可变参数类型,必须定义在参数列表的结尾。

*/

public static int newAdd(int a,int... arr){

 

 

int sum = 0;

for (int i = 0; i < arr.length; i++) {

sum+=arr[i];

}

return sum;

 

 

System.out.println(arr);

return 0;

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

179


 

 

 

}

 

 

public static int add(int[] arr){

 

 

int sum = 0;

for (int i = 0; i < arr.length; i++) {

sum+=arr[i];

}

return sum;

 

 

}

 

 

public static int add(int a,int b){

return a+b;

}

 

 

public static int add(int a,int b,int c){

return a+b+c;

}

}


 

 

131.

静态导入,其实导入的是类中的静态成员(演示):


 

package cn.itcast.p4.news.demo;

 

 

import java.util.ArrayList;

import java.util.List;

import static java.util.Collections.*;//静态导入,其实到入的是类中的静态成员。

//import static java.util.Collections.max;//静态导入,其实到入的是类中的静态成员。

 

import static java.lang.System.*;

 

 

public class StaticImportDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

List list = new ArrayList();

 

 

list.add("abc3");

list.add("abc7");

list.add("abc1");

 

 

out.println(list);

 

 

180


 

 

 

sort(list);

 

 

System.out.println(list);

String max = max(list);

System.out.println("max="+max);

}

}


 

 

132.

集合(ListSetMap)大总结:


 

集合类的由来:

对象用于封装特有数据,对象多了需要存储,如果对象的个数不确定。

就使用集合容器进行存储。

 

 

集合特点:

1,用于存储对象的容器。

2,集合的长度是可变的。

3,集合中不可以存储基本数据类型值。

 

 

集合容器因为内部的数据结构不同,有多种具体容器。

不断的向上抽取,就形成了集合框架。

 

 

框架的顶层Collection接口:

 

 

Collection的常见方法:

 

 

1,添加。

boolean add(Object obj):

boolean addAll(Collection coll):

 

 

 

2,删除。

boolean remove(object obj):

boolean removeAll(Collection coll);

void clear();

 

 

3,判断:

boolean contains(object obj):

boolean containsAll(Colllection coll);

boolean isEmpty():判断集合中是否有元素。

 

 

4,获取:

int size():

Iterator iterator():取出元素的方式:迭代器。

该对象必须依赖于具体容器,因为每一个容器的数据结构都不同。

所以该迭代器对象是在容器中进行内部实现的。

对于使用容器者而言,具体的实现不重要,只要通过容器获取到该实现的迭代器的对象即可,

 

181


 

 

 

也就是iterator方法。

 

 

Iterator接口就是对所有的Collection容器进行元素取出的公共接口。

其实就是抓娃娃游戏机中的夹子!

 

 

5,其他:

boolean retainAll(Collection coll);取交集。

Object[] toArray():将集合转成数组。

 

--------------------------------------------------------------------------------

Collection

|--List:有序(存入和取出的顺序一致),元素都有索引(角标),元素可以重复。

|--Set:元素不能重复,无序。

 

 

List:特有的常见方法:有一个共性特点就是都可以操作角标。

 

 

1,添加

void add(index,element);

void add(index,collection);

 

 

 

2,删除;

Object remove(index):

 

 

 

3,修改:

Object set(index,element);

 

 

 

4,获取:

Object get(index);

int indexOf(object);

int lastIndexOf(object);

List subList(from,to);

 

 

list集合是可以完成对元素的增删改查。

 

List:

|--Vector:内部是数组数据结构,是同步的。增删,查询都很慢!

|--ArrayList:内部是数组数据结构,是不同步的。替代了Vector。查询的速度快。

|--LinkedList:内部是链表数据结构,是不同步的。增删元素的速度很快。

 

LinkedList:

 

 

addFirst();

addLast():

jdk1.6

 

182


 

 

 

offerFirst();

offetLast();

 

 

getFirst();.//获取但不移除,如果链表为空,抛出NoSuchElementException.

getLast();

jdk1.6

peekFirst();//获取但不移除,如果链表为空,返回null.

peekLast():

 

 

removeFirst();//获取并移除,如果链表为空,抛出NoSuchElementException.

removeLast();

jdk1.6

pollFirst();//获取并移除,如果链表为空,返回null.

pollLast();

 

 

作业:

1,自己去查文档演示Vector中的elements()方法。

 

 

2,LinkedList中的,addFirst addLast getFirst,getLast removeFirst removeLast。

 

 

3,既然集合是存储对象的,请定义ArryaList集合,并存储Person对象。如new Person("lisi",20);

并取出。将姓名和年龄打印出来。

 

--------------------------------------------------------------------------------

 

 

Set:元素不可以重复,是无序。

Set接口中的方法和Collection一致。

|--HashSet: 内部数据结构是哈希表 ,是不同步的。

如何保证该集合的元素唯一性呢?

是通过对象的hashCode和equals方法来完成对象唯一性的。

如果对象的hashCode值不同,那么不用判断equals方法,就直接存储到哈希表中。

如果对象的hashCode值相同,那么要再次判断对象的equals方法是否为true。

如果为true,视为相同元素,不存。如果为false,那么视为不同元素,就进行存储。

 

 

记住:如果元素要存储到HashSet集合中,必须覆盖hashCode方法和equals方法。

一般情况下,如果定义的类会产生很多对象,比如人,学生,书,通常都需要覆盖equals,hashCode

方法。

建立对象判断是否相同的依据。

 

 

|--TreeSet:可以对Set集合中的元素进行排序。是不同步的。

判断元素唯一性的方式:就是根据比较方法的返回结果是否是0,是0,就是相同元素,不存。

 

 

TreeSet对元素进行排序的方式一:

让元素自身具备比较功能,元就需要实现Comparable接口。覆盖compareTo方法。

 

 

如果不要按照对象中具备的自然顺序进行排序。如果对象中不具备自然顺序。怎么办?

可以使用TreeSet集合第二种排序方式二:

 

183


 

 

 

让集合自身具备比较功能,定义一个类实现Comparator接口,覆盖compare方法。

将该类对象作为参数传递给TreeSet集合的构造函数。

 

if(this.hashCode()== obj.hashCode() && this.equals(obj))

 

 

 

哈希表确定元素是否相同

1,判断的是两个元素的哈希值是否相同。

如果相同,在判断两个对象的内容是否相同。

 

 

2,判断哈希值相同,其实判断的是对象的hashCode的方法。判断内容相同,用的是equals方法。

 

 

注意:如果哈希值不同,是不需要判断equals。

 

--------------------------------------------------------------------------------

 

 

泛型:

jdk1.5出现的安全机制。

 

 

好处:

1,将运行时期的问题ClassCastException转到了编译时期。

2,避免了强制转换的麻烦。

 

 

<>:什么时候用?当操作的引用数据类型不确定的时候。就使用<>。将要操作的引用数据类型传入即可.

其实<>就是一个用于接收具体引用数据类型的参数范围。

 

 

在程序中,只要用到了带有<>的类或者接口,就要明确传入的具体引用数据类型 。

 

 

泛型技术是给编译器使用的技术,用于编译时期。确保了类型的安全。

 

 

运行时,会将泛型去掉,生成的class文件中是不带泛型的,这个称为泛型的擦除。

为什么擦除呢?因为为了兼容运行的类加载器。

 

 

泛型的补偿:在运行时,通过获取元素的类型进行转换动作。不用使用者在强制转换了。

 

 

泛型的通配符:? 未知类型。

 

 

泛型的限定:

? extends E: 接收E类型或者E的子类型对象。上限

一般存储对象的时候用。比如 添加元素 addAll.

 

 

? super E: 接收E类型或者E的父类型对象。 下限。

一般取出对象的时候用。比如比较器。

 

--------------------------------------------------------------------------------

 

 

集合的一些技巧:

 

184


 

 

 

 

 

需要唯一吗?

需要:Set

需要制定顺序:

需要: TreeSet

不需要:HashSet

但是想要一个和存储一致的顺序(有序):LinkedHashSet

不需要:List

需要频繁增删吗?

需要:LinkedList

不需要:ArrayList

 

 

如何记录每一个容器的结构和所属体系呢?

 

 

看名字!

 

List

|--ArrayList

|--LinkedList

 

 

Set

|--HashSet

|--TreeSet

 

 

后缀名就是该集合所属的体系。

 

 

前缀名就是该集合的数据结构。

 

 

看到array:就要想到数组,就要想到查询快,有角标.

看到link:就要想到链表,就要想到增删快,就要想要 add get remove+frist last的方法

看到hash:就要想到哈希表,就要想到唯一性,就要想到元素需要覆盖hashcode方法和equals方法。

看到tree:就要想到二叉树,就要想要排序,就要想到两个接口Comparable,Comparator 。

而且通常这些常用的集合容器都是不同步的。

 

--------------------------------------------------------------------------------

 

 

Map:一次添加一对元素。Collection 一次添加一个元素。

Map也称为双列集合,Collection集合称为单列集合。

其实map集合中存储的就是键值对。

map集合中必须保证键的唯一性。

 

 

 

常用方法:

1,添加。

value put(key,value):返回前一个和key关联的值,如果没有返回null.

 

 

2,删除。

 

185


 

 

 

void clear():清空map集合。

value remove(key):根据指定的key翻出这个键值对。

 

 

3,判断。

boolean containsKey(key):

boolean containsValue(value):

boolean isEmpty();

 

 

4,获取。

value get(key):通过键获取值,如果没有该键返回null。

当然可以通过返回null,来判断是否包含指定键。

int size(): 获取键值对的个数。

 

 

 

Map常用的子类:

|--Hashtable :内部结构是哈希表,是同步的。不允许null作为键,null作为值。

|--Properties:用来存储键值对型的配置文件的信息,可以和IO技术相结合。

|--HashMap : 内部结构是哈希表,不是同步的。允许null作为键,null作为值。

|--TreeMap : 内部结构是二叉树,不是同步的。可以对Map集合中的键进行排序。

 

 

 

List list = new ArrayList();//非同步的。

 

 

list = MyCollections.synList(list);//返回一个同步的list.

 

 

 

给非同步的集合加锁:

 

class MyCollections{

 

 

public static List synList(List list){

 

 

return new MyList(list);

}

 

 

private class MyList implements List{

private List list;

 

 

private static final Object lock = new Object();

MyList(List list){

this.list = list;

}

 

 

public boolean add(Object obj){

synchronized(lock)

{

return list.add(obj);

 

186


 

 

 

}

}

 

 

public boolean remove(Object obj){

synchronized(lock)

{

return list.remove(obj);

}

}

 

 

}

}


 

 

133.

集合框架查补乊遍历集合:


 

//遍历List方法1,使用普通for循环:

for (int i = 0; i < list.size(); i++) {

String temp=(String)list.get(i);

System.out.println(temp);

//list.remove(i);//遍历删除元素,不过不推荐这种方式!

}

//遍历List方法2,使用增强for循环(应该使用泛型定义类型!):

for(String temp:list){

System.out.println(temp);

}

//遍历List方法3,使用Iterator迭代器:

for(Iterator iter=list.iterator();iter.hasNext();){

String temp=(String)iter.next();

System.out.println(temp);

}

或者:

Iterator iter=c.iterator();

while(iter.hasNext()){

Object obj=iter.next();

iter.remove();//如果要遍历删除集合中的元素,建议使用这种方式!

System.out.println(obj);

}

//遍历Set方法1,使用增强for循环:

for(String temp:set){

System.out.println(temp);

}

//遍历Set方法2,使用Iterator迭代器:

for(Iterator iter=list.iterator();iter.hasNext();){

String temp=(String)iter.next();

System.out.println(temp);

}

//遍历Map

187


 

 

 

Map maps=new HashMap();

Set keySet=maps.keySet();

for(Integer id:keySet){

System.out.println(maps.get(id).name);

}


 

 

134.

其他类对象-日期类对象的使用:


 

示例1:

package cn.itcast.p1.otherapi;

 

 

import java.util.Calendar;

 

 

public class CalendarDemo {

 

 

public static void main(String[] args) {

 

 

Calendar c = Calendar.getInstance();

 

 

int year = 2012;

showDays(year);

}

 

 

public static void showDays(int year) {

 

 

Calendar c = Calendar.getInstance();

c.set(year, 2, 1);

 

 

c.add(Calendar.DAY_OF_MONTH, -1);

 

 

showDate(c);

}


 

 

public

int

int

int

int

 

static void showDate(Calendar c) {

year = c.get(Calendar.YEAR);

month = c.get(Calendar.MONTH)+1;

day = c.get(Calendar.DAY_OF_MONTH);

week = c.get(Calendar.DAY_OF_WEEK);


 

 

System.out.println(year+""+month+""+day+""+getWeek(week));

}

 

 

public static String getWeek(int i) {

 

 

String[] weeks = { "","星期日","星期一","星期二","星期三","星期四","星期五","星期六

"};

 

 

188


 

 

 

return weeks[i];

}

}

示例2:

package cn.itcast.p1.otherapi;


 

 

import

import

import

import

 

java.text.DateFormat;

java.text.ParseException;

java.text.SimpleDateFormat;

java.util.Date;


 

 

public class DateDemo {

 

 

/**

* @param args

* @throws ParseException

*/

public static void main(String[] args) throws ParseException {

 

 

methodDemo_3();

 

 

}

 

 

/**

* 将日期格式的字符串-->日期对象。

* 使用的是DateFormat类中的parse()方法。

*

* @throws ParseException

*/

public static void methodDemo_3() throws ParseException {

 

 

String str_date = "2012年4月19日";

str_date = "2011---8---17";

 

 

DateFormat dateFormat = DateFormat.getDateInstance(DateFormat.LONG);

 

 

dateFormat = new SimpleDateFormat("yyyy---MM---dd");

 

 

 

Date date = dateFormat.parse(str_date);

 

 

System.out.println(date);

 

 

}

 

 

/**

* 对日期对象进行格式化。

 

189


 

 

 

* 将日期对象-->日期格式的字符串。

* 使用的是DateFormat类中的format方法。

*

*/

public static void methodDemo_2() {

 

 

Date date = new Date();

 

 

//获取日期格式对象。具体着默认的风格。 FULL LONG等可以指定风格。

DateFormat dateFormat = DateFormat.getDateInstance(DateFormat.LONG);


dateFormat

DateFormat.getDateTimeInstance(DateFormat.LONG,DateFormat.LONG);

=


//

System.out.println(dateFormat);


 

 

//如果风格是自定义的如何解决呢?

dateFormat = new SimpleDateFormat("yyyy--MM--dd");

 

 

String str_date = dateFormat.format(date);

 

 

System.out.println(str_date);

}

 

 

/**

* 日期对象和毫秒值之间的转换。

*


*

*

*

*

毫秒值-->日期对象 :

1,通过Date对象的构造方法 new Date(timeMillis);

2,还可以通过setTime设置。

因为可以通过Date对象的方法对该日期中的各个字段(年月日等)进行操作。


*

* 日期对象-->毫秒值:

* 2,getTime方法。

* 因为可以通过具体的数值进行运算。

*/

public static void methodDemo_1() {

long time = System.currentTimeMillis();//


//

System.out.println(time);//1335671230671


 

 

Date date = new Date();//将当前日期和时间封装成Date对象。

System.out.println(date);//Sun Apr 29 11:48:02 CST 2012

 

 

Date date2 = new Date(1335664696656l);//将指定毫秒值封装成Date对象。

System.out.println(date2);

}

}

 

 

 

 

190


 

 

 

 

 

135.

 

 

 

 

其他类对象-Math类对象的使用:


 

示例1:

package cn.itcast.p1.otherapi;

 

 

public class Mat {

 

 

public static void main(String[] args) {


 

 

//

//

//

//

//

//

 

for (int i = 0; i < 10; i++) {

int d = (int) (Math.random() * 10+1);

int d=(int) Math.ceil(12.23);

double d = Math.ceil(12.31);

Double d=Math.floor(12.36);

long d=Math.round(63.50);


double d=Math.pow(2, 3);

System.out.println(d);

}

}

示例2:

package cn.itcast.p1.otherapi;

 

 

import java.util.Random;

 

 

public class MathDemo {

 

 

public static void main(String[] args) {

 

 

/*

* Math:提供了操作数学运算的方法,都是静态的。

* 常用的方法:

* ceil():返回大于参数的最小整数。

* floor():返回小于参数的最大整数。

* round():返回四舍五入的整数。

* pow(a,b):ab次方。

*/

double d1 = Math.ceil(12.56);

double d2 = Math.floor(12.56);

double d3 = Math.round(12.46);


 

 

//

//

//

 

 

//

//

 

sop("d1="+d1);

sop("d2="+d2);

sop("d3="+d3);

 

 

double d = Math.pow(10, 2);

sop("d="+d);

 

 

 

 

 

 

 

 

 

 

 

191


 

 

 

 

Random r = new Random();

for (int i = 0; i < 10; i++) {


 

 

//

//

//

 

double d = Math.ceil(Math.random()*10);

double d = (int)(Math.random()*6+1);

double d = (int)(r.nextDouble()*6+1);


int d = r.nextInt(6)+1;

System.out.println(d);

}

}

 

 

public static void sop(String string) {

System.out.println(string);

}

 

 

}


 

 

136.

其他类对象-Runtime类对象的使用:


 

package cn.itcast.p1.otherapi;

 

 

import java.io.IOException;

 

 

public class RuntimeDemo {

 

 

/**

* @param args

* @throws IOException

* @throws InterruptedException

*/

public static void main(String[] args) throws IOException, InterruptedException

{

 

 

/*

* Runtime:没有构造方法摘要,说明该类不可以创建对象。

* 又发现还有非静态的方法。说明该类应该提供静态的返回该类对象的方法。

* 而且只有一个,说明Runtime类使用了单例设计模式。

*

*/

 

 

Runtime r = Runtime.getRuntime();


 

 

//

 

execute: 执行。 xxx.exe

 

Process p = r.exec("notepad.exe");

Thread.sleep(5000);

 

 

 

 

 

 

 

192


 

 

 

p.destroy();

}

 

 

}


 

 

137.

其他类对象-System类对象的使用:


 

package cn.itcast.p1.otherapi;


 

 

import

import

import

import

 

java.util.Arrays;

java.util.List;

java.util.Properties;

java.util.Set;


 

 

public class SystemDemo {


 

 

private   static   final   String

System.getProperty("line.separator");

/**

* @param args

*/

public static void main(String[] args) {

 

 

/*

* System:类中的方法和属性都是静态的。

* 常见方法:

* long currentTimeMillis();获取当前时间的毫秒值。

*/

 

LINE_SEPARATOR

 

=


 

 

//

//

//

//

//

 

 

 

//

 

 

//

 

long l1 = 1335664696656l;//System.currentTimeMillis();

System.out.println(l1/1000/60/60/24);//1335664696656

code..

long l2 = System.currentTimeMillis();

System.out.println(l2-l1);

 

 

System.out.println("hello-"+LINE_SEPARATOR+" world");

demo_1();

//给系统设置一些属性信息。这些信息是全局,其他程序都可以使用。

System.setProperty("myclasspath", "c:\myclass");

}

 

 

public static void demo_1(){

//获取系统的属性信息,并存储到了Properties集合中。

/*

* properties集合中存储都是String类型的键和值。

* 最好使用它自己的存储和取出的方法来完成元素的操作。

*/

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

193


 

 

 

Properties prop = System.getProperties();

 

 

Set nameSet = prop.stringPropertyNames();

 

 

for(String name : nameSet){

String value = prop.getProperty(name);

 

 

System.out.println(name+"::"+value);

}

}

}


 

 

138.

时间对象Date练习:


 

package cn.itcast.p1.otherapi.test;


 

 

import

import

import

import

 

java.text.DateFormat;

java.text.ParseException;

java.text.SimpleDateFormat;

java.util.Date;


 

 

 

/*

*

*

*

*

*

*

*

*

 

 

练习:

"2012-3-17"到"2012-4-6"

中间有多少天?

思路:

两个日期相减就哦了。

咋减呢?

必须要有两个可以进行减法运算的数。

能减可以是毫秒值。如何获取毫秒值?通过date对象。

如何获取date对象呢?可以将字符串转成date对象。


*

* 1,将日期格式的字符串转成Date对象。

* 2,将Date对象转成毫秒值。

* 3,相减,在变成天数

*

*/

public class DateTest {

 

 

/**

* @param args

* @throws ParseException

*/

public static void main(String[] args) throws ParseException {

 

 

String str_date1 = "2012-3-17";

String str_date2 = "2012-4-18";

 

194


 

 

 

test(str_date1,str_date2);

}

 

 

public static void test(String str_date1,String str_date2) throws ParseException

{

 

 

//1,将日期字符串转成日期对象。

//定义日期格式对象。

DateFormat dateFormat = DateFormat.getDateInstance();

dateFormat = new SimpleDateFormat("yyyy-MM-dd");

 

 

Date date1 = dateFormat.parse(str_date1);

Date date2 = dateFormat.parse(str_date2);

 

 

long time1 = date1.getTime();

long time2 = date2.getTime();

 

 

long time = Math.abs(time1-time2);

 

 

int day = getDay(time);

System.out.println(day);

}

 

 

private static int getDay(long time) {

 

 

int day = (int)(time/1000/60/60/24);

return day;

}

}

 

第八章:IO


 

 

139.

流的简述及演示案例:


 

输入流和输出流相对于内存设备而言.

 

 

将外设中的数据读取到内存中:输入

将内存的数写入到外设中:输出。

 

 

字符流的由来:

其实就是:字节流读取文字字节数据后,不直接操作而是先查指定的编码表。获取对应的文字。

在对这个文字进行操作。简单说:字节流+编码表

-------------------------------------------------------------------------------

字节流的两个顶层父类:

1,InputStream 2,OutputStream.

 

 

195


 

 

 

字符流的两个顶层父类:

1,Reader 2,Writer

 

 

这些体系的子类都以父类名作为后缀。

而且子类名的前缀就是该对象的功能。

 

 

FileWriter演示:

package cn.itcast.p2.io.filewriter;

 

 

import java.io.FileWriter;

import java.io.IOException;

 

 

//需求:将一些文字存储到硬盘一个文件中。

 

public class FileWriterDemo {


 

 

private   static   final

 

String

 

LINE_SEPARATOR

 

=


System.getProperty("line.separator");

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

//创建一个可以往文件中写入字符数据的字符输出流对象。

/*

* 既然是往一个文件中写入文字数据,那么在创建对象时,就必须明确该文件(用于存储数据的目的

* 地)。

*

* 如果文件不存在,则会自动创建。

* 如果文件存在,则会被覆盖。

*

* 如果构造函数中加入true,可以实现对文件进行续写!

*/

FileWriter fw = new FileWriter("demo.txt",true);

/*

* 调用Writer对象中的write(string)方法,写入数据。

*

* 其实数据写入到临时存储缓冲区中。

*

*/

fw.write("abcde"+LINE_SEPARATOR+"hahaha");


//

fw.write("xixi");

/*

* 进行刷新,将数据直接写到目的地中。

*/

 

 

 

 

 

 

196


 

 

 

//

 

 

 

 

 

 

 

 

//

 

 

 

fw.flush();

/*

* 关闭流,关闭资源。在关闭前会先调用flush刷新缓冲中的数据到目的地。

*/

fw.close();

 

 

fw.write("haha");// java.io.IOException: Stream closed


 

 

}

 

 

}

 

 

IOException演示:

package cn.itcast.p2.io.filewriter;

 

 

import java.io.FileWriter;

import java.io.IOException;

 

 

public class IOExceptionDemo {

 

 

private static final String LINE_SEPARATOR = System

.getProperty("line.separator");

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) {

 

 

FileWriter fw = null;

try {

 

 

fw = new FileWriter("k:\\demo.txt");

 

 

fw.write("abcde" + LINE_SEPARATOR + "hahaha");

 

 

} catch (IOException e) {

System.out.println(e.toString());

} finally {

if (fw != null)

try {

fw.close();

} catch (IOException e) {

// code....

throw new RuntimeException("关闭失败");

}

}

 

197


 

 

 

}

}

FileReader演示1:

package cn.itcast.p3.io.filereader;

 

 

import java.io.FileNotFoundException;

import java.io.FileReader;

import java.io.IOException;

 

 

//需求:读取一个文本文件。将读取到的字符打印到控制台.

 

public class FileReaderDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

//1,创建读取字符数据的流对象。

/*

* 在创建读取流对象时,必须要明确被读取的文件。一定要确定该文件是存在的。

*

* 用一个读取流关联一个已存在文件。

*/

FileReader fr = new FileReader("demo.txt");

 

 

int ch = 0;

 

 

while((ch=fr.read())!=-1){

System.out.println((char)ch);

}

/*

//用Reader中的read方法读取字符。

int ch = fr.read();

System.out.println((char)ch);

int ch1 = fr.read();

System.out.println(ch1);

int ch2 = fr.read();

System.out.println(ch2);

*/

fr.close();

}

}

 

 

FileReader演示2:

package cn.itcast.p3.io.filereader;

 

198


 

 

 

 

import java.io.FileReader;

import java.io.IOException;

 

 

//需求:读取一个文本文件。将读取到的字符打印到控制台.

 

public class FileReaderDemo2 {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

FileReader fr = new FileReader("demo.txt");

/*

* 使用read(char[])读取文本文件数据。

*

* 先创建字符数组。

*/

char[] buf = new char[1024];

 

 

int len = 0;

 

 

while((len=fr.read(buf))!=-1){

System.out.println(new String(buf,0,len));

}

/*

int num = fr.read(buf);//将读取到的字符存储到数组中。

System.out.println(num+":"+new String(buf,0,num));

int num1 = fr.read(buf);//将读取到的字符存储到数组中。

System.out.println(num1+":"+new String(buf,0,num1));

int num2 = fr.read(buf);//将读取到的字符存储到数组中。

System.out.println(num2+":"+new String(buf));

*/

fr.close();

}

 

 

}


 

 

140.

字符流(ReaderWriter)基本使用:


 

案例1:

package cn.itcast.p1.io.charstream.test;

 

 

import java.io.FileReader;

import java.io.FileWriter;

 

199


 

 

 

import java.io.IOException;

 

 

/*

* 需求:将盘中的一个文件进行复制。

*

* 思路:

* 1,需要读取源,

* 2,将读到的源数据写入到目的地。

* 3,既然是操作文本数据,使用字符流。

*

*/

 

 

public class CopyTextTest {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

//1,读取一个已有的文本文件,使用字符读取流和文件相关联。

FileReader fr = new FileReader("IO流_2.txt");

//2,创建一个目的,用于存储读到数据。

FileWriter fw = new FileWriter("copytext_1.txt");

//3,频繁的读写操作。

int ch = 0;

while((ch=fr.read())!=-1){

fw.write(ch);

}

//4,关闭流资源。

 

fw.close();

fr.close();

}

 

 

}

案例2:

package cn.itcast.p1.io.charstream.test;

 

 

import java.io.FileReader;

import java.io.FileWriter;

import java.io.IOException;

 

 

public class CopyTextTest_2 {

 

 

private static final int BUFFER_SIZE = 1024;

 

 

200


 

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

FileReader fr = null;

FileWriter fw = null;

try {

fr = new FileReader("IO流_2.txt");

fw = new FileWriter("copytest_2.txt");

 

 

//创建一个临时容器,用于缓存读取到的字符。

char[] buf = new char[BUFFER_SIZE];//这就是缓冲区。

 

 

//定义一个变量记录读取到的字符数,(其实就是往数组里装的字符个数 )

int len = 0;

 

 

while((len=fr.read(buf))!=-1){

fw.write(buf, 0, len);

}

 

 

} catch (Exception e) {


//

System.out.println("读写失败");


throw new RuntimeException("读写失败");

}finally{

if(fw!=null)

try {

fw.close();

} catch (IOException e) {

 

 

e.printStackTrace();

}

if(fr!=null)

try {

fr.close();

} catch (IOException e) {

 

 

e.printStackTrace();

}

}

}

 

 

}


 

 

141.

BufferedReader演示:


 

package cn.itcast.p2.io.charstream.buffer;

 

201


 

 

 

 

import

import

import

import

 

 

 

 

java.io.BufferedReader;

java.io.FileNotFoundException;

java.io.FileReader;

java.io.IOException;


 

 

public class BufferedReaderDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

FileReader fr = new FileReader("buf.txt");

 

 

BufferedReader bufr = new BufferedReader(fr);

 

 

String line = null;

 

 

while((line=bufr.readLine())!=null){

System.out.println(line);

}

/*

String line1 = bufr.readLine();

System.out.println(line1);

String line2 = bufr.readLine();

System.out.println(line2);

String line3 = bufr.readLine();

System.out.println(line3);

String line4 = bufr.readLine();

System.out.println(line4);

String line5 = bufr.readLine();

System.out.println(line5);

*/

 

 

bufr.close();

 

 

}

 

 

/**

* @throws FileNotFoundException

* @throws IOException

*/

public static void demo() throws FileNotFoundException, IOException {

FileReader fr = new FileReader("buf.txt");

 

 

202


 

 

 

char[] buf = new char[1024];

 

 

int len = 0;

while((len=fr.read(buf))!=-1){

System.out.println(new String(buf,0,len));

}

 

 

fr.close();

}

 

 

}


 

 

142.

BufferedWriter演示:


 

package cn.itcast.p2.io.charstream.buffer;

 

 

import java.io.BufferedWriter;

import java.io.FileWriter;

import java.io.IOException;

 

 

public class BufferedWriterDemo {


 

 

private   static   final

 

String

 

LINE_SEPARATOR

 

=


System.getProperty("line.separator");

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

FileWriter fw = new FileWriter("buf.txt");

 

 

//为了提高写入的效率。使用了字符流的缓冲区。

//创建了一个字符写入流的缓冲区对象,并和指定要被缓冲的流对象相关联

BufferedWriter bufw = new BufferedWriter(fw);

 

 

//使用缓冲区的写入方法将数据先写入到缓冲区中。


//

//

//

//

bufw.write("abcdefq"+LINE_SEPARATOR+"hahahha");

bufw.write("xixiixii");

bufw.newLine();

bufw.write("heheheheh");

 

 

for(int x=1; x<=4; x++){

bufw.write("abcdef"+x);

bufw.newLine();

bufw.flush();

 

 

 

 

 

 

 

 

 

 

 

 

 

 

203


 

 

 

}

 

 

//使用缓冲区的刷新方法将数据刷目的地中。


//

 

 

 

 

 

 

//

 

 

//

 

 

 

}

 

 

 

 

 

 

 

 

 

 

 

}

bufw.flush();

 

 

//关闭缓冲区。其实关闭的就是被缓冲的流对象。

bufw.close();

 

 

fw.write("hehe");

 

 

fw.close();


 

 

143.

用缓冲区方法buffer读写文件:


 

package cn.itcast.p3.io.charstream.buffer.test;


 

 

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.BufferedWriter;

java.io.FileNotFoundException;

java.io.FileReader;

java.io.FileWriter;

java.io.IOException;


 

 

public class CopyTextByBufTest {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

FileReader fr = new FileReader("buf.txt");

BufferedReader bufr = new BufferedReader(fr);

 

 

FileWriter fw = new FileWriter("buf_copy.txt");

BufferedWriter bufw = new BufferedWriter(fw);

 

 

String line = null;

while((line=bufr.readLine())!=null){

bufw.write(line);

bufw.newLine();

bufw.flush();

}

 

 

204


 

 

 

/*

int ch = 0;

 

 

while((ch=bufr.read())!=-1){

 

 

bufw.write(ch);

}

*/

bufw.close();

bufr.close();

}

 

 

}


 

 

144.

自定义缓冲区(练习):


 

package cn.itcast.p4.io.charstream.mybuffer;

 

 

import java.io.FileReader;

import java.io.IOException;

import java.io.Reader;

 

 

/**

* 自定义的读取缓冲区。其实就是模拟一个BufferedReader.

*


*

*

*

*

分析:

缓冲区中无非就是封装了一个数组,

并对外提供了更多的方法对数组进行访问。

其实这些方法最终操作的都是数组的角标。


*

* 缓冲的原理:

* 其实就是从源中获取一批数据装进缓冲区中。

* 在从缓冲区中不断的取出一个一个数据。

*

* 在此次取完后,在从源中继续取一批数据进缓冲区。

* 当源中的数据取光时,用-1作为结束标记。

*/

public class MyBufferedReader extends Reader {

 

 

private Reader r;

 

 

//定义一个数组作为缓冲区。

private char[] buf = new char[1024];

 

 

//定义一个指针用于操作这个数组中的元素。当操作到最后一个元素后,指针应该归零。

private int pos = 0;

 

 

205


 

 

 

//定义一个计数器用于记录缓冲区中的数据个数。 当该数据减到0,就从源中继续获取数据到缓冲区中。

private int count = 0;

 

 

MyBufferedReader(Reader r){

this.r = r;

}

 

 

/**

* 该方法从缓冲区中一次取一个字符。

* @return

* @throws IOException

*/

public int myRead() throws IOException{

 

 

if(count==0){

count = r.read(buf);

pos = 0;

}

if(count<0)

return -1;

 

 

char ch = buf[pos++];

 

 

count--;

 

 

return ch;

 

 

/*

//1,从源中获取一批数据到缓冲区中。需要先做判断,只有计数器为0时,才需要从源中获取数据。

if(count==0){

count = r.read(buf);

 

 

if(count<0)

return -1;

 

 

//每次获取数据到缓冲区后,角标归零.

pos = 0;

char ch = buf[pos];

 

 

pos++;

count--;

 

 

return ch;

 

 

}else if(count>0){

 

 

char ch = buf[pos];

 

206


 

 

 

 

pos++;

count--;

 

 

return ch;

 

 

}*/

 

 

}

 

 

public String myReadLine() throws IOException{

 

 

StringBuilder sb = new StringBuilder();

 

 

int ch = 0;

while((ch = myRead())!=-1){

 

 

if(ch=='\r')

continue;

if(ch=='\n')

return sb.toString();

//将从缓冲区中读到的字符,存储到缓存行数据的缓冲区中。

sb.append((char)ch);

 

 

}

 

 

if(sb.length()!=0)

return sb.toString();

return null;

}

 

 

public void myClose() throws IOException {

 

 

r.close();

}

 

 

@Override

public int read(char[] cbuf, int off, int len) throws IOException {

 

 

return 0;

}

 

 

@Override

public void close() throws IOException {

}

}

 

 

207


 

 

 

 

 

145.

 

 

 

 

装饰设计模式及演示:


 

装饰设计模式:对一组对象的功能进行增强时,就可以使用该模式进行问题的解决。

装饰和继承都能实现一样的特点:进行功能的扩展增强,但是装饰比继承灵活,装饰的特点:装饰类和被装饰类

都必须所属同一个接口或者父类。

演示:

package cn.itcast.p5.wrapper;

 

 

public class PersonDemo {

public static void main(String[] args) {

 

 

Person p = new Person();


//

p.chifan();


 

 

NewPerson p1 = new NewPerson(p);

p1.chifan();

 

 

NewPerson2 p2 = new NewPerson2();

p2.chifan();

}

 

 

}

 

 

class Person{

void chifan(){

System.out.println("吃饭");

}

}

//这个类的出现是为了增强Person而出现的。

class NewPerson{

private Person p ;

NewPerson(Person p){

this.p = p;

}

 

 

public void chifan(){

System.out.println("开胃酒");

p.chifan();

System.out.println("甜点");

 

}

 

 

}

 

 

class NewPerson2 extends Person{

public void chifan(){

 

208


 

 

 

System.out.println("开胃酒");

super.chifan();

System.out.println("甜点");

}

}


 

 

146.

LineNumberReader演示:


 

package cn.itcast.p6.io.charstream.linenumber;

 

 

import java.io.FileReader;

import java.io.IOException;

import java.io.LineNumberReader;

 

 

public class LineNumberReaderDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

FileReader fr = new FileReader("IO流_2.txt");

LineNumberReader lnr = new LineNumberReader(fr);

 

 

String line = null;

lnr.setLineNumber(100);

while((line=lnr.readLine())!=null){

System.out.println(lnr.getLineNumber()+":"+line);

}

 

 

lnr.close();

}

}


 

 

147.

字节流(InputStreamOutputStream)使用演


 

示:

 

package cn.itcast.p7.io.bytestream.demo;


 

 

import

import

import

import

 

java.io.FileInputStream;

java.io.FileNotFoundException;

java.io.FileOutputStream;

java.io.IOException;

 

 

 

 

 

 

 

 

 

209


 

 

 

public class ByteStreamDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

demo_read();

}

 

 

public static void demo_read() throws IOException {

 

 

//1,创建一个读取流对象。和指定文件关联。

FileInputStream fis = new FileInputStream("bytedemo.txt");


 

 

//

//

//

//

 

 

 

 

 

//

//

//

//

//

//

 

 

//

//

//

//

 

 

 

//

//

 

System.out.println(fis.available());

byte[] buf = new byte[fis.available()];

fis.read(buf);

System.out.println(new String(buf));

 

 

 

//建议使用这种读取数据的方式

byte[] buf = new byte[1024];

int len = 0;

 

 

while((len=fis.read(buf))!=-1){

System.out.println(new String(buf,0,len));

}

 

 

int ch = 0;

while((ch=fis.read())!=-1){

System.out.println((char)ch);

}

 

 

//一次读取一个字节。

int ch = fis.read();

System.out.println(ch);

 

 

fis.close();

 

 

}

 

 

public static void demo_write() throws IOException {

 

 

//1,创建字节输出流对象。用于操作文件.

FileOutputStream fos = new FileOutputStream("bytedemo.txt");

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

210


 

 

 

 

 

//2,写数据。直接写入到了目的地中。

fos.write("abcdefg".getBytes());


 

 

//

 

fos.flush();


fos.close();//关闭资源动作要完成。

}

 

 

}


 

 

148.

用字节流复制mp3的几种方法比较:


 

package cn.itcast.p7.io.bytestream.test;


 

 

import

import

import

import

import

 

java.io.BufferedInputStream;

java.io.BufferedOutputStream;

java.io.FileInputStream;

java.io.FileOutputStream;

java.io.IOException;


 

 

public class CopyMp3Test {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

copy_4();

 

 

}

// 千万不要用,效率没有!

public static void copy_4() throws IOException {

FileInputStream fis = new FileInputStream("c:\\0.mp3");

FileOutputStream fos = new FileOutputStream("c:\\4.mp3");

 

 

int ch = 0;

 

 

while((ch =fis.read())!=-1){

fos.write(ch);

}

 

 

fos.close();

fis.close();

}

 

 

//不建议。

 

211


 

 

 

public static void copy_3() throws IOException {

FileInputStream fis = new FileInputStream("c:\\0.mp3");

FileOutputStream fos = new FileOutputStream("c:\\3.mp3");

 

 

byte[] buf = new byte[fis.available()];

fis.read(buf);

fos.write(buf);

fos.close();

fis.close();

}

 

 

public static void copy_2() throws IOException {

 

 

FileInputStream fis = new FileInputStream("c:\\0.mp3");

BufferedInputStream bufis = new BufferedInputStream(fis);

 

 

FileOutputStream fos = new FileOutputStream("c:\\2.mp3");

BufferedOutputStream bufos = new BufferedOutputStream(fos);

 

 

 

 

 

int ch = 0;

 

 

while((ch=bufis.read())!=-1){

bufos.write(ch);

}

 

 

bufos.close();

bufis.close();

}

 

 

public static void copy_1() throws IOException {

 

 

FileInputStream fis = new FileInputStream("c:\\0.mp3");

FileOutputStream fos = new FileOutputStream("c:\\1.mp3");

 

 

byte[] buf = new byte[1024];

 

 

int len = 0;

 

 

while((len=fis.read(buf))!=-1){

fos.write(buf,0,len);

}

 

 

fos.close();

fis.close();

}

 

212


 

 

 

}


 

 

149.

IO不集合容器结合练习题:


 

package test;

 

 

import java.util.ArrayList;

 

 

/*

一个ArrayList对象alist中存有若干个字符串元素,现欲遍历该ArrayList对象,

删除其中所有值为"abc"的字符串元素,请问下面的实现正确么?如有问题,

会出现什么问题?如何更正?

。。。

int size = alist.size();

for(int i = 0; i < size; i++) {

if("abc".equals(alist.get(i))) {

alist.remove(i);

}

}

*/

 

 

public class Test13 {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

ArrayList al = new ArrayList();

 

 

al.add("haha");

al.add("abc");

al.add("abc");

al.add("abc");


 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

}

 

int size = al.size();

 

 

for(int x=0; x

if("abc".equals(al.get(x))){

al.remove(x);

x--;

}

}

 

 

System.out.println(al);

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

213


 

 

 

}


 

 

150.

读叏键盘录入数据:


 

package cn.itcast.io.p1.transstream.demo;

 

 

import java.io.IOException;

import java.io.InputStream;

/*

* 读取一个键盘录入的数据,并打印在控制台上。

*

* 键盘本身就是一个标准的输入设备。

* 对于java而言,对于这种输入设备都有对应的对象。

*/

public class ReadKey {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {


 

 

//

//

//

 

readKey();

System.out.println((int)'\r');

System.out.println((int)'\n');

 

 

readKey2();

 

 

}

 

 

public static void readKey2() throws IOException {

 

 

/*

* 获取用户键盘录入的数据,

* 并将数据变成大写显示在控制台上,

* 如果用户输入的是over,结束键盘录入。

*

* 思路:

* 1,因为键盘录入只读取一个字节,要判断是否是over,需要将读取到的字节拼成字符串。

* 2,那就需要一个容器。StringBuilder.

* 3,在用户回车之前将录入的数据变成字符串判断即可。

*/

 

 

//1,创建容器。

StringBuilder sb = new StringBuilder();

 

 

//2,获取键盘读取流。

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

214


 

 

 

InputStream in = System.in;

 

 

//3,定义变量记录读取到的字节,并循环获取。

int ch = 0;

 

 

while((ch=in.read())!=-1){


 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

 

//

//

 

在存储之前需要判断是否是换行标记 ,因为换行标记不存储。

if(ch=='\r')

continue;

if(ch=='\n'){

String temp = sb.toString();

if("over".equals(temp))

break;

System.out.println(temp.toUpperCase());

sb.delete(0, sb.length());

}

else

//将读取到的字节存储到StringBuilder中。

sb.append((char)ch);

 

 

System.out.println(ch);

}

 

 

}

 

 

public static void readKey() throws IOException {

 

 

InputStream in = System.in;

 

 

int ch = in.read();//阻塞式方法。

System.out.println(ch);

int ch1 = in.read();//阻塞式方法。

System.out.println(ch1);

int ch2 = in.read();//阻塞式方法。

System.out.println(ch2);

 

 

in.close();

 

 

InputStream in2 = System.in;

int ch3 = in2.read();


 

 

}

}


 

 

151.

转换流演示:

 

 

 

 

215


 

 

 

演示1:

package cn.itcast.io.p1.transstream.demo;


 

 

import

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.BufferedWriter;

java.io.IOException;

java.io.InputStream;

java.io.InputStreamReader;

java.io.OutputStream;

java.io.OutputStreamWriter;


 

 

public class TransStreamDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

//字节流。

InputStream in = System.in;


//

//

//

//

 

 

 

 

 

 

//

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

int ch = in.read();

System.out.println(ch);

int ch1 = in.read();

System.out.println(ch1);

 

 

//将字节转成字符的桥梁。装换流。

InputStreamReader isr = new InputStreamReader(in);

 

 

int ch = isr.read();

System.out.println((char)ch);

 

 

//字符流。

BufferedReader bufr = new BufferedReader(isr);

 

 

OutputStream out = System.out;

 

 

OutputStreamWriter osw = new OutputStreamWriter(out);

 

 

BufferedWriter bufw = new BufferedWriter(osw);

 

 

String line = null;

 

 

while((line=bufr.readLine())!=null){

if("over".equals(line))

break;

System.out.println(line.toUpperCase());

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

216


 

 

 

//

//

 

 

 

osw.write(line.toUpperCase()+"\r\n");

osw.flush();


 

 

bufw.write(line.toUpperCase());

bufw.newLine();

bufw.flush();

}

}

}

演示2:

package cn.itcast.io.p1.transstream.demo;


 

 

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.BufferedWriter;

java.io.FileOutputStream;

java.io.IOException;

java.io.InputStreamReader;

java.io.OutputStreamWriter;


 

 

public class TransStreamDemo2 {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

 

/*

* 1,需求:将键盘录入的数据写入到一个文件中。

*

* 2,需求:将一个文本文件内容显示在控制台上。

*

* 3,需求:将一个文件文件中的内容复制到的另一个文件中。

*/

 

 

BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));


 

 

BufferedWriter bufw =

 

new

 

BufferedWriter(new

 

OutputStreamWriter(new


FileOutputStream("b.txt")));

 

 

 

String line = null;

 

 

while((line=bufr.readLine())!=null){

if("over".equals(line))

break;

 

217


 

 

 

 

bufw.write(line.toUpperCase());

bufw.newLine();

bufw.flush();

}

 

 

}

 

 

}

演示3:

package cn.itcast.io.p1.transstream.demo;


 

 

import

import

import

import

import

import

import

import

import

 

java.io.FileInputStream;

java.io.FileNotFoundException;

java.io.FileOutputStream;

java.io.FileReader;

java.io.FileWriter;

java.io.IOException;

java.io.InputStreamReader;

java.io.OutputStreamWriter;

java.io.UnsupportedEncodingException;


 

 

public class TransStreamDemo3 {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

readText_2();

}

 

 

public static void readText_2() throws IOException, FileNotFoundException {


 

 

InputStreamReader  isr  =

 

new

 

InputStreamReader(new


FileInputStream("gbk_1.txt"),"utf-8");

char[] buf = new char[10];

int len = isr.read(buf);

String str = new String(buf,0,len);

System.out.println(str);

 

 

isr.close();

}

 

 

public static void readText_1() throws IOException {

 

 

218


 

 

 

FileReader fr = new FileReader("gbk_1.txt");

 

 

char[] buf = new char[10];

int len = fr.read(buf);

String str = new String(buf,0,len);

System.out.println(str);

 

 

fr.close();

 

 

}

 

 

public static void writeText_3() throws IOException {


 

 

OutputStreamWriter  osw

 

=

 

new

 

OutputStreamWriter(new


FileOutputStream("u8_1.txt"),"UTF-8");

 

 

osw.write("你好");

osw.close();

 

 

}

 

 

public static void writeText_2() throws IOException {


 

 

OutputStreamWriter

 

osw

 

=

 

new

 

OutputStreamWriter(new


FileOutputStream("gbk_3.txt"),"GBK");


 

 

//

 

OutputStreamWriter  osw  =  new

 

OutputStreamWriter(new


FileOutputStream("gbk_3.txt"),"GBK");


//

FileWriter fw = new FileWriter("gbk_1.txt");

 

 

/*


*

*

*

*

*

这两句代码的功能是等同的。

FileWriter:其实就是转换流指定了本机默认码表的体现。而且这个转换流的子类对象,可以方

便操作文本文件。

简单说:操作文件的字节流+本机默认的编码表。

这是按照默认码表来操作文件的便捷类。


*

* 如果操作文本文件需要明确具体的编码。FileWriter就不行了。必须用转换流。

*/

osw.write("你好");

 

osw.close();

 

 

}

 

 

public static void writeText_1() throws IOException {

 

 

219


 

 

 

FileWriter fw = new FileWriter("gbk_1.txt");

 

 

fw.write("你好");

 

fw.close();

}

 

 

}


 

 

152.

File类对象+过滤器的使用演示:


 

演示1:

package cn.itcast.io.p2.file.demo;

 

 

import java.io.File;

 

 

public class FileDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {


 

 

//

 

constructorDemo();


 

 

}

 

 

public static void constructorDemo() {

 

 

//可以将一个已存在的,或者不存在的文件或者目录封装成file对象。

File f1 = new File("c:\\a.txt");

 

 

File f2 = new File("c:\\","a.txt");

 

 

File f = new File("c:\\");

 

 

File f3 = new File(f,"a.txt");

 

 

File f4 = new File("c:"+File.separator+"abc"+File.separator+"a.txt");

System.out.println(f4);

 

 

}

}

演示2:

(三种常见过滤器):

1-后缀名过滤器

package cn.itcast.io.p2.filter;

 

220


 

 

 

 

import java.io.File;

import java.io.FilenameFilter;

 

 

public class FilterByJava implements FilenameFilter {

 

 

@Override

public boolean accept(File dir, String name) {


 

 

//

 

System.out.println(dir+"---"+name);


 

 

return name.endsWith(".java");

}

 

 

}

2-根据后缀名可以指定任意后缀名过滤的过滤器

package cn.itcast.io.p2.filter;

 

 

import java.io.File;

import java.io.FilenameFilter;

 

 

public class SuffixFilter implements FilenameFilter {

 

 

private String suffix ;

public SuffixFilter(String suffix) {

super();

this.suffix = suffix;

}

@Override

public boolean accept(File dir, String name) {

 

 

return name.endsWith(suffix);

}

}

3-隐藏属性过滤器

package cn.itcast.io.p2.filter;

 

 

import java.io.File;

import java.io.FileFilter;

 

 

public class FilterByHidden implements FileFilter {

 

 

@Override

public boolean accept(File pathname) {

 

 

return !pathname.isHidden();

}

 

221


 

 

 

}

 

 

(与过滤器结合使用的演示):

package cn.itcast.io.p2.file.demo;

 

 

import java.io.File;

 

 

import cn.itcast.io.p2.filter.FilterByHidden;

import cn.itcast.io.p2.filter.SuffixFilter;

 

 

public class FileListDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

listDemo_2();

}

 

 

public static void listDemo_3() {

 

 

File dir = new File("c:\\");

 

 

File[] files = dir.listFiles(new FilterByHidden());

 

 

for(File file : files){

System.out.println(file);

}

}

 

 

public static void listDemo_2() {

 

 

File dir = new File("c:\\");

 

 

String[] names = dir.list(new SuffixFilter(".txt"));

 

 

for(String name : names){

System.out.println(name);

}

 

 

}

 

 

public static void listDemo() {

File file = new File("c:\\");

 

 

/*

 

222


 

 

 

*

*

*

*

 

 

 

获取当前目录下的文件以及文件夹的名称,包含隐藏文件。

调用list方法的File对象中封装的必须是目录。

否则会发生NullPointerException

如果访问的系统级目录也会发生空指针异常。


*

* 如果目录存在但是没有内容,会返回一个数组,但是长度为0.

*

*/

String[] names = file.list();

System.out.println(names.length);

 

 

for(String name : names){

System.out.println(name);

}

}

}

演示3:

package cn.itcast.io.p2.file.demo;


 

 

import

import

import

import

 

java.io.File;

java.io.IOException;

java.text.DateFormat;

java.util.Date;


 

 

public class FileMethodDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

/*

* File对象的常见方法。

*

* 1,获取。


*

*

*

*

1.1 获取文件名称。

1.2 获取文件路径。

1.3 获取文件大小。

1.4 获取文件修改时间。


*

* 2,创建与删除。

*

* 3,判断。

*

* 4, 重命名

*

 

223


 

 

 

*/


//

//

//

//

 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

//

//

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

 

 

//

//

getDemo();

createAndDeleteDemo();

isDemo();

renameToDemo();

 

 

listRootsDemo();

}

 

 

public static void listRootsDemo() {

 

 

File file = new File("d:\\");

 

 

System.out.println("getFreeSpace:"+file.getFreeSpace());

System.out.println("getTotalSpace:"+file.getTotalSpace());

System.out.println("getUsableSpace:"+file.getUsableSpace());

 

 

File[] files = File.listRoots();

for(File file : files){

System.out.println(file);

}

}

 

 

public static void renameToDemo() {

 

 

File f1 = new File("c:\\9.mp3");

 

 

File f2 = new File("d:\\aa.mp3");

 

 

boolean b = f1.renameTo(f2);

 

 

System.out.println("b="+b);

}

 

 

public static void isDemo() throws IOException{

File f = new File("aaa");

 

 

f.mkdir();

f.createNewFile();

 

 

boolean b = f.exists();

System.out.println("b="+b);

// 最好先判断是否存在。

System.out.println(f.isFile());

System.out.println(f.isDirectory());

}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

224


 

 

 

public static void createAndDeleteDemo() throws IOException {

 

 

File dir = new File("abc\\q\\e\\c\\z\\r\\w\\y\\f\\e\\g\\s");


 

 

//

//

//

 

 

 

//

 

 

//

//

 

 

 

 

 

 

//

//

 

 

//

//

 

 

 

 

 

 

//

 

boolean b = dir.mkdir();//make directory

System.out.println("b="+b);

dir.mkdirs();//创建多级目录

System.out.println(dir.delete());

 

 

System.out.println(dir.delete());

 

 

文件的创建和删除。

File file = new File("file.txt");

/*

* 和输出流不一样,如果文件不存在,则创建,如果文件存在,则不创建。

*

*/

boolean b = file.createNewFile();

System.out.println("b="+b);

 

 

boolean b = file.delete();

System.out.println("b="+b);

 

 

}

 

 

public static void getDemo(){

File file = new File("E:\\java0331\\day22e\\a.txt");

File file = new File("a.txt");

 

 

String name = file.getName();

 

 

String absPath = file.getAbsolutePath();//绝对路径。

 

String path = file.getPath();

 

 

long len = file.length();

 

 

long time = file.lastModified();

 

 

Date date = new Date(time);


 

 

DateFormat          dateFormat

 

=


DateFormat.getDateTimeInstance(DateFormat.LONG,DateFormat.LONG);

 

 

String str_time = dateFormat.format(date);

 

 

System.out.println("parent:"+file.getParent());

 

225


 

 

 

 

System.out.println("name:"+name);

System.out.println("absPath:"+absPath);

System.out.println("path:"+path);

System.out.println("len:"+len);

System.out.println("time:"+time);

System.out.println("str_time:"+str_time);

}

}


 

 

153.

IO流操作规徇及需求演示:


 

流的操作规律:

之所以要弄清楚这个规律,是因为流对象太多,开发时不知道用哪个对象合适。

 

 

想要知道开发时用到哪些对象。只要通过四个明确即可。

 

 

1,明确源和目的(汇)

源:InputStream Reader

目的:OutputStream Writer

 

 

2,明确数据是否是纯文本数据。

源:是纯文本:Reader

否:InputStream

目的:是纯文本 Writer

否:OutputStream

 

 

到这里,就可以明确需求中具体要使用哪个体系。

 

 

3,明确具体的设备。

源设备:

硬盘:File

键盘:System.in

内存:数组

网络:Socket流

 

 

目的设备:

硬盘:File

控制台:System.out

内存:数组

网络:Socket流

 

 

4,是否需要其他额外功能。

1,是否需要高效(缓冲区);

是,就加上buffer.

2,转换。

 

 

226


 

 

 

--------------------------------------------------------------------------------

 

 

需求1:复制一个文本文件。

1,明确源和目的。

源:InputStream Reader

目的:OutputStream Writer

2,是否是纯文本?

是!

源:Reader

目的:Writer

 

 

3,明确具体设备。

源:

硬盘:File

目的:

硬盘:File

 

FileReader fr = new FileReader("a.txt");

FileWriter fw = new FileWriter("b.txt");

 

 

4,需要额外功能吗?

需要,需要高效。

BufferedReader bufr = new BufferedReader(new FileReader("a.txt"));

BufferedWriter bufw = new BufferedWriter(new FileWriter("b.txt"));

 

 

--------------------------------------------------------------------------------

需求2:读取键盘录入信息,并写入到一个文件中。

 

 

1,明确源和目的。

源:InputStream Reader

目的:OutputStream Writer

2,是否是纯文本呢?

是,

源:Reader

目的:Writer

3,明确设备

源:

键盘。System.in

目的:

硬盘。File

 

InputStream in = System.in;

FileWriter fw = new FileWriter("b.txt");

这样做可以完成,但是麻烦。将读取的字节数据转成字符串。再由字符流操作。

4,需要额外功能吗?


需要。转换。

将字节流转成字符流。因为名确的源是Reader,这样操作文本数据做便捷。


所以要将已有的字节流转成字符流。使用字节-->字符 。InputStreamReader

 

227


 

 

 

InputStreamReader isr = new InputStreamReader(System.in);

FileWriter fw = new FileWriter("b.txt");

 

 

还需要功能吗?

需要:想高效。

BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));

BufferedWriter bufw = new BufferedWriter(new FileWriter("b.txt"));

 

 

-------------------------------------------------------------------------------

需求3:将一个文本文件数据显示在控制台上。

1,明确源和目的。

源:InputStream Reader

目的:OutputStream Writer

2,是否是纯文本呢?

是,

源:Reader

目的:Writer

3,明确具体设备

源:

硬盘:File

目的:

控制台:System.out

 

FileReader fr = new FileReader("a.txt");

OutputStream out = System.out;//PrintStream

4,需要额外功能吗?

需要,转换。

FileReader fr= new FileReader("a.txt");

OutputStreamWriter osw = new OutputStreamWriter(System.out);

需要,高效。

BufferedReader bufr = new BufferedReader(new FileReader("a.txt"));

BufferedWriter bufw = new BufferedWriter(new OutputStreamWriter(System.out));

 

 

--------------------------------------------------------------------------------

需求4:读取键盘录入数据,显示在控制台上。

1,明确源和目的。

源:InputStream Reader

目的:OutputStream Writer

2,是否是纯文本呢?

是,

源:Reader

目的:Writer

3,明确设备。

源:

键盘:System.in

目的:

控制台:System.out

 

228


 

 

 

 

InputStream in = System.in;

OutputStream out = System.out;

 

 

4,明确额外功能?

需要转换,因为都是字节流,但是操作的却是文本数据。

所以使用字符流操作起来更为便捷。

InputStreamReader isr = new InputStreamReader(System.in);

OutputStreamWriter osw = new OutputStreamWriter(System.out);

 

 

为了将其高效。

BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));

BufferedWriter bufw = new BufferedWriter(new OutputStreamWriter(System.out));

 

 

--------------------------------------------------------------------------------

5,将一个中文字符串数据按照指定的编码表写入到一个文本文件中.

 

 

1,目的。OutputStream,Writer

2,是纯文本,Writer。

3,设备:硬盘File

FileWriter fw = new FileWriter("a.txt");

fw.write("你好");

 

 

注意:既然需求中已经明确了指定编码表的动作。

那就不可以使用FileWriter,因为FileWriter内部是使用默认的本地码表。

只能使用其父类。OutputStreamWriter.

OutputStreamWriter 接 收 一 个 字 节 输 出 流 对 象 , 既 然 是 操 作 文 件 , 那 么 该 对 象 应 该 是

FileOutputStream


 

 

OutputStreamWriter

 

osw

 

=

 

new

 

OutputStreamWriter(new


FileOutputStream("a.txt"),charsetName);

 

 

需要高效吗?


BufferedWriter

bufw

= new BufferedWriter(new

OutputStreamWriter(new


FileOutputStream("a.txt"),charsetName));

 

 

什么时候使用转换流呢?

1,源或者目的对应的设备是字节流,但是操作的却是文本数据,可以使用转换作为桥梁。

提高对文本操作的便捷。

2,一旦操作文本涉及到具体的指定编码表时,必须使用转换流 。


 

 

154.

递弻简述:


 

package cn.itcast.io.p1.digui;

 

 

public class DiGuiDemo {

 

 

229


 

 

 

/**

* @param args

*/

public static void main(String[] args) {

/*

* 递归:

* 函数自身直接或者间接的调用到了自身。

* 一个功能在被重复使用,并每次使用时,参与运算的结果和上一次调用有关。

* 这时可以用递归来解决问题。

* 注意:

* 1,递归一定明确条件。否则容易栈溢出。

* 2,注意一下递归的次数。

*/


//

//

show();

toBin(6);

int sum = getSum(9000);

 

 

System.out.println(sum);

}

 

 

public static int getSum(int num){

 

 

int x = 9;

if(num==1)

return 1;

 

 

return num+getSum(num-1);

 

 

}

 

 

public static void toBin(int num){

if(num>0){

 

 

toBin(num/2);

System.out.println(num%2);

 

 

}

}

 

 

/*

public static void show(){

 

 

method();

 

 

}

public static void method(){

show();

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

230


 

 

 

}

*/

}


 

 

155.

深度遍历(递弻使用):


 

package cn.itcast.io.p1.file.test;

 

 

import java.io.File;

 

 

/*

* 需求:对指定目录进行所有内容的列出(包含子目录中的内容)

* 也可以理解为深度遍历。

*/

 

 

public class FileTest {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

File dir = new File("e:\\demodir");

 

 

listAll(dir,0);

}

 

 

public static void listAll(File dir,int level) {

 

 

System.out.println(getSpace(level)+dir.getName());

//获取指定目录下当前的所有文件夹或者文件对象

 

level++;

File[] files = dir.listFiles();

 

 

for(int x=0; xlength; x++){

 

 

if(files[x].isDirectory()){

listAll(files[x],level);

}

else

System.out.println(getSpace(level)+files[x].getName());

}

}

 

 

private static String getSpace(int level) {

 

 

231


 

 

 

StringBuilder sb = new StringBuilder();

 

 

sb.append("|--");

for(int x=0; x

sb.insert(0,"| ");

}

 

 

return sb.toString();

}

 

 

}


 

 

156.

删除一个带内容的目录(递弻使用):


 

package cn.itcast.io.p1.file.test;

 

 

import java.io.File;

 

 

/*

* 删除一个带内容的目录。

* 原理:必须从最里面往外删。

* 需要进行深度遍历。

*/

public class RemoveDirTest {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

File dir = new File("e:\\demodir");


//

dir.delete();

removeDir(dir);

}

 

 

public static void removeDir(File dir) {

 

 

File[] files = dir.listFiles();

 

 

for(File file : files){

 

 

if(file.isDirectory()){

removeDir(file);

}else{

System.out.println(file+":"+file.delete());

}

}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

232


 

 

 

System.out.println(dir+":"+dir.delete());

}

}


 

 

157.

Properties集合+IO流操作演示:


 

package cn.itcast.io.p2.properties;


 

 

import

import

import

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.File;

java.io.FileInputStream;

java.io.FileOutputStream;

java.io.FileReader;

java.io.FileWriter;

java.io.IOException;

java.util.Properties;

java.util.Set;


 

 

public class PropertiesDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

/*

* Map

* |--Hashtable


*

*

*

*

*

*

|--Properties:

 

 

Properties集合:

特点:

1,该集合中的键和值都是字符串类型。

2,集合中的数据可以保存到流中,或者从流获取。


*

* 通常该集合用于操作以键值对形式存在的配置文件。

*/


 

 

//

//

 

methodDemo_4();

myLoad();

 

 

test();

}

 

 

//对已有的配置文件中的信息进行修改。

/*

 

 

 

 

 

 

 

 

 

 

 

 

 

 

233


 

 

 

* 读取这个文件。

* 并将这个文件中的键值数据存储到集合中。

* 在通过集合对数据进行修改。

* 在通过流将修改后的数据存储到文件中。

*/

public static void test() throws IOException{

//读取这个文件。

File file = new File("info.txt");

if(!file.exists()){

file.createNewFile();

}

FileReader fr = new FileReader(file);

 

 

//创建集合存储配置信息。

Properties prop = new Properties();

 

 

//将流中信息存储到集合中。

prop.load(fr);

 

 

prop.setProperty("wangwu", "16");

 

 

FileWriter fw = new FileWriter(file);

 

 

prop.store(fw,"");


 

 

//

 

prop.list(System.out);

 

 

fw.close();

fr.close();

 

 

}

 

 

//模拟一下load方法。

public static void myLoad() throws IOException{

 

 

Properties prop = new Properties();

 

 

BufferedReader bufr = new BufferedReader(new FileReader("info.txt"));

 

 

String line = null;

 

 

while((line=bufr.readLine())!=null){

 

 

if(line.startsWith("#"))

continue;

 

 

String[] arr = line.split("=");

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

234


 

 

 

 

//

 

 

 

 

System.out.println(arr[0]+"::"+arr[1]);

prop.setProperty(arr[0], arr[1]);

}

 

 

prop.list(System.out);

 

 

bufr.close();

 

 

}

 

 

public static void methodDemo_4() throws IOException {

 

 

Properties prop = new Properties();

 

 

//集合中的数据来自于一个文件。

//注意;必须要保证该文件中的数据是键值对。

//需要使用到读取流。

FileInputStream fis = new FileInputStream("info.txt");

 

 

//使用load方法。

prop.load(fis);

 

 

prop.list(System.out);

 

 

}

 

 

public static void methodDemo_3() throws IOException {

Properties prop = new Properties();

 

 

//存储元素。

prop.setProperty("zhangsan","30");

prop.setProperty("lisi","31");

prop.setProperty("wangwu","36");

prop.setProperty("zhaoliu","20");

 

 

//想要将这些集合中的字符串键值信息持久化存储到文件中。

//需要关联输出流。

FileOutputStream fos = new FileOutputStream("info.txt");

 

 

//将集合中数据存储到文件中,使用store方法。

prop.store(fos, "info");

 

 

fos.close();

 

 

}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

235


 

 

 

/**

* 演示Properties集合和流对象相结合的功能。

*/

 

 

public static void methodDemo_2(){

Properties prop = new Properties();

 

 

//存储元素。


//

//

//

//

prop.setProperty("zhangsan","30");

prop.setProperty("lisi","31");

prop.setProperty("wangwu","36");

prop.setProperty("zhaoliu","20");


 

 

prop = System.getProperties();

prop.list(System.out);

}

 

 

/*

* Properties集合的存和取。

*/

 

 

public static void propertiesDemo(){

//创建一个Properties集合。

 

Properties prop = new Properties();

 

 

//存储元素。

prop.setProperty("zhangsan","30");

prop.setProperty("lisi","31");

prop.setProperty("wangwu","36");

prop.setProperty("zhaoliu","20");

 

 

//修改元素。

prop.setProperty("wangwu","26");

 

 

//取出所有元素。

Set names = prop.stringPropertyNames();

 

 

for(String name : names){

String value = prop.getProperty(name);

System.out.println(name+":"+value);

}

}

}


 

 

158.

Properties定义一个程序运行次数的程序:

 

 

 

 

236


 

 

 

package cn.itcast.io.p2.properties;


 

 

import

import

import

import

import

 

java.io.File;

java.io.FileInputStream;

java.io.FileOutputStream;

java.io.IOException;

java.util.Properties;


 

 

/*

* 定义功能,获取一个应用程序运行的次数,如果超过5次,给出使用次数已到请注册的提示。并不要在运行

程序。

*


*

*

*

*

*

*

*

*

*

*

*

*

思路:

1,应该有计数器。

每次程序启动都需要计数一次,并且是在原有的次数上进行计数。

2,计数器就是一个变量。 突然冒出一想法,程序启动时候进行计数,计数器必须存在于内存并进行运算。

可是程序一结束,计数器消失了。那么再次启动该程序,计数器又重新被初始化了。

而我们需要多次启动同一个应用程序,使用的是同一个计数器。

这就需要计数器的生命周期变长,从内存存储到硬盘文件中。

 

 

3,如何使用这个计数器呢?

首先,程序启动时,应该先读取这个用于记录计数器信息的配置文件。

获取上一次计数器次数。 并进行试用次数的判断。

其次,对该次数进行自增,并自增后的次数重新存储到配置文件中。


*

*

* 4,文件中的信息该如何进行存储并体现。

* 直接存储次数值可以,但是不明确该数据的含义。 所以起名字就变得很重要。

* 这就有了名字和值的对应,所以可以使用键值对。

* 可是映射关系map集合搞定,又需要读取硬盘上的数据,所以map+io = Properties.

*/

public class PropertiesTest {

/**

* @param args

* @throws IOException

* @throws Exception

*/

public static void main(String[] args) throws IOException {

 

 

getAppCount();

 

 

}

 

 

public static void getAppCount() throws IOException{

 

 

//将配置文件封装成File对象。

File confile = new File("count.properties");

 

237


 

 

 

 

if(!confile.exists()){

confile.createNewFile();

}

 

 

FileInputStream fis = new FileInputStream(confile);

 

 

Properties prop = new Properties();

 

 

prop.load(fis);

 

 

//从集合中通过键获取次数。

String value = prop.getProperty("time");

//定义计数器。记录获取到的次数。

int count =0;

if(value!=null){

count = Integer.parseInt(value);

if(count>=5){


//

//

System.out.println("使用次数已到,请注册,给钱!");

return;


throw new RuntimeException("使用次数已到,请注册,给钱!");

}

}

count++;

 

 

//将改变后的次数重新存储到集合中。

prop.setProperty("time", count+"");

 

 

FileOutputStream fos = new FileOutputStream(confile);

 

 

prop.store(fos, "");

 

 

fos.close();

fis.close();

}

}


 

 

159.

建立一个指定扩展名文件的列表(练习):


 

package cn.itcast.io.p3.test;


 

 

import

import

import

import

import

import

 

java.io.BufferedWriter;

java.io.File;

java.io.FileWriter;

java.io.FilenameFilter;

java.io.IOException;

java.util.ArrayList;

 

 

 

 

 

 

 

 

 

 

 

238


 

 

 

import java.util.List;

 

 

/*

* 获取指定目录下,指定扩展名的文件(包含子目录中的)

* 这些文件的绝对路径写入到一个文本文件中。

*

* 简单说,就是建立一个指定扩展名的文件的列表。

*

* 思路:

* 1,必须进行深度遍历。

* 2,要在遍历的过程中进行过滤。将符合条件的内容都存储到容器中。

* 3,对容器中的内容进行遍历并将绝对路径写入到文件中。

*/

public class Test {

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

File dir = new File("e:\\java0331");

 

 

FilenameFilter filter = new FilenameFilter(){

@Override

public boolean accept(File dir, String name) {

 

 

return name.endsWith(".java");

}

};

 

 

List list = new ArrayList();

 

 

getFiles(dir,filter,list);

 

 

File destFile = new File(dir,"javalist.txt");

 

 

write2File(list,destFile);

}

/**

* 对指定目录中的内容进行深度遍历,并按照指定过滤器,进行过滤,

* 将过滤后的内容存储到指定容器List中。

* @param dir

* @param filter

* @param list

*/

public static void getFiles(File dir,FilenameFilter filter,List list){

 

 

239


 

 

 

File[] files = dir.listFiles();

 

 

for(File file : files){

if(file.isDirectory()){

//递归啦!

getFiles(file,filter,list);

}else{

//对遍历到的文件进行过滤器的过滤。将符合条件File对象,存储到List集合中。

if(filter.accept(dir, file.getName())){

list.add(file);

}

}

}

 

 

}

 

 

public static void write2File(List list,File destFile)throws IOException{

 

 

BufferedWriter bufw = null;

try {

bufw = new BufferedWriter(new FileWriter(destFile));

for(File file : list){

bufw.write(file.getAbsolutePath());

bufw.newLine();

bufw.flush();

}

 

 

} /*catch(IOException e){

 

 

throw new RuntimeException("写入失败");

}*/finally{

if(bufw!=null)

try {

bufw.close();

} catch (IOException e) {

 

 

throw new RuntimeException("关闭失败");

}

}

}

}


 

 

160.

打印流简述:


 

package cn.itcast.io.p4.print.demo;

 

 

import java.io.FileNotFoundException;

 

240


 

 

 

import java.io.IOException;

import java.io.PrintStream;

 

 

public class PrintStreamDemo {

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

/*

* PrintStream:

* 1,提供了打印方法可以对多种数据类型值进行打印。并保持数据的表示形式。

* 2,它不抛IOException.

*

* 构造函数,接收三种类型的值:

* 1,字符串路径。

* 2,File对象。

* 3,字节输出流。

*/

PrintStream out = new PrintStream("print.txt");


 

 

//

//

 

 

//

 

 

//

 

int by = read();

write(by);

 

 

out.write(610);//只写最低8位,

 

 

out.print(97);//将97先变成字符保持原样将数据打印到目的地。


 

out.close();

}

}


 

 

161.

PrintWriter简述及使用:


 

package cn.itcast.io.p4.print.demo;


 

 

import

import

import

import

import

 

java.io.BufferedReader;

java.io.FileWriter;

java.io.IOException;

java.io.InputStreamReader;

java.io.PrintWriter;


 

 

public class PrintWriterDemo {

 

 

/**

* @param args

* @throws IOException

 

241


 

 

 

*/

public static void main(String[] args) throws IOException {

/*

* PrintWriter:字符打印流。

* 构造函数参数:

* 1,字符串路径。

* 2,File对象。

* 3,字节输出流。

* 4,字符输出流。

*

*/

BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));

 

 

PrintWriter out = new PrintWriter(new FileWriter("out.txt"),true);

 

 

String line = null;

while((line=bufr.readLine())!=null){

if("over".equals(line))

break;

out.println(line.toUpperCase());


//

out.flush();


}

 

 

out.close();

bufr.close();

}

}


 

 

162.

SequenceInputStream序列流使用:


 

package cn.itcast.io.p4.sequence.demo;


 

 

import

import

import

import

import

import

import

import

 

java.io.FileInputStream;

java.io.FileOutputStream;

java.io.IOException;

java.io.SequenceInputStream;

java.util.ArrayList;

java.util.Collections;

java.util.Enumeration;

java.util.Iterator;


 

 

public class SequenceInputStreamDemo {

 

 

/**

* @param args

* @throws IOException

*/

 

242


 

 

 

public static void main(String[] args) throws IOException {

 

 

/*

* 需求:将1.txt 2.txt 3.txt文件中的数据合并到一个文件中。

*/


 

 

//

//

//

//

//

 

Vector v = new Vector();

v.add(new FileInputStream("1.txt"));

v.add(new FileInputStream("2.txt"));

v.add(new FileInputStream("3.txt"));

Enumeration en = v.elements();

 

 

ArrayList al = new ArrayList();

for(int x=1; x<=3; x++){

al.add(new FileInputStream(x+".txt"));

}

 

 

Enumeration en = Collections.enumeration(al);

 

 

/*

final Iterator it = al.iterator();

Enumeration en = new Enumeration(){

 

 

@Override

public boolean hasMoreElements() {

 

 

return it.hasNext();

}

 

 

@Override

public FileInputStream nextElement() {

 

 

return it.next();

}

 

 

};*/

 

 

SequenceInputStream sis = new SequenceInputStream(en);

 

 

FileOutputStream fos = new FileOutputStream("1234.txt");

 

 

byte[] buf = new byte[1024];

 

 

int len = 0;

 

 

while((len=sis.read(buf))!=-1){

fos.write(buf,0,len);

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

243


 

 

 

}

 

 

fos.close();

sis.close();

 

 

}

 

 

}


 

 

163.

文件切割器:


 

package cn.itcast.io.p1.splitfile;


 

 

import

import

import

import

import

 

java.io.File;

java.io.FileInputStream;

java.io.FileOutputStream;

java.io.IOException;

java.util.Properties;


 

 

/*

* 文件切割器。

*/

 

 

public class SplitFileDemo {

 

 

private static final int SIZE = 1024 * 1024;

 

 

/**

* @param args

* @throws Exception

*/

public static void main(String[] args) throws Exception {

 

 

File file = new File("c:\\aa.mp3");

 

 

splitFile_2(file);

}

 

 

private static void splitFile_2(File file) throws IOException {

 

 

// 用读取流关联源文件。

FileInputStream fis = new FileInputStream(file);

 

 

// 定义一个1M的缓冲区。

byte[] buf = new byte[SIZE];

 

 

// 创建目的。

 

244


 

 

 

FileOutputStream fos = null;

 

 

int len = 0;

int count = 1;

 

 

 

/*

* 切割文件时,必须记录住被切割文件的名称,以及切割出来碎片文件的个数。 以方便于合并。

* 这个信息为了进行描述,使用键值对的方式。用到了properties对象

*

*/

Properties prop = new Properties();

 

 

 

 

 

File dir = new File("c:\\partfiles");

if (!dir.exists())

dir.mkdirs();

 

 

while ((len = fis.read(buf)) != -1) {

 

 

fos = new FileOutputStream(new File(dir, (count++) + ".part"));

fos.write(buf, 0, len);

fos.close();

}

 

 

//将被切割文件的信息保存到prop集合中。

prop.setProperty("partcount", count+"");

prop.setProperty("filename", file.getName());

 

 

 

 

 

fos = new FileOutputStream(new File(dir,count+".properties"));

 

 

//将prop集合中的数据存储到文件中。

prop.store(fos, "save file info");

 

 

fos.close();

fis.close();

 

 

}

 

 

public static void splitFile(File file) throws IOException {

 

 

// 用读取流关联源文件。

FileInputStream fis = new FileInputStream(file);

 

 

245


 

 

 

// 定义一个1M的缓冲区。

byte[] buf = new byte[SIZE];

 

 

// 创建目的。

FileOutputStream fos = null;

 

 

int len = 0;

int count = 1;

 

 

File dir = new File("c:\\partfiles");

if (!dir.exists())

dir.mkdirs();

 

 

while ((len = fis.read(buf)) != -1) {

 

 

fos = new FileOutputStream(new File(dir, (count++) + ".part"));

fos.write(buf, 0, len);

}

 

 

fos.close();

fis.close();

 

 

}

 

 

}


 

 

164.

合并文件(流)+配置文件:


 

package cn.itcast.io.p1.splitfile;


 

 

import

import

import

import

import

import

import

import

import

 

java.io.File;

java.io.FileInputStream;

java.io.FileOutputStream;

java.io.IOException;

java.io.SequenceInputStream;

java.util.ArrayList;

java.util.Collections;

java.util.Enumeration;

java.util.Properties;


 

 

public class MergeFile {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

246


 

 

 

 

File dir = new File("c:\\partfiles");

 

 

mergeFile_2(dir);

}

 

 

public static void mergeFile_2(File dir) throws IOException {

 

 

/*

* 获取指定目录下的配置文件对象。

*/

File[] files = dir.listFiles(new SuffixFilter(".properties"));

 

 

if(files.length!=1)

throw new RuntimeException(dir+",该目录下没有properties扩展名的文件或者不唯一

");

//记录配置文件对象。

File confile = files[0];

 

 

//获取该文件中的信息================================================。

 

Properties prop = new Properties();

FileInputStream fis = new FileInputStream(confile);

 

 

prop.load(fis);

 

 

String filename = prop.getProperty("filename");

int count = Integer.parseInt(prop.getProperty("partcount"));

 

 

//获取该目录下的所有碎片文件。 ==============================================

File[] partFiles = dir.listFiles(new SuffixFilter(".part"));

 

 

if(partFiles.length!=(count-1)){

throw new RuntimeException(" 碎片文件不符合要求,个数不对!应该"+count+"个");

}

 

 

//将碎片文件和流对象关联 并存储到集合中。

ArrayList al = new ArrayList();

for(int x=0; x

 

 

al.add(new FileInputStream(partFiles[x]));

}

 

 

//将多个流合并成一个序列流。

Enumeration en = Collections.enumeration(al);

SequenceInputStream sis = new SequenceInputStream(en);

 

 

247


 

 

 

FileOutputStream fos = new FileOutputStream(new File(dir,filename));

 

 

byte[] buf = new byte[1024];

 

 

int len = 0;

while((len=sis.read(buf))!=-1){

fos.write(buf,0,len);

}

 

 

fos.close();

sis.close();

}

 

 

public static void mergeFile(File dir) throws IOException{

 

 

 

ArrayList al = new ArrayList();

 

 

for(int x=1; x<=3 ;x++){

al.add(new FileInputStream(new File(dir,x+".part")));

}

 

 

Enumeration en = Collections.enumeration(al);

SequenceInputStream sis = new SequenceInputStream(en);

 

 

FileOutputStream fos = new FileOutputStream(new File(dir,"1.bmp"));

 

 

byte[] buf = new byte[1024];

 

 

int len = 0;

while((len=sis.read(buf))!=-1){

fos.write(buf,0,len);

}

 

 

fos.close();

sis.close();

}

}


 

 

165.

Serializable标记接口:


 

package cn.itcast.io.p2.bean;

 

 

import java.io.Serializable;

/*

* Serializable:用于给被序列化的类加入ID号。

* 用于判断类和对象是否是同一个版本。

 

248


 

 

 

*/

public class Person implements Serializable/*标记接口*/ {

/**

* transient:非静态数据不想被序列化可以使用这个关键字修饰。

*/

private static final long serialVersionUID = 9527l;

private transient String name;

private static int age;

 

 

public Person(String name, int age) {

super();

this.name = name;

this.age = age;

}

 

 

public String getName() {

return name;

}

 

 

public void setName(String name) {

this.name = name;

}

public int getAge() {

return age;

}

public void setAge(int age) {

this.age = age;

}

 

 

}


 

 

166.

ObjectStream简述:


 

package cn.itcast.io.p2.objectstream;


 

 

import

import

import

import

import

import

 

java.io.FileInputStream;

java.io.FileNotFoundException;

java.io.FileOutputStream;

java.io.IOException;

java.io.ObjectInputStream;

java.io.ObjectOutputStream;


 

 

import cn.itcast.io.p2.bean.Person;

 

 

public class ObjectStreamDemo {

 

 

/**

 

249


 

 

 

* @param args

* @throws IOException

* @throws ClassNotFoundException

*/

public static void main(String[] args) throws IOException, ClassNotFoundException

{


 

 

//

 

writeObj();

readObj();

}

 

 

public static void readObj() throws IOException, ClassNotFoundException {


 

 

ObjectInputStream  ois

 

=

 

new

 

ObjectInputStream(new


FileInputStream("obj.object"));

//对象的反序列化。

Person p = (Person)ois.readObject();

 

 

System.out.println(p.getName()+":"+p.getAge());

 

 

ois.close();

 

 

}

 

 

public static void writeObj() throws IOException, IOException {


 

 

ObjectOutputStream  oos

 

=

 

new

 

ObjectOutputStream(new


FileOutputStream("obj.object"));

//对象序列化。 被序列化的对象必须实现Serializable接口。

oos.writeObject(new Person("小强",30));

 

oos.close();

}

}


 

 

167.

RandomAccessFile随机访问文件简述:


 

package cn.itcast.io.p3.randomfile;

 

 

import java.io.FileNotFoundException;

import java.io.IOException;

import java.io.RandomAccessFile;

 

 

public class RandomAccessFileDemo {

 

 

/**

* @param args

 

250


 

 

 

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

/*

* RandomAccessFile

* 一看这个类名字,纠结。不是io体系中的子类。

*

* 特点:

* 1,该对象即能读,又能写。

* 2,该对象内部维护了一个byte数组,并通过指针可以操作数组中的元素,

* 3,可以通过getFilePointer方法获取指针的位置,和通过seek方法设置指针的位置。

* 4,其实该对象就是将字节输入流和输出流进行了封装。

* 5,该对象的源或者目的只能是文件。通过构造函数就可以看出。

*/


 

 

//

//

 

writeFile();

readFile();

randomWrite();

}

 

 

public static void randomWrite() throws IOException{

RandomAccessFile raf = new RandomAccessFile("ranacc.txt", "rw");

 

 

//往指定位置写入数据。

raf.seek(3*8);

 

 

raf.write("哈哈".getBytes());

raf.writeInt(108);

 

 

raf.close();

}

 

 

public static void readFile() throws IOException {

 

 

RandomAccessFile raf = new RandomAccessFile("ranacc.txt", "r");

 

 

//通过seek设置指针的位置。

raf.seek(1*8);//随机的读取。只要指定指针的位置即可。

 

byte[] buf = new byte[4];

raf.read(buf);

 

 

String name = new String(buf);

 

 

int age = raf.readInt();

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

251


 

 

 

System.out.println("name="+name);

System.out.println("age="+age);

 

 

System.out.println("pos:"+raf.getFilePointer());

 

 

raf.close();

 

 

}

 

 

//使用RandomAccessFile对象写入一些人员信息,比如姓名和年龄。

public static void writeFile() throws IOException{

/*

* 如果文件不存在,则创建,如果文件存在,不创建

*

*/

RandomAccessFile raf = new RandomAccessFile("ranacc.txt","rw");

 

 

raf.write("张三".getBytes());

raf.writeInt(97);

raf.write("小强".getBytes());

raf.writeInt(99);

//

raf.close();

}

 

 

}


 

 

168.

管道流简述:


 

package cn.itcast.io.p4.piped;

 

 

import java.io.IOException;

import java.io.PipedInputStream;

import java.io.PipedOutputStream;

 

 

public class PipedStream {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

PipedInputStream input = new PipedInputStream();

PipedOutputStream output = new PipedOutputStream();

 

 

input.connect(output);

 

252


 

 

 

 

new Thread(new Input(input)).start();

new Thread(new Output(output)).start();

 

 

}

 

 

}

 

 

class Input implements Runnable{

 

 

private PipedInputStream in;

Input(PipedInputStream in){

this.in = in;

}

public void run(){

 

 

try {

byte[] buf = new byte[1024];

int len = in.read(buf);

 

 

String s = new String(buf,0,len);

 

 

System.out.println("s="+s);

in.close();

} catch (Exception e) {

// TODO: handle exception

}

 

 

}

}

 

 

class Output implements Runnable{

private PipedOutputStream out;

Output(PipedOutputStream out){

this.out = out;

}

public void run(){

 

 

try {

Thread.sleep(5000);

out.write("hi,管道来了!".getBytes());

} catch (Exception e) {

// TODO: handle exception

}

}

}

 

 

253


 

 

 

 

 

169.

 

 

 

 

DataStream数据流简述:


 

package cn.itcast.io.p5.datastream;

 

 

import java.io.DataInputStream;

import java.io.DataOutputStream;

import java.io.FileInputStream;

 

 

import java.io.FileOutputStream;

import java.io.IOException;

 

 

public class DataSteamDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {


 

 

//

 

writeData();

readData();

 

 

}

 

 

public static void readData() throws IOException {

 

 

DataInputStream dis = new DataInputStream(new FileInputStream("data.txt"));

 

 

String str = dis.readUTF();

 

 

System.out.println(str);

}

 

 

public static void writeData() throws IOException {


 

 

DataOutputStream   dos

 

=

 

new

 

DataOutputStream(new


FileOutputStream("data.txt"));

 

 

dos.writeUTF("你好");

 

dos.close();

 

 

}

 

 

}

 

 

254


 

 

 

 

 

170.

 

 

 

 

字节数组流简述:


 

package cn.itcast.io.p6.bytestream;

 

 

import java.io.ByteArrayInputStream;

import java.io.ByteArrayOutputStream;

import java.io.IOException;

 

 

public class ByteArrayStreamDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) {

 

 

ByteArrayInputStream bis = new ByteArrayInputStream("abcedf".getBytes());

 

 

ByteArrayOutputStream bos = new ByteArrayOutputStream();

 

 

int ch = 0;

 

 

while((ch=bis.read())!=-1){

bos.write(ch);

}

 

 

System.out.println(bos.toString());

}

 

 

}


 

 

171.

编解码简述:


 

package cn.itcast.io.p7.encode;

 

 

import java.io.IOException;

import java.io.UnsupportedEncodingException;

 

 

public class EncodeDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

255


 

 

 

/*

* 字符串 --> 字节数组:编码。

* 字节数组 --> 字符串:解码。

*

* 你好:GBK: -60 -29 -70 -61

*

* 你好: utf-8: -28 -67 -96 -27 -91 -67

*

*

* 如果你编错了,解不出来。

* 如果编对了,解错了,有可能有救。

*/

String str = "谢谢";

 

byte[] buf = str.getBytes("gbk");

 

 

String s1 = new String(buf,"UTF-8");

 

 

System.out.println("s1="+s1);

 

 

byte[] buf2 = s1.getBytes("UTF-8");//获取源字节.

 

printBytes(buf2);//-17 -65 -67 -17 -65 -67 -17 -65 -67

//-17 -65 -67 -17 -65 -67 -17 -65 -67 -17 -65 -67

//-48 -69 -48 -69

String s2 = new String(buf2,"GBK");

 

 

System.out.println("s2="+s2);


 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

encodeDemo(str);

 

 

}

 

 

/**

* @param str

* @throws UnsupportedEncodingException

*/

public static void encodeDemo(String str)

throws UnsupportedEncodingException {

//编码;

byte[] buf = str.getBytes("UTF-8");

 

 

printBytes(buf);

 

 

//解码:

String s1 = new String(buf,"UTF-8");

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

256


 

 

 

System.out.println("s1="+s1);

}

 

 

private static void printBytes(byte[] buf) {

for(byte b : buf){

System.out.print(b +" ");

}

}

}


 

 

172.

"联通"问题(绅节):


 

package cn.itcast.io.p7.encode;

 

 

import java.io.IOException;

import java.io.UnsupportedEncodingException;

 

 

public class LianTong {

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

String str = "联通";

/*

11000001

10101010

11001101

10101000

*/

 

 

byte[] buf = str.getBytes("gbk");

 

 

for(byte b :buf){

System.out.println(Integer.toBinaryString(b&255));

}

}

}


 

 

173.

定义方法按最大字节数叏子串(练习):


 

package cn.itcast.io.p7.encode;

 

 

import java.io.IOException;

 

 

 

 

257


 

 

 

public class Test {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

String str = "ab你好cd谢谢";


//

 

//

//

//

x+1));

//

str = "ab琲琲cd琲琲";

 

int len = str.getBytes("gbk").length;

for(int x=0; x

System.out.println(" 截 取 "+(x+1)+" 个 字 节 结 果 是 : "+cutStringByByte(str,

 

}


 

 

int len = str.getBytes("utf-8").length;

for(int x=0; x

System.out.println(" 截 取"+(x+1)+" 个 字 节 结 果 是 :"+cutStringByU8Byte(str,

x+1));

}


 

 

 

 

 

//

//

//

//

//

 

 

 

 

String str = "";

byte[] buf = str.getBytes("gbk");

for(byte b : buf){

System.out.println(b);//-84 105

}


 

 

}

 

 

/*

java中,字符串abcd与字符串ab你好的长度是一样,都是四个字符。

但对应的字节数不同,一个汉字占两个字节。

定义一个方法,按照最大的字节数来取子串。

如:对于ab你好,如果取三个字节,那么子串就是ab字的半个,

那么半个就要舍弃。如果去四个字节就是ab,取五个字节还是ab.

*/

 

 

public static String cutStringByU8Byte(String str, int len) throws IOException

{

 

 

byte[] buf = str.getBytes("utf-8");

 

 

int count = 0;

 

258


 

 

 

for(int x=len-1; x>=0; x--){

if(buf[x]<0)

count++;

else

break;

}

 

 

if(count%3==0)

return new String(buf,0,len,"utf-8");

else if(count%3==1)

return new String(buf,0,len-1,"utf-8");

else

return new String(buf,0,len-2,"utf-8");

 

 

}

 

 

public static String cutStringByByte(String str,int len) throws IOException{

 

 

byte[] buf = str.getBytes("gbk");

int count = 0;

for(int x=len-1; x>=0; x--){

if(buf[x]<0)

count++;

else

break;

}

 

 

if(count%2==0)

return new String(buf,0,len,"gbk");

else

return new String(buf,0,len-1,"gbk");

}

}


 

 

174.

IO精华练习:


 

习题1:

//定义一个比较器

package cn.itcast.io.p8.test;

 

 

import java.util.Comparator;

 

 

public class ComparatorByMath implements Comparator {

 

 

@Override

public int compare(Student o1, Student o2) {

 

 

259


 

 

 

int temp = o1.getMa() - o2.getMa();

 

 

return temp==0?o1.getName().compareTo(o2.getName()):temp;

}

 

 

}

 

 

//习题

package cn.itcast.io.p8.test;


 

 

import

import

import

import

import

 

java.io.File;

java.io.IOException;

java.util.Collections;

java.util.Comparator;

java.util.Set;


 

 

/*

有五个学生,每个学生有3门课的成绩,定义一种比较直观的文本文件格式,

输入学生姓名和成绩,输入的格式:name,30,30,30从键盘输入以上数据(包括姓名,三门课成绩),

按总分数从高到低的顺序将学生信息存放在磁盘文件"stu.txt"中。

 

 

思路:

1,3门课的成绩都是数据,为了便于操作,将其封装到学生对象中。

学生本身就是问题领域中涉及的对象,对学生描述。

学生:

姓名,语文成绩,英语成绩,数学成绩,总分.

2,数据来源于键盘录入,将这些数据都封装到每一个学生对象中。

3,按照总分排个序,很熟,但是这些数据都存储到了学生对象中,其实是学生对象排序。

学生对象很多,先想到的就是存起来--集合-不重复排序-TreeSet。

4,将排序后的信息写入到一个文件中。定义操作文件的输出流。

将信息写入到文件中。

 

aa,10,30,50

zz,30,60,30

qq,30,90,70

cc,70,80,90

pp,80,80,80

*/

 

 

public class StudentInfoTest {

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

Comparator comp = Collections.reverseOrder();

 

260


 

 

 

comp = Collections.reverseOrder(new ComparatorByMath());

Set set = StudentInfoTool.getStudents(comp);

 

 

File file = new File("stuinfo.txt");

StudentInfoTool.write2File(set, file);

}

 

 

}

习题2:

//定义一个学生类

package cn.itcast.io.p8.test;

 

 

public class Student implements Comparable {

 

 

private String name;

private int cn,en,ma;

private int sum;

public Student(String name, int cn, int en, int ma) {

super();

this.name = name;

this.cn = cn;

this.en = en;

this.ma = ma;

sum = cn + en + ma;

}

 

 

@Override

public int hashCode() {

final int prime = 31;

int result = 1;

result = prime * result + ((name == null) ? 0 : name.hashCode());

result = prime * result + sum;

return result;

}

 

 

@Override

public boolean equals(Object obj) {

if (this == obj)

return true;

if (obj == null)

return false;

if (getClass() != obj.getClass())

return false;

Student other = (Student) obj;

if (name == null) {

if (other.name != null)

return false;

 

261


 

 

 

} else if (!name.equals(other.name))

return false;

if (sum != other.sum)

return false;

return true;

}

 

 

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

public int getCn() {

return cn;

}

public void setCn(int cn) {

this.cn = cn;

}

public int getEn() {

return en;

}

public void setEn(int en) {

this.en = en;

}

public int getMa() {

return ma;

}

public void setMa(int ma) {

this.ma = ma;

}

public int getSum() {

return sum;

}

public void setSum(int sum) {

this.sum = sum;

}

 

 

@Override

public int compareTo(Student o) {

 

 

int temp = this.sum-o.sum;

return temp==0?this.name.compareTo(o.name):temp;

}

 

 

@Override

public String toString() {

 

262


 

 

 

 

return "Student:["+name+","+cn+","+en+","+ma+"]";

}

 

 

}

 

 

//习题

package cn.itcast.io.p8.test;


 

 

import

import

import

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.BufferedWriter;

java.io.File;

java.io.FileWriter;

java.io.IOException;

java.io.InputStreamReader;

java.util.Comparator;

java.util.Set;

java.util.TreeSet;


 

 

public class StudentInfoTool {

 

 

/**

* 定义功能,获取键盘录入的信息。 并将信息封装成学生对象。存储到容器中。

* 按照学生的自然排序完成排序动作。

* @return

* @throws IOException

*/

public static Set getStudents() throws IOException {

 

 

return getStudents(null);

}

/**

* 定义功能,获取键盘录入的信息。 并将信息封装成学生对象。存储到容器中。

* 按照指定比较器完成排序的动作。

*

* @throws IOException

*/


public

static

Set getStudents(Comparator

comp)

throws


IOException {

 

 

// 获取键盘录入。

BufferedReader bufr = new BufferedReader(new InputStreamReader(

System.in));

 

 

// 创建一个集合对象。TreeSet.

Set set = null;

 

 

263


 

 

 

if(comp==null)

set = new TreeSet();

else

set = new TreeSet(comp);

String line = null;

 

 

while ((line = bufr.readLine()) != null) {

 

 

if ("over".equals(line))// 定义键盘录入的结束标记。

break;

 

 

// 对获取的信息进行切割,获取指定的数据内容。

String[] info_arr = line.split(",");

Student stu = new Student(info_arr[0],

Integer.parseInt(info_arr[1]),

Integer.parseInt(info_arr[2]),

Integer.parseInt(info_arr[3]));

 

 

// 把学生对象存储到集合中去。

set.add(stu);

}

 

 

return set;

}

/**

* 定义功能,将集合中的对象信息写入到指定文件中进行存储。

*

* @throws IOException

*/

public static void write2File(Set set, File file)

throws IOException {

 

 

BufferedWriter bufw = null;

try {

bufw = new BufferedWriter(new FileWriter(file));

 

 

for (Student stu : set) {

bufw.write(stu.toString() + "\t"+stu.getSum());

bufw.newLine();

bufw.flush();

}

} finally {

if (bufw != null)

bufw.close();

}

}

}

 

264


 

 

 

 

 

175.

 

 

 

 

IO流大总结:


 

IO流:

输入流:

输出流:

 

 

字节流:

字符流:为了处理文字数据方便而出现的对象。

其实这些对象的内部使用的还是字节流(因为文字最终也是字节数据)

只不过,通过字节流读取了相对应的字节数,没有对这些字节直接操作。

而是去查了指定的(本机默认的)编码表,获取到了对应的文字。

简单说:字符流就是 : 字节流+编码表。

--------------------------------------------------------------------------------

缓冲区:提高效率的,提高谁的效率?提高流的操作数据的效率。

所以创建缓冲区之前必须先有流。

缓冲区的基本思想:其实就是定义容器将数据进行临时存储。

对于缓冲区对象,其实就是将这个容器进行了封装,并提供了更多高效的操作方法。

 

 

缓冲区可以提高流的操作效率。

 

 

其实是使用了一种设计思想完成。设计模式:装饰设计模式。

 

Writer

|--TextWriter

|--MediaWriter

现在要对该体系中的对象进行功能的增强。增强的最常见手段就是缓冲区。

先将数据写到缓冲区中,再将缓冲区中的数据一次性写到目的地。

 

 

按照之前学习过的基本的思想,那就是对对象中的写方法进行覆盖。

产生已有的对象子类,复写write方法。不往目的地写,而是往缓冲区写。

 

 

所以这个体系会变成这样。

Writer

|--TextWriter write:往目的地

|--BufferTextWriter write:往缓冲区写

|--MediaWriter

|--BufferMediaWriter

 

 

想要写一些其他数据。就会子类。DataWriter,为了提高其效率,还要创建该类的子类。BufferDataWriter

Writer

|--TextWriter write:往目的地

|--BufferTextWriter write:往缓冲区写

|--MediaWriter

|--BufferMediaWriter

|--DataWriter

|--BufferDataWriter

 

265


 

 

 

 

 

发现这个体系相当的麻烦。每产生一个子类都要有一个高效的子类。

而且这写高效的子类使用的功能原理都一样,都是缓冲区原理。无论数据是什么。

都是通过缓冲区临时存储提高效率的。

那么,对于这个体系就可以进行优化,因为没有必要让每一个对象都具备相同功能的子类。

 

 

哪个对象想要进行效率的提高,只要让缓冲区对其操作即可。也就说,单独将缓冲区进行封装变成对象。

 

 

//它的出现为了提高对象的效率。所以必须在创建它的时候先有需要被提高效率的对象

class BufferWriter

{

[];

BufferedWriter(Writer w)

{

 

 

}

/*

BufferWriter(TextWriter w)

{

 

 

}

BufferedWriter(MediaWriter w)

{

 

 

}

*/

}

BufferWriter的出现增强了Writer中的write方法。

但是增强过后,BufferWriter对外提供的还是write方法。只不过是高效的。

所以写的实质没有变,那么BufferWriter也是Writer中的一员。

所以体系就会变成这样。

Writer

|--TextWriter

|--MediaWriter

|--BufferWriter

|--DataWriter

BufferWriter出现了避免了继承体系关系的臃肿,比继承更为灵活。

如果是为了增强功能,这样方式解决起来更为方便。

所以就把这种优化,总结出来,起个名字:装饰设计模式。

装饰类和被装饰类肯定所属于同一个体系。

 

 

既然明确了BufferedReader由来。

我们也可以独立完成缓冲区的建立

 

 

原理;

1,使用流的read方法从源中读取一批数据存储到缓冲区的数组中。

2,通过计数器记录住存储的元素个数。

 

266


 

 

 

3,通过数组的角标来获取数组中的元素(从缓冲区中取数据).

4,指针会不断的自增,当增到数组长度,会归0.计数器会自减,当减到0时,就在从源拿一批数据进缓冲

区。

 

 

内容补足:

MyBufferedReader

LineNumberReader :可以定义行号。

--------------------------------------------------------------------------------

字符流:

FileReader

FileWriter

 

 

BufferedReader

BufferedWriter

 

 

字节流:

InputStream OutputStream。

 

 

操作文件的字节流对象。

FileOutputStream

FileInputStream

BufferedOutputStream

BufferedInputStream

 

 

字符流和字节流之间的转换动作。

 

--------------------------------------------------------------------------------

转换流:

 

InputStreamReader isr = new InputStreamReader(new FileInputStream("a.txt"));

InputStreamReader isr = new InputStreamReader(new FileInputStream("a.txt"),"gbk");

FileReader fr = new FileReader("a.txt");

 

 

 

FileWriter fw = new FileWriter("b.txt");

OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("b.txt"));


OutputStreamWriter

osw

=   new   OutputStreamWriter(new


FileOutputStream("b.txt"),"gbk");

 

 

转换流:字节流+编码表。

转换流的子类:FileReader,FileWriter:字节流+本地默认码表(GBK)。

 

 

如果操作文本文件使用的本地默认编码表完成编码。可以使用FileReader,或者FileWriter。因为这样写简

便。

如果对操作的文本文件需要使用指定编码表进行编解码操作,这时必须使用转换流来完成。

 

 

 

 

267


 

 

 

--------------------------------------------------------------------------------

IO流的操作规律总结:

 

 

1,明确体系:

数据源:InputStream ,Reader

数据汇:OutputStream,Writer

 

 

2,明确数据:因为数据分两种:字节,字符。

数据源:是否是纯文本数据呢?

是:Reader

否:InputStream

 

 

数据汇:

是:Writer

否:OutputStream

到这里就可以明确具体要使用哪一个体系了。

剩下的就是要明确使用这个体系中的哪个对象。

 

 

3,明确设备:

数据源:

键盘:System.in

硬盘:FileXXX

内存:数组。

网络:socket socket.getInputStream();

 

 

数据汇:

控制台:System.out

硬盘:FileXXX

内存:数组

网络:socket socket.getOutputStream();

 

 

4,明确额外功能:

1,需要转换?是,使用转换流。InputStreamReader OutputStreamWriter

2,需要高效?是,使用缓冲区。Buffered

3,需要其他?

--------------------------------------------------------------------------------

1,复制一个文本文件。

 

 

1,明确体系:

源:InputStream ,Reader

目的:OutputStream ,Writer

2,明确数据:

源:是纯文本吗?是 Reader

目的;是纯文本吗?是 Writer

3,明确设备:

源:硬盘上的一个文件。 FileReader

目的:硬盘上的一个文件。FileWriter

 

268


 

 

 

FileReader fr = new FileReader("a.txt");

FileWriter fw = new FileWriter("b.txt");

4,需要额外功能吗?

需要,高效,使用buffer

BufferedReader bufr = new BufferedReader(new FileReader("a.txt"));

BufferedWriter bufw = new BufferedWriter(new FileWriter("b.txt"));

 

 

 

2,读取键盘录入,将数据存储到一个文件中。

1,明确体系:

源:InputStream ,Reader

目的:OutputStream ,Writer

2,明确数据:

源:是纯文本吗?是 Reader

目的;是纯文本吗?是 Writer

3,明确设备:

源:键盘,System.in

目的:硬盘,FileWriter

InputStream in = System.in;

FileWriter fw = new FileWriter("a.txt");

4,需要额外功能吗?

需要,因为源明确的体系时Reader。可是源的设备是System.in。

所以为了方便于操作文本数据,将源转成字符流。需要转换流。InputStreamReader

InputStreamReader isr = new InputStreamReader(System.in);

FileWriter fw = new FileWriter("a.txt");

需要高效不?需要。Buffer

BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));

BufferedWriter bufw = new BufferedWriter(new FileWriter("a.txt"));

 

 

 

3,读取一个文本文件,将数据展现在控制台上。

1,明确体系:

源:InputStream ,Reader

目的:OutputStream ,Writer

2,明确数据:

源:是纯文本吗?是 Reader

目的;是纯文本吗?是 Writer

3,明确设备:

源:硬盘文件,FileReader。

目的:控制台:System.out。

FileReader fr = new FileReader("a.txt");

OutputStream out = System.out;

4,需要额外功能?

因为源是文本数据,确定是Writer体系。所以为了方便操作字符数据,

需要使用字符流,但是目的又是一个字节输出流。

需要一个转换流,OutputStreamWriter

FileReader fr = new FileReader("a.txt");

 

269


 

 

 

OutputStreamWriter osw = new OutputStreamWriter(System.out);

 

 

需要高效吗?需要。

BufferedReader bufr = new BufferedReader(new FileReader("a.txt"));


BufferedWriter

bufw

=   new   BufferedWriter(new


OutputStreamWriter(System.out));

 

 

 

4,读取键盘录入,将数据展现在控制台上。

1,明确体系:

源:InputStream ,Reader

目的:OutputStream ,Writer

2,明确数据:

源:是纯文本吗?是 Reader

目的;是纯文本吗?是 Writer

3,明确设备:

源:键盘:System.in

目的:控制台:System.out

InputStream in = System.in;

OutputStream out = System.out;

4,需要额外功能吗?

因为处理的数据是文本数据,同时确定是字符流体系。

为方便操作字符数据的可以将源和目的都转成字符流。使用转换流。

为了提高效率,使用Buffer


BufferedReader

bufr

=new

BufferedReader(new


InputStreamReader(Systme.in));


BufferedWriter

bufw

=   new

BufferedWriter(new


OutputStreamWriter(System.out));

 

 

 

5,读取一个文本文件,将文件按照指定的编码表UTF-8进行存储,保存到另一个文件中。

1,明确体系:

源:InputStream ,Reader

目的:OutputStream ,Writer

2,明确数据:

源:是纯文本吗?是 Reader

目的;是纯文本吗?是 Writer

 

 

3,明确设备:

源:硬盘:FileReader.

目的:硬盘:FileWriter

 

FileReader fr = new FileReader("a.txt");

FileWriter fw = new FileWriter("b.txt");

4,额外功能:

注意:目的中虽然是一个文件,但是需要指定编码表。

而直接操作文本文件的FileWriter本身内置的是本地默认码表。无法明确具体指定码表。

 

270


 

 

 

这时就需要转换功能。OutputStreamWriter,而这个转换流需要接受一个字节输出流,而且

对应的目的是一个文件。这时就使用字节输出流中的操作文件的流对象。FileOutputStream.

FileReader fr = new FileReader("a.txt");

OutputStreamWriter osw = new OutputStreamWriter(new

FileOutputStream("b.txt"),"UTF-8");

 

 

需要高效吗?

BufferedReader bufr = new BufferedReader(new FileReader("a.txt"));

BufferedWriter bufw =

new BufferedWriter(new OutputStreamWriter(new

FileOutputStream("b.txt"),"UTF-8"));

 

 

 

目前为止,10个流对象重点掌握。

字符流:

FileReader

FileWriter

 

 

BufferedReader

BufferedWriter

 

 

InputStreamReader

OutputStreamWrier

字节流:

 

FileInputStream

FileOutputStream

 

 

BufferedInputStream

BufferedOutputStream

--------------------------------------------------------------------------------

File类:

用于将文件和文件夹封装成对象。

 

 

1,创建。

boolean createNewFile():如果该文件不存在,会创建,如果已存在,则不创建。不会像输出流

一样会覆盖。

boolean mkdir();

boolean mkdirs();

2,删除。

boolean delete();

void deleteOnExit();

 

 

3,获取:

String getAbsolutePath();

String getPath();

String getParent();

 

271


 

 

 

String getName();

long length();

long lastModified();

 

 

 

4,判断:

boolean exists();

boolean isFile();

boolean isDirectory();

 

 

 

5,

--------------------------------------------------------------------------------

IO中的其他功能流对象:

 

 

1,打印流:

PrintStream:字节打印流。

特点:

1,构造函数接收File对象,字符串路径,字节输出流。意味着打印目的可以有很多。

2,该对象具备特有的方法 打印方法 print println,可以打印任何类型的数据。

3,特有的print方法可以保持任意类型数据表现形式的原样性,将数据输出到目的地。

对于OutputStream父类中的write,是将数据的最低字节写出去。

 

 

PrintWriter:字符打印流。

特点:

1,当操作的数据是字符时,可以选择PrintWriter,比PrintStream要方便。

2,它的构造函数可以接收 File对象,字符串路径,字节输出流,字符输出流。

3,构造函数中,如果参数是输出流,那么可以通过指定另一个参数true完成自动刷新,该true对

println方法有效。

 

 

什么时候用?

当需要保证数据表现的原样性时,就可以使用打印流的打印方法来完成,这样更为方便。

保证原样性的原理:其实就是将数据变成字符串,在进行写入操作。

 

 

 

SequenceInputStream:

特点:

1,将多个字节读取流和并成一个读取流,将多个源合并成一个源,操作起来方便。

2,需要的枚举接口可以通过Collections.enumeration(collection);

 

 

 

ObjectInputStream 和 ObjectOutputStream

 

 

对象的序列化和反序列化。

 

writeObject readObject

 

 

272


 

 

 

Serializable标记接口

 

 

关键字:transient

 

RandomAccessFile:

特点:

1,即可读取,又可以写入。

2,内部维护了一个大型的byte数组,通过对数组的操作完成读取和写入。

3,通过getFilePointer方法获取指针的位置,还可以通过seek方法设置指针的位置。

4,该对象的内容应该封装了字节输入流和字节输出流。

5,该对象只能操作文件。

 

 

通过seek方法操作指针,可以从这个数组中的任意位置上进行读和写

可以完成对数据的修改。

但是要注意:数据必须有规律。

 

 

 

管道流:需要和多线程技术相结合的流对象。

PipedOutputStream

PipedInputStream

 

 

用操作基本数据类型值的对象:

DataInputStream

DataOutputStream

 

 

设备是内存的流对象。

ByteArrayInputStream ByteArrayOutputStream

CharArrayReader CharArrayWriter

--------------------------------------------------------------------------------

IO流体系:

 

 

字符流:

Reader

|--BufferedReader:

|--LineNumberReader

|--CharArrayReader

|--StringReader

|--InputStreamReaer

|--FileReader

 

 

Writer

|--BufferedWriter

|--CharArrayWriter

|--StringWriter

|--OutputStreamWriter

|--FileWriter

|--PrintWriter

 

273


 

 

 

 

 

字节流:

InputStream

|--FileInputStream:

|--FilterInputStream

|--BufferedInputStream

|--DataInputStream

|--ByteArrayInputStream

|--ObjectInputStream

|--SequenceInputStream

|--PipedInputStream

 

 

OutputStream

|--FileOutputStream

|--FilterOutputStream

|--BufferedOutputStream

|--DataOutputStream

|--ByteArrayOutputStream

|--ObjectOutputStream

|--PipedOutputStream

|--PrintStream

 

第九章:GUI编程


 

 

176.

GUI简单演示:


 

package cn.itcast.gui.p1.awt;


 

 

import

import

import

import

import

import

import

 

java.awt.Button;

java.awt.FlowLayout;

java.awt.Frame;

java.awt.event.ActionEvent;

java.awt.event.ActionListener;

java.awt.event.WindowAdapter;

java.awt.event.WindowEvent;


 

 

public class FrameDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

Frame f = new Frame("my frame");


//

//

f.setSize(500, 400);

f.setLocation(400, 200);

 

 

 

274


 

 

 

f.setBounds(400, 200, 500, 400);

f.setLayout(new FlowLayout());//设置流式布局

 

 

Button but = new Button("一个按钮");

 

 

f.add(but);//将按钮添加到窗体中。

 

f.addWindowListener(new WindowAdapter() {

 

 

@Override

public void windowClosing(WindowEvent e) {


 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

System.out.println("closing......."+e);

System.exit(0);

}

 

 

});

 

 

//在按钮上加上一个监听。

but.addActionListener(new ActionListener() {

 

 

@Override

public void actionPerformed(ActionEvent e) {

System.out.println("button run .....");


System.exit(0);

}

});

 

 

f.setVisible(true);

System.out.println("over");

}

 

 

}


 

 

177.

事件监听机制及使用案例:


 

package cn.itcast.gui.p1.awt;


 

 

import

import

import

import

import

import

import

import

import

 

java.awt.Button;

java.awt.FlowLayout;

java.awt.Frame;

java.awt.TextField;

java.awt.event.ActionEvent;

java.awt.event.ActionListener;

java.awt.event.KeyAdapter;

java.awt.event.KeyEvent;

java.awt.event.MouseAdapter;

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

275


 

 

 

import java.awt.event.MouseEvent;

import java.awt.event.WindowAdapter;

import java.awt.event.WindowEvent;

 

 

public class MouseAndKeyDemo {

 

 

private Frame f;

private TextField tf;

private Button but;

 

 

public MouseAndKeyDemo() {

init();

 

 

}

 

 

private void init() {

 

 

f = new Frame("演示鼠标和键盘监听");

f.setBounds(400,200,500,400);

f.setLayout(new FlowLayout());

 

 

tf = new TextField(35);

but = new Button("一个按钮");

 

f.add(tf);

f.add(but);

 

 

myEvent();

 

 

f.setVisible(true);

 

 

}

 

 

private void myEvent() {

 

 

//给文本框添加键盘监听。

tf.addKeyListener(new KeyAdapter() {

 

 

@Override

public void keyPressed(KeyEvent e) {


 

 

//

 

System.out.println("key


run..."+KeyEvent.getKeyText(e.getKeyCode())+"::::"+e.getKeyCode());


//

//

//

//

int code = e.getKeyCode();

if(!(code>=KeyEvent.VK_0 && code<=KeyEvent.VK_9)){

System.out.println("必须是数字");

e.consume();

 

 

 

 

 

 

276


 

 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

//

 

 

//

 

 

 

 

 

 

 

 

 

 

 

//

//

 

 

 

}

 

 

if(e.isControlDown() && e.getKeyCode()==KeyEvent.VK_ENTER){

System.out.println("enter run ...");

}

}

 

 

});

 

 

f.addWindowListener(new WindowAdapter() {

 

 

@Override

public void windowClosing(WindowEvent e) {

 

 

System.exit(0);

}

 

 

});

 

 

but.addActionListener(new ActionListener() {

 

 

@Override

public void actionPerformed(ActionEvent e) {

System.out.println("action run.....");

}

});

 

 

//在按钮上添加一个鼠标监听.

but.addMouseListener(new MouseAdapter() {

 

 

private int count = 1;

@Override

public void mouseEntered(MouseEvent e) {

 

 

System.out.println("mouse enter..."+count++);

 

 

tf.setText("mouse enter..."+count++);

 

 

}

@Override

public void mouseClicked(MouseEvent e) {

 

 

if(e.getClickCount()==2)

tf.setText("mouse double click..."+count++);

System.out.println("mouse click..."+count++);

System.out.println(e);

}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

277


 

 

 

 

});

}

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

new MouseAndKeyDemo();

}

 

 

}

 

第十章:网络编程


 

 

178.

IP简述使用:


 

package cn.itcast.net.p1.ip;

 

 

import java.net.InetAddress;

import java.net.UnknownHostException;

 

 

public class IPDemo {

/**

* @param args

* @throws UnknownHostException

*/

public static void main(String[] args) throws UnknownHostException {

 

 

//获取本地主机ip地址对象。

InetAddress ip = InetAddress.getLocalHost();

 

 

//获取其他主机的ip地址对象。

ip = InetAddress.getByName("192.168.1.110");

//InetAddress.getByName("my_think");

 

 

System.out.println(ip.getHostAddress());

System.out.println(ip.getHostName());

}

}


 

 

179.

使用UDP建立信息发送端:


 

package cn.itcast.net.p2.udp;

 

 

278


 

 

 

import

import

import

import

import

import

 

 

 

java.io.IOException;

java.net.DatagramPacket;

java.net.DatagramSocket;

java.net.InetAddress;

java.net.SocketException;

java.net.UnknownHostException;


 

 

public class UDPSendDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

System.out.println("发送端启动......");

/*

* 创建UDP传输的发送端。

* 思路:

* 1,建立udp的socket服务。

* 2,将要发送的数据封装到数据包中。

* 3,通过udp的socket服务将数据包发送出去。

* 4,关闭socket服务。

*/

//1,udpsocket服务。使用DatagramSocket对象。

DatagramSocket ds = new DatagramSocket(8888);

 

 

//2,将要发送的数据封装到数据包中。

String str = "udp传输演示:哥们来了!";

//使用DatagramPacket将数据封装到的该对象包中。

byte[] buf = str.getBytes();

 

 

DatagramPacket dp =

new

DatagramPacket(buf,buf.length,InetAddress.getByName("192.168.1.100"),10000);

 

 

//3,通过udp的socket服务将数据包发送出去。使用send方法。

ds.send(dp);

 

 

//4,关闭资源。

ds.close();

}

}


 

 

180.

使用UDP建立信息接收端:


 

package cn.itcast.net.p2.udp;

 

279


 

 

 

 

import java.io.IOException;

import java.net.DatagramPacket;

import java.net.DatagramSocket;

 

 

public class UDPReceDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

System.out.println("接收端启动......");

/*

* 建立UDP接收端的思路。

* 1,建立udp socket服务,因为是要接收数据,必须要明确一个端口号。

* 2,创建数据包,用于存储接收到的数据。方便用数据包对象的方法解析这些数据.

* 3,使用socket服务的receive方法将接收的数据存储到数据包中。

* 4,通过数据包的方法解析数据包中的数据。

* 5,关闭资源

*/

 

 

//1,建立udp socket服务。

DatagramSocket ds = new DatagramSocket(10000);

 

 

//2,创建数据包。

byte[] buf = new byte[1024];

DatagramPacket dp = new DatagramPacket(buf,buf.length);

 

 

//3,使用接收方法将数据存储到数据包中。

ds.receive(dp);//阻塞式的。

 

 

//4,通过数据包对象的方法,解析其中的数据,比如,地址,端口,数据内容。

String ip = dp.getAddress().getHostAddress();

int port = dp.getPort();

String text = new String(dp.getData(),0,dp.getLength());

 

 

System.out.println(ip+":"+port+":"+text);

 

 

//5,关闭资源。

ds.close();

}

}


 

 

181.

使用UDP建立信息交互方式:

 

 

 

 

280


 

 

 

(信息发送端)

package cn.itcast.net.p2.udp;


 

 

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.IOException;

java.io.InputStreamReader;

java.net.DatagramPacket;

java.net.DatagramSocket;

java.net.InetAddress;


 

 

public class UDPSendDemo2 {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

System.out.println("发送端启动......");

/*

* 创建UDP传输的发送端。

* 思路:

* 1,建立udp的socket服务。

* 2,将要发送的数据封装到数据包中。

* 3,通过udp的socket服务将数据包发送出去。

* 4,关闭socket服务。

*/

//1,udpsocket服务。使用DatagramSocket对象。

DatagramSocket ds = new DatagramSocket(8888);


 

 

 

//

 

 

String str = "udp传输演示:哥们来了!";


BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));

String line = null;

 

 

while((line=bufr.readLine())!=null){

 

 

 

byte[] buf = line.getBytes();

DatagramPacket dp =

new

DatagramPacket(buf,buf.length,InetAddress.getByName("192.168.1.100"),10000);

ds.send(dp);

 

 

if("886".equals(line))

break;

}

 

281


 

 

 

 

 

//4,关闭资源。

ds.close();

 

 

}

 

 

}

(信息接收端)

package cn.itcast.net.p2.udp;

 

 

import java.io.IOException;

import java.net.DatagramPacket;

import java.net.DatagramSocket;

 

 

public class UDPReceDemo2 {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

System.out.println("接收端启动......");

/*

* 建立UDP接收端的思路。

* 1,建立udp socket服务,因为是要接收数据,必须要明确一个端口号。

* 2,创建数据包,用于存储接收到的数据。方便用数据包对象的方法解析这些数据.

* 3,使用socket服务的receive方法将接收的数据存储到数据包中。

* 4,通过数据包的方法解析数据包中的数据。

* 5,关闭资源

*/

 

 

//1,建立udp socket服务。

DatagramSocket ds = new DatagramSocket(10000);

 

 

while(true){

 

 

//2,创建数据包。

byte[] buf = new byte[1024];

DatagramPacket dp = new DatagramPacket(buf,buf.length);

 

 

//3,使用接收方法将数据存储到数据包中。

ds.receive(dp);//阻塞式的。

 

 

//4,通过数据包对象的方法,解析其中的数据,比如,地址,端口,数据内容。

String ip = dp.getAddress().getHostAddress();

int port = dp.getPort();

 

282


 

 

 

String text = new String(dp.getData(),0,dp.getLength());

 

 

System.out.println(ip+":"+port+":"+text);

 

 

}

//5,关闭资源。


//

 

 

 

 

 

}

 

 

}

ds.close();


 

 

182.

聊天程序(多线程):


 

(发送端)

package cn.itcast.net.p3.chat;


 

 

import

import

import

import

import

 

java.io.BufferedReader;

java.io.InputStreamReader;

java.net.DatagramPacket;

java.net.DatagramSocket;

java.net.InetAddress;


 

 

public class Send implements Runnable {

 

 

private DatagramSocket ds;

 

 

public Send(DatagramSocket ds){

this.ds = ds;

}

 

 

@Override

public void run() {

 

 

try {


BufferedReader

bufr

=

new

BufferedReader(new


InputStreamReader(System.in));

String line = null;

 

 

while((line=bufr.readLine())!=null){

 

 

 

byte[] buf = line.getBytes();

DatagramPacket dp =

new

DatagramPacket(buf,buf.length,InetAddress.getByName("192.168.1.255"),10001);

ds.send(dp);

 

283


 

 

 

 

if("886".equals(line))

break;

}

ds.close();

} catch (Exception e) {

}

}

}

 

 

(接收端)

package cn.itcast.net.p3.chat;

 

 

import java.net.DatagramPacket;

import java.net.DatagramSocket;

 

 

public class Rece implements Runnable {

 

 

private DatagramSocket ds;

 

 

public Rece(DatagramSocket ds) {

this.ds = ds;

}

 

 

@Override

public void run() {

try {

while (true) {

 

 

// 2,创建数据包。

byte[] buf = new byte[1024];

DatagramPacket dp = new DatagramPacket(buf, buf.length);

 

 

// 3,使用接收方法将数据存储到数据包中。

ds.receive(dp);// 阻塞式的。

 

 

// 4,通过数据包对象的方法,解析其中的数据,比如,地址,端口,数据内容。

String ip = dp.getAddress().getHostAddress();

int port = dp.getPort();

String text = new String(dp.getData(), 0, dp.getLength());

 

 

System.out.println(ip + "::" + text);

if(text.equals("886")){

System.out.println(ip+"....退出聊天室");

}

}

} catch (Exception e) {

 

284


 

 

 

 

}

}

}

 

 

(开启发送和接收两个线程开始运行聊天)

package cn.itcast.net.p3.chat;

 

 

import java.io.IOException;

import java.net.DatagramSocket;

import java.net.SocketException         ;

 

 

public class ChatDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

DatagramSocket send = new DatagramSocket();

 

 

DatagramSocket rece = new DatagramSocket(10001);

new Thread(new Send(send)).start();

new Thread(new Rece(rece)).start();

}

}


 

 

183.

使用TCP建立客户端:


 

package cn.itcast.net.p4.tcp;


 

 

import

import

import

import

 

java.io.IOException;

java.io.OutputStream;

java.net.Socket;

java.net.UnknownHostException;


 

 

public class ClientDemo {

 

 

/**

* @param args

* @throws IOException

* @throws UnknownHostException

*/

public static void main(String[] args) throws UnknownHostException, IOException

{

//客户端发数据到服务端

 

285


 

 

 

/*

* Tcp传输,客户端建立的过程。

* 1,创建tcp客户端socket服务。使用的是Socket对象。


*

建议该对象一创建就明确目的地。要连接的主机。


* 2,如果连接建立成功,说明数据传输通道已建立。


*

*

*

该通道就是socket流 ,是底层建立好的。 既然是流,说明这里既有输入,又有输出。

想要输入或者输出流对象,可以找Socket来获取。

可以通过getOutputStream(),和getInputStream()来获取两个字节流。


* 3,使用输出流,将数据写出。

* 4,关闭资源。

*/

//创建客户端socket服务。

Socket socket = new Socket("192.168.1.100",10002);

 

 

//获取socket流中的输出流。

OutputStream out = socket.getOutputStream();

 

 

//使用输出流将指定的数据写出去。

out.write("tcp演示:哥们又来了!".getBytes());

 

 

//关闭资源。

socket.close();

 

 

}

}


 

 

184.

使用TCP建立服务端:


 

package cn.itcast.net.p4.tcp;


 

 

import

import

import

import

import

 

java.io.IOException;

java.io.InputStream;

java.io.OutputStream;

java.net.ServerSocket;

java.net.Socket;


 

 

public class ServerDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {


//

服务端接收客户端发送过来的数据,并打印在控制台上。

/*

* 建立tcp服务端的思路:

* 1,创建服务端socket服务。通过ServerSocket对象。

 

 

 

 

 

 

286


 

 

 

* 2,服务端必须对外提供一个端口,否则客户端无法连接。

* 3,获取连接过来的客户端对象。

* 4,通过客户端对象获取socket流读取客户端发来的数据


*

并打印在控制台上。


* 5,关闭资源。关客户端,关服务端。

*/

 

 

//1创建服务端对象。

ServerSocket ss = new ServerSocket(10002);

 

 

//2,获取连接过来的客户端对象。

Socket s = ss.accept();//阻塞式.

 

String ip = s.getInetAddress().getHostAddress();

 

 

//3,通过socket对象获取输入流,要读取客户端发来的数据

InputStream in = s.getInputStream();

 

 

byte[] buf = new byte[1024];

 

 

int len = in.read(buf);

String text = new String(buf,0,len);

System.out.println(ip+":"+text);

 

 

s.close();

ss.close();

}

}


 

 

185.

使用TCP建立交互方式:


 

(客户端)

package cn.itcast.net.p4.tcp;


 

 

import

import

import

import

import

 

java.io.IOException;

java.io.InputStream;

java.io.OutputStream;

java.net.Socket;

java.net.UnknownHostException;


 

 

public class ClientDemo2 {

 

 

/**

* @param args

* @throws IOException

* @throws UnknownHostException

*/

 

287


 

 

 

public static void main(String[] args) throws UnknownHostException, IOException

{

//客户端发数据到服务端

/*

* Tcp传输,客户端建立的过程。

* 1,创建tcp客户端socket服务。使用的是Socket对象。


*

建议该对象一创建就明确目的地。要连接的主机。


* 2,如果连接建立成功,说明数据传输通道已建立。


*

*

*

该通道就是socket流 ,是底层建立好的。 既然是流,说明这里既有输入,又有输出。

想要输入或者输出流对象,可以找Socket来获取。

可以通过getOutputStream(),和getInputStream()来获取两个字节流。


* 3,使用输出流,将数据写出。

* 4,关闭资源。

*/

 

 

Socket socket = new Socket("192.168.1.100",10002);

 

 

OutputStream out = socket.getOutputStream();

 

 

out.write("tcp演示:哥们又来了!".getBytes());

 

 

//读取服务端返回的数据,使用socket读取流。

InputStream in = socket.getInputStream();

 

 

byte[] buf = new byte[1024];

 

 

int len = in.read(buf);

 

 

String text = new String(buf,0,len);

 

 

System.out.println(text);

//关闭资源。

socket.close();

}

}

 

 

(服务端)

package cn.itcast.net.p4.tcp;


 

 

import

import

import

import

import

 

java.io.IOException;

java.io.InputStream;

java.io.OutputStream;

java.net.ServerSocket;

java.net.Socket;


 

 

public class ServerDemo2 {

 

 

288


 

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {


//

服务端接收客户端发送过来的数据,并打印在控制台上。

/*

* 建立tcp服务端的思路:

* 1,创建服务端socket服务。通过ServerSocket对象。

* 2,服务端必须对外提供一个端口,否则客户端无法连接。

* 3,获取连接过来的客户端对象。

* 4,通过客户端对象获取socket流读取客户端发来的数据


*

并打印在控制台上。


* 5,关闭资源。关客户端,关服务端。

*/

 

 

//1创建服务端对象。

ServerSocket ss = new ServerSocket(10002);

 

 

//2,获取连接过来的客户端对象。

Socket s = ss.accept();

 

 

String ip = s.getInetAddress().getHostAddress();

 

 

//3,通过socket对象获取输入流,要读取客户端发来的数据

InputStream in = s.getInputStream();

 

 

byte[] buf = new byte[1024];

 

 

int len = in.read(buf);

String text = new String(buf,0,len);

System.out.println(ip+":"+text);

 

 

//使用客户端socket对象的输出流给客户端返回数据

OutputStream out = s.getOutputStream();

out.write("收到".getBytes());

 

s.close();

ss.close();

}

}


 

 

186.

TCP创建一个英文大写服务器(练习):


 

(需求与分析)

package cn.itcast.net.p5.tcptest;

 

 

289


 

 

 

public class TextTransTest {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

/*

* 客户端输入字母数据,发送给服务端,

* 服务端收到后显示在控制台,并将该数据转成大写返回给客户端

* 直到客户端输入over.转换结束.

*

* 创建一个英文大写转换服务器.

*

* 分析:

* 有客户端和服务端,使用tcp传输

*/

}

}

 

 

(客户端)

package cn.itcast.net.p5.tcptest;


 

 

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.IOException;

java.io.InputStreamReader;

java.io.PrintWriter;

java.net.Socket;

java.net.UnknownHostException;


 

 

public class TransClient {

 

 

/**

* @param args

* @throws IOException

* @throws UnknownHostException

*/

public static void main(String[] args) throws UnknownHostException, IOException

{

 

 

/*


*

*

*

*

*

*

思路:

客户端:

1,需要先有socket端点。

2,客户端的数据源:键盘。

3,客户端的目的:socket.

4,接收服务端的数据,源:socket。

 

 

 

 

 

 

 

 

 

290


 

 

 

* 5,将数据显示在打印出来:目的:控制台.

* 6,在这些流中操作的数据,都是文本数据。

* 转换客户端:

* 1,创建socket客户端对象。

* 2,获取键盘录入。

* 3,将录入的信息发送给socket输出流。

*/

 

 

//1,创建socket客户端对象。

Socket s = new Socket("192.168.1.100",10004);

 

 

//2,获取键盘录入。

BufferedReader bufr =

new BufferedReader(new InputStreamReader(System.in));

 

 

//3,socket输出流。


//

new BufferedWriter(new OutputStreamWriter(s.getOutputStream()));

PrintWriter out = new PrintWriter(s.getOutputStream(),true);

 

 

//4,socket输入流,读取服务端返回的大写数据


BufferedReader  bufIn

=

new

BufferedReader(new


InputStreamReader(s.getInputStream()));

 

 

String line = null;

 

 

while((line=bufr.readLine())!=null){

 

 

if("over".equals(line))

break;


 

 

//

//

 

out.print(line+"\r\n");

out.flush();


out.println(line);

 

 

//读取服务端发回的一行大写数。

String upperStr = bufIn.readLine();

System.out.println(upperStr);

}

 

 

s.close();

}

}

 

 

(服务端)

package cn.itcast.net.p5.tcptest;

 

 

import java.io.BufferedReader;

 

291


 

 

 

import

import

import

import

import

 

 

 

java.io.IOException;

java.io.InputStreamReader;

java.io.PrintWriter;

java.net.ServerSocket;

java.net.Socket;


 

 

public class TransServer {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

/*

* 转换服务端。

* 分析:

* 1,serversocket服务。

* 2,获取socket对象。

* 3,源:socket,读取客户端发过来的需要转换的数据。

* 4,目的:显示在控制台上。

* 5,将数据转成大写发给客户端。

*/

 

 

//1,

ServerSocket ss = new ServerSocket(10004);

 

 

//2,获取socket对象。

Socket s = ss.accept();

 

 

//获取ip.

String ip = s.getInetAddress().getHostAddress();

System.out.println(ip+"......connected");

 

 

//3,获取socket读取流,并装饰。


BufferedReader   bufIn

=

new

BufferedReader(new


InputStreamReader(s.getInputStream()));

 

 

//4,获取socket的输出流,并装饰。

PrintWriter out = new PrintWriter(s.getOutputStream(),true);

 

 

String line = null;

while((line=bufIn.readLine())!=null){

 

 

System.out.println(line);

out.println(line.toUpperCase());


//

out.print(line.toUpperCase()+"\r\n");

 

292


 

 

 

//

 

 

 

out.flush();


}

 

 

s.close();

ss.close();

}

}


 

 

187.

上传的客户端不服务端:


 

(客户端)

package cn.itcast.net.p6.uploadtext;


 

 

import

import

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.File;

java.io.FileReader;

java.io.IOException;

java.io.InputStreamReader;

java.io.PrintWriter;

java.net.Socket;

java.net.UnknownHostException;


 

 

public class UploadClient {

 

 

/**

* @param args

* @throws IOException

* @throws UnknownHostException

*/

public static void main(String[] args) throws UnknownHostException, IOException

{

 

 

System.out.println("上传客户端。。。。。。");

 

File file = new File("c:\\client.txt");

System.out.println(file.exists());

 

 

//1

Socket s = new Socket("192.168.1.100",10005);

 

 

//2

BufferedReader bufr =

new BufferedReader(new FileReader(file));

 

 

//3,

PrintWriter out = new PrintWriter(s.getOutputStream(),true);

 

 

293


 

 

 

String line = null;

while((line=bufr.readLine())!=null){

 

 

out.println(line);

}

 

 

//告诉服务端,客户端写完了。

s.shutdownOutput();


//

out.println("!@#$%^&*(");

 

 

//4,


BufferedReader   bufIn

=

new

BufferedReader(new


InputStreamReader(s.getInputStream()));

 

 

String str = bufIn.readLine();

System.out.println(str);

 

 

bufr.close();

s.close();

}

}

 

 

(服务端)

package cn.itcast.net.p6.uploadtext;


 

 

import

import

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.BufferedWriter;

java.io.FileWriter;

java.io.IOException;

java.io.InputStreamReader;

java.io.PrintWriter;

java.net.ServerSocket;

java.net.Socket;


 

 

public class UploadServer {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

System.out.println("上传服务端。。。。。。。。。");

 

//1

ServerSocket ss = new ServerSocket(10005);

 

 

//2,

 

294


 

 

 

Socket s = ss.accept();

 

 

System.out.println(s.getInetAddress().getHostAddress()+".....connected");

 

 

//3,


BufferedReader   bufIn

=

new

BufferedReader(new


InputStreamReader(s.getInputStream()));

 

 

//4,

BufferedWriter bufw = new BufferedWriter(new FileWriter("c:\\server.txt"));

 

 

String line = null;

 

 

while((line=bufIn.readLine())!=null){


 

 

//

//

 

if("over".equals(line))

break;


bufw.write(line);

bufw.newLine();

bufw.flush();

}

 

 

PrintWriter out = new PrintWriter(s.getOutputStream(),true);

out.println("上传成功");

 

bufw.close();

 

 

s.close();

ss.close();

}

}


 

 

188.

上传图片的设计案例:


 

(上传的任务)

package cn.itcast.net.p1.uploadpic;


 

 

import

import

import

import

import

import

 

java.io.File;

java.io.FileOutputStream;

java.io.IOException;

java.io.InputStream;

java.io.OutputStream;

java.net.Socket;


 

 

public class UploadTask implements Runnable {

 

 

private static final int SIZE = 1024*1024*2;

 

295


 

 

 

private Socket s;

 

 

public UploadTask(Socket s) {

this.s = s;

}

 

 

@Override

public void run() {

 

 

int count = 0;

String ip = s.getInetAddress().getHostAddress();

System.out.println(ip + ".....connected");

 

 

try{

 

 

// 读取客户端发来的数据。

InputStream in = s.getInputStream();

 

 

// 将读取到数据存储到一个文件中。

File dir = new File("c:\\pic");

if (!dir.exists()) {

dir.mkdirs();

}

File file = new File(dir, ip + ".jpg");

//如果文件已经存在于服务端

while(file.exists()){

file = new File(dir,ip+"("+(++count)+").jpg");

}

 

 

 

FileOutputStream fos = new FileOutputStream(file);

 

 

byte[] buf = new byte[1024];

 

 

int len = 0;

 

 

while ((len = in.read(buf)) != -1) {

 

 

fos.write(buf, 0, len);

 

 

if(file.length()>SIZE){

System.out.println(ip+"文件体积过大");

 

fos.close();

s.close();

 

 

 

 

296


 

 

 

System.out.println(ip+"...."+file.delete());

 

 

return ;

}

}

 

 

// 获取socket输出流,将上传成功字样发给客户端。

OutputStream out = s.getOutputStream();

out.write("上传成功".getBytes());

 

fos.close();

s.close();

}catch(IOException e){

 

 

}

}

}

(上传的客户端)

package cn.itcast.net.p1.uploadpic;


 

 

import

import

import

import

import

import

 

java.io.FileInputStream;

java.io.IOException;

java.io.InputStream;

java.io.OutputStream;

java.net.Socket;

java.net.UnknownHostException;


 

 

public class UploadPicClient {

 

 

/**

* @param args

* @throws IOException

* @throws UnknownHostException

*/

public static void main(String[] args) throws UnknownHostException, IOException

{

 

 

//1,创建客户端socket。

Socket s = new Socket("192.168.1.100",10006);

 

 

//2,读取客户端要上传的图片文件。

FileInputStream fis = new FileInputStream("c:\\0.bmp");

 

 

//3,获取socket输出流,将读到图片数据发送给服务端。

OutputStream out = s.getOutputStream();

 

 

byte[] buf = new byte[1024];

 

297


 

 

 

 

int len = 0;

 

 

while((len=fis.read(buf))!=-1){

out.write(buf,0,len);

}

 

 

//告诉服务端说:这边的数据发送完毕。让服务端停止读取。

s.shutdownOutput();

 

 

//读取服务端发回的内容。

InputStream in = s.getInputStream();

byte[] bufIn = new byte[1024];

 

 

int lenIn = in.read(buf);

String text = new String(buf,0,lenIn);

System.out.println(text);

 

 

fis.close();

s.close();

}

}

(上传的服务端)

package cn.itcast.net.p1.uploadpic;


 

 

import

import

import

import

import

import

import

 

java.io.File;

java.io.FileOutputStream;

java.io.IOException;

java.io.InputStream;

java.io.OutputStream;

java.net.ServerSocket;

java.net.Socket;


 

 

public class UploadPicServer {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

//创建tcp的socket服务端。

ServerSocket ss = new ServerSocket(10006);

 

 

while(true){

Socket s = ss.accept();

 

 

298


 

 

 

new Thread(new UploadTask(s)).start();

}

//获取客户端。

 

// ss.close();

}

}


 

 

189.

模拟IE浏览器:


 

package cn.itcast.net.p2.ie_server;


 

 

import

import

import

import

import

 

java.io.IOException;

java.io.InputStream;

java.io.PrintWriter;

java.net.Socket;

java.net.UnknownHostException;


 

 

public class MyBrowser {

 

 

/**

* @param args

* @throws IOException

* @throws UnknownHostException

*/

public static void main(String[] args) throws UnknownHostException, IOException

{

 

 

Socket s = new Socket("192.168.1.100",8080);

 

 

//模拟浏览器,给tomcat服务端发送符合http协议的请求消息。

PrintWriter out = new PrintWriter(s.getOutputStream(),true);

out.println("GET /myweb/1.html HTTP/1.1");

out.println("Accept: */*");

out.println("Host: 192.168.1.100:8080");

out.println("Connection: close");

out.println();

out.println();

 

 

InputStream in = s.getInputStream();

 

 

byte[] buf = new byte[1024];

int len = in.read(buf);

 

 

String str =new String(buf,0,len);

System.out.println(str);

 

 

299


 

 

 

s.close();

 

 

//http://192.168.1.100:8080/myweb/1.html

}

}


 

 

190.

模拟Tomcat服务器:


 

package cn.itcast.net.p2.ie_server;


 

 

import

import

import

import

import

 

java.io.IOException;

java.io.InputStream;

java.io.PrintWriter;

java.net.ServerSocket;

java.net.Socket;


 

 

public class MyTomcat {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

ServerSocket ss = new ServerSocket(9090);

 

 

Socket s = ss.accept();

System.out.println(s.getInetAddress().getHostAddress()+".....connected");

 

 

InputStream in = s.getInputStream();

 

 

byte[] buf = new byte[1024];

 

 

int len = in.read(buf);

String text = new String(buf,0,len);

System.out.println(text);

 

 

//给客户端一个反馈信息。

PrintWriter out = new PrintWriter(s.getOutputStream(),true);

 

 

out.println("欢迎光临");

 

s.close();

ss.close();

}

}

 

 

300


 

 

 

 

 

191.

 

 

 

 

URL使用演示:


 

package cn.itcast.net.p2.ie_server;


 

 

import

import

import

import

 

java.io.IOException;

java.io.InputStream;

java.net.URL;

java.net.URLConnection;


 

 

public class URLDemo {

 

 

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

String str_url = "http://192.168.1.100:8080/myweb/1.html";

 

 

URL url = new URL(str_url);


 

 

//

//

//

//

//

//

 

 

//

 

 

 

 

 

 

 

 

//

//

 

 

//

 

System.out.println("getProtocol:"+url.getProtocol());

System.out.println("getHost:"+url.getHost());

System.out.println("getPort:"+url.getPort());

System.out.println("getFile:"+url.getFile());

System.out.println("getPath:"+url.getPath());

System.out.println("getQuery:"+url.getQuery());

 

 

InputStream in = url.openStream();

 

 

//获取url对象的Url连接器对象。将连接封装成了对象:java中内置的可以解析的具体协议的对象

+socket.

URLConnection conn = url.openConnection();

 

 

String value = conn.getHeaderField("Content-Type");

System.out.println(value);

 

 

System.out.println(conn);

 

 

//sun.net.www.protocol.http.HttpURLConnection:http://192.168.1.100:8080/myweb

/1.html

 

 

InputStream in = conn.getInputStream();

 

 

byte[] buf = new byte[1024];

int len = in.read(buf);

 

301


 

 

 

 

String text = new String(buf,0,len);

 

 

System.out.println(text);

 

 

in.close();

}

}


 

 

192.

网络编程小结:


 

最常见的客户端:

浏览器 :IE。

最常见的服务端:

服务器:Tomcat。

为了了解其原理:

自定义服务端,使用已有的客户端IE,了解一下客户端给服务端发了什么请求?

发送的请求是:


GET / HTTP/1.1 请求行 请求方式 /myweb/1.html 请求的资源路径

请求消息头 . 属性名:属性值

http协议版本。


Accept: image/gif,

image/x-xbitmap,

image/jpeg,

image/pjpeg,


application/x-shockwave-flash,

application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, */*

Accept: */*

Accept-Language: zh-cn,zu;q=0.5

Accept-Encoding: gzip, deflate

User-Agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; InfoPath.2)

Host: 192.168.1.100:9090

//Host: www.huyouni.com:9090

Connection: Keep-Alive

//空行

//请求体。

 

 

 

//服务端发回应答消息。


HTTP/1.1 200 OK //应答行,http的协议版本

应答状态码

应答状态描述信息


应答消息属性信息。 属性名:属性值

Server: Apache-Coyote/1.1

ETag: W/"199-1323480176984"

Last-Modified: Sat, 10 Dec 2011 01:22:56 GMT

Content-Type: text/html

Content-Length: 199

Date: Fri, 11 May 2012 07:51:39 GMT

Connection: close

//空行

//应答体。

 

302


 

 

 

这是我的网页

欢迎光临

这是一个tomcat服务器中的资源。是一个html网页。

--------------------------------------------------------------------------------

网络结构,

1,C/S client/server

特点:

该结构的软件,客户端和服务端都需要编写。

可发成本较高,维护较为麻烦。

 

 

好处:

客户端在本地可以分担一部分运算。

2,B/S browser/server

特点:

该结构的软件,只开发服务器端,不开发客户端,因为客户端直接由浏览器取代。

开发成本相对低,维护更为简单。

缺点:所有运算都要在服务端完成。

 

第十一章:反射机制


 

 

193.

反射简述及三种获叏Class对象的方法:


 

package cn.itcast.reflect.demo;

 

 

import cn.itcast.bean.Person;

/*

* JAVA反射机制是在运行状态中,对于任意一个类 (class文件),都能够知道这个类的所有属性和方法;

* 对于任意一个对象,都能够调用它的任意一个方法和属性;

* 这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

*

*

* 动态获取类中信息,就是java反射 。

* 可以理解为对类的解剖。

*

*

* 要想要对字节码文件进行解剖,必须要有字节码文件对象.

* 如何获取字节码文件对象呢?

*

*/

public class ReflectDemo {

 

 

303


 

 

 

/**

* @param args

* @throws ClassNotFoundException

*/

public static void main(String[] args) throws ClassNotFoundException {

 

 

getClassObject_3();

 

 

}

/*

* 方式三:

* 只要通过给定的类的 字符串名称就可以获取该类,更为扩展。

* 可是用Class类中的方法完成。

* 该方法就是forName.

* 这种方式只要有名称即可,更为方便,扩展性更强。

*/

public static void getClassObject_3() throws ClassNotFoundException {

 

 

String className = "cn.itcast.bean.Person";

 

 

Class clazz = Class.forName(className);

 

 

System.out.println(clazz);

}

/*

* 方式二:

* 2,任何数据类型都具备一个静态的属性.class来获取其对应的Class对象。

* 相对简单,但是还是要明确用到类中的静态成员。

* 还是不够扩展。

*

*/

public static void getClassObject_2() {

 

 

Class clazz = Person.class;

 

 

Class clazz1 = Person.class;

System.out.println(clazz==clazz1);

}

/*

* 获取字节码对象的方式:

* 1Object类中的getClass()方法的。

* 想要用这种方式,必须要明确具体的类,并创建对象。

* 麻烦 .

*

*/

public static void getClassObject_1(){

 

 

304


 

 

 

Person p = new Person();

Class clazz = p.getClass();

 

 

Person p1 = new Person();

Class clazz1 = p1.getClass();

 

 

System.out.println(clazz==clazz1);

}

}


 

 

194.

利用反射获叏构造函数:


 

package cn.itcast.reflect.demo;

 

 

import java.io.FileReader;

import java.lang.reflect.Constructor;

 

 

public class ReflectDemo2 {

 

 

/**

* @param args

* @throws Exception

* @throws InstantiationException

* @throws ClassNotFoundException

*/


public static void main(String[]

args)

throws

ClassNotFoundException,


InstantiationException, Exception {

 

 

createNewObject_2();

 

 

}

 

 

public static void createNewObject_2() throws Exception {


 

 

//

 

cn.itcast.bean.Person p = new cn.itcast.bean.Person("小强",39);

 

/*

* 当获取指定名称对应类中的所体现的对象时,

* 而该对象初始化不使用空参数构造该怎么办呢?

* 既然是通过指定的构造 函数进行对象的初始化,

* 所以应该先获取到该构造函数。 通过字节码文件对象即可完成。

* 该方法是:getConstructor(paramterTypes);

*

*/

String name = "cn.itcast.bean.Person";

//找寻该名称类文件,并加载进内存,并产生Class对象。

Class clazz = Class.forName(name);

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

305


 

 

 

//获取到了指定的构造函数对 象。

Constructor constructor = clazz.getConstructor(String.class,int.class);

 

 

//通过该构造器对象的newInstance方法进行对象的初始化。

Object obj = constructor.newInstance("小明",38);

}


 

 

public static void createNewObject()

 

throws

 

ClassNotFoundException,


InstantiationException, IllegalAccessException{

 

 

//早期:new时候,先根据被new的类的名称找寻该类的字节码文件,并加载进内存,

// 并创建该字节码文件对象,并接着创建该字节文件的对应的Person对象.

// cn.itcast.bean.Person p = new cn.itcast.bean.Person();

//现在:

String name = "cn.itcast.bean.Person";

//找寻该名称类文件,并加载进内存,并产生Class对象。

Class clazz = Class.forName(name);

//如何产生该类的对象呢?

Object obj = clazz.newInstance();

}

}


 

 

195.

利用反射获叏字段:


 

package cn.itcast.reflect.demo;

 

 

import java.lang.reflect.Field;

 

 

public class ReflectDemo3 {

 

 

/**

* @param args

* @throws Exception

*/

public static void main(String[] args) throws Exception {

 

 

getFieldDemo();

}

/*

* 获取字节码文件中的字段。

*/

public static void getFieldDemo() throws Exception {

 

 

Class clazz = Class.forName("cn.itcast.bean.Person");

 

 

Field field = null;//clazz.getField("age");//只能获取公有的,

 

 

306


 

 

 

field = clazz.getDeclaredField("age");//只获取本类,但包含私有。

 

 

//对私有字段的访问取消权限检查。暴力访问。

field.setAccessible(true);

 

 

Object obj = clazz.newInstance();

 

 

field.set(obj, 89);

 

 

Object o = field.get(obj);

 

 

System.out.println(o);


 

 

//

//

 

 

}

 

 

 

 

}

 

cn.itcast.bean.Person p = new cn.itcast.bean.Person();

p.age = 30;


 

 

196.

利用反射获叏Class中的公共函数:


 

package cn.itcast.reflect.demo;

 

 

import java.lang.reflect.Constructor;

import java.lang.reflect.Method;

 

 

public class ReflectDemo4 {

 

 

public ReflectDemo4() {

}

 

 

/**

* @param args

* @throws Exception

*/

public static void main(String[] args) throws Exception {

 

 

getMethodDemo_3();

 

 

}

 

 

public static void getMethodDemo_3() throws Exception {

 

 

Class clazz = Class.forName("cn.itcast.bean.Person");

 

 

Method method = clazz.getMethod("paramMethod", String.class,int.class);

 

 

Object obj = clazz.newInstance();

 

307


 

 

 

 

 

method.invoke(obj, "小强",89);

 

}

 

 

public static void getMethodDemo_2() throws Exception {

 

 

Class clazz = Class.forName("cn.itcast.bean.Person");

 

 

Method method = clazz.getMethod("show", null);//获取空参数一般方法。


 

//

Object obj = clazz.newInstance();


Constructor constructor = clazz.getConstructor(String.class,int.class);

Object obj = constructor.newInstance("小明",37);

 

method.invoke(obj, null);

 

 

}

/*

* 获取指定Class中的所有公共函数。

*/

public static void getMethodDemo() throws Exception {

 

 

Class clazz = Class.forName("cn.itcast.bean.Person");

 

 

Method[] methods = clazz.getMethods();//获取的都是公有的方法。

methods = clazz.getDeclaredMethods();//只获取本类中所有方法,包含私有。

for(Method method : methods){

System.out.println(method);

}

}

}

 

第十二章:正则表达式


 

 

197.

正则表达式简述:


 

package cn.itcast.regex.demo;

/*

* 正则表达式。

*

* 正则表达式用于操作字符串数据。

* 通过一些特定的符号来体现的。

* 所以我们为了掌握正则表达式,必须要学习一些符号。

*

* 虽然简化了,但是阅读性差。

308


 

 

 

*/

public class RegexDemo {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

String qq = "123k4567";


//

 

 

 

 

 

//

//

 

 

//

//

//

//

checkQQ(qq);

 

 

String regex = "[1-9][0-9]{4,14}";//正则表达式。

 

boolean b = qq.matches(regex);

System.out.println(qq+":"+b);

 

 

String str = "aoooooooob";

String reg = "ao{4,6}b";

boolean b = str.matches(reg);

System.out.println(str+":"+b);

}

/*

* 需求:定义一个功能对QQ号进行校验。

* 要求:长度5~15. 只能是数字, 0不能开头

*/

public static void checkQQ(String qq){

 

 

int len = qq.length();

 

 

if(len>=5 && len<=15){

 

 

if(!qq.startsWith("0")){

try {

long l = Long.parseLong(qq);

 

 

System.out.println(l+":正确");

}catch(NumberFormatException e){

System.out.println(qq+":含有非法字符");

}

 

 

}else{

System.out.println(qq+":不能0开头");

}

}else{

System.out.println(qq+":长度错误");

}

}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

309


 

 

 

}


 

 

198.

正则表达式中常用操作方法演示:


 

package cn.itcast.regex.demo;

 

 

import java.util.regex.Matcher;

import java.util.regex.Pattern;

 

 

public class RegexDemo2 {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

/*

* 正则表达式对字符串的常见操作:

* 1, 匹配。


*

其实使用的就是String类中的matches方法。


*

* 2,切割。


*

其实使用的就是String类中的split方法。


*

* 3,替换。


*

其实使用的就是String类中的replaceAll()方法。


*

* 4,获取。

*

*/

functionDemo_4();

}

/*

* 获取

* 将正则规则进行对象的封装。

* Pattern p = Pattern.compile("a*b");

* //通过正则对象的matcher方法字符串相关联。获取要对字符串操作的匹配器对象Matcher .

* Matcher m = p.matcher("aaaaab");

* //通过Matcher匹配器对象的方法对字符串进行操作。

* boolean b = m.matches();

*/

public static void functionDemo_4() {

 

 

String str = "da jia hao,ming tian bu fang jia!";

 

 

String regex = "\\b[a-z]{3}\\b";

 

 

310


 

 

 

//1,将正则封装成对象。

Pattern p = Pattern.compile(regex);

//2, 通过正则对象获取匹配器对象。

Matcher m = p.matcher(str);

 

 

//使用Matcher对象的方法对字符串进行操作。

//既然要获取三个字母组成的单词

//查找。 find();

System.out.println(str);

while(m.find()){

System.out.println(m.group());//获取匹配的子序列

 

System.out.println(m.start()+":"+m.end());

}

}

/*

* 替换

*/

public static void functionDemo_3() {

 

 

String str = "zhangsanttttxiaoqiangmmmmmmzhaoliu";

 

 

str = str.replaceAll("(.)\\1+", "$1");

 

 

System.out.println(str);

 

 

String tel = "15800001111";//158****1111;

 

 

tel = tel.replaceAll("(\\d{3})\\d{4}(\\d{4})", "$1****$2");

 

 

System.out.println(tel);

}

/*

* 切割。

*

* 组:((A)(B(C)))

*/

public static void functionDemo_2(){

 

 

String str = "zhangsanttttxiaoqiangmmmmmmzhaoliu";

 

 

String[] names = str.split("(.)\\1+");//str.split("\\.");

 

 

for(String name : names){

System.out.println(name);

}

}

 

311


 

 

 

/*

* 演示匹配。

*/

public static void functionDemo_1(){

//匹配手机号码是否正确。

String tel = "15800001111";

String regex = "1[358]\\d{9}";

boolean b = tel.matches(regex);

System.out.println(tel+":"+b);

}

}


 

 

199.

正则表达式常用练习:


 

package cn.itcast.regex.test;

 

 

import java.util.TreeSet;

 

 

public class RegexTest {

 

 

/**

* @param args

*/

public static void main(String[] args) {

 

 

/*

* 1,治疗口吃:我我...我我...我我我要...要要要要...要要要要..学学学学学...学学编编...


*

编编编编....程程...程程...程程程


* 2,对ip地址排序。

* 3,对邮件地址校验。

*/

test_3();

}

//对邮件地址校验。

public static void test_3() {

 

 

String mail = "[email protected]";

 

 

String regex = "[a-zA-Z0-9_]+@[a-zA-Z0-9]+(\\.[a-zA-Z]{1,3})+";

 

 

regex = "\\w+@\\w+(\\.\\w+)+";//[email protected]

 

 

boolean b = mail.matches(regex);

 

 

System.out.println(mail+":"+b);

}

/*

 

312


 

 

 

* 1,治口吃。

*/

public static void test_1(){

 

 

String str = "我我...我我...我我我要...要要要要...要要要要..学学学学学...学学编编...

编编编编....程程...程程...程程程";

 

 

//1,将字符串中.去掉。 用替换。

str = str.replaceAll("\\.+", "");

System.out.println(str);

 

 

//2,替换叠词。

str = str.replaceAll("(.)\\1+", "$1");

System.out.println(str);

}

/*

* ip地址排序。

*

* 192.168.10.34 127.0.0.1 3.3.3.3 105.70.11.55

*/

public static void test_2(){

 

 

String ip_str = "192.168.10.34 127.0.0.1 3.3.3.3 105.70.11.55";

//1,为了让ip可以按照字符串顺序比较,只要让ip的每一段的位数相同。

//所以,补零,按照每一位所需做多0进行补充。每一段都加两个0.

 

ip_str = ip_str.replaceAll("(\\d+)", "00$1");

System.out.println(ip_str);

 

 

//然后每一段保留数字3位。

ip_str = ip_str.replaceAll("0*(\\d{3})", "$1");

System.out.println(ip_str);

 

 

//1,将ip地址切出。

String[] ips = ip_str.split(" +");

 

 

TreeSet ts = new TreeSet();

 

 

for(String ip : ips){


//

 

 

 

 

 

 

 

 

 

 

}

System.out.println(ip);

ts.add(ip);

}

 

 

for(String ip : ts){

System.out.println(ip.replaceAll("0*(\\d+)", "$1"));

}

 

 

 

 

 

 

 

 

 

 

 

 

313


 

 

 

}


 

 

200.

网络爬虫:


 

package cn.itcast.regex.test;


 

 

import

import

import

import

import

import

import

import

import

 

java.io.BufferedReader;

java.io.FileReader;

java.io.IOException;

java.io.InputStreamReader;

java.net.URL;

java.util.ArrayList;

java.util.List;

java.util.regex.Matcher;

java.util.regex.Pattern;


/*

* 网页爬虫:其实就一个程序用于在互联网中获取符合指定规则的数据。

* 爬取邮箱地址。

*/

public class RegexTest2 {

/**

* @param args

* @throws IOException

*/

public static void main(String[] args) throws IOException {

 

 

 

List list = getMailsByWeb();

 

 

for(String mail : list){

System.out.println(mail);

}

}

 

 

public static List getMailsByWeb() throws IOException {

 

 

//1,读取源文件。


//

BufferedReader

bufr

=

new

BufferedReader(new


FileReader("c:\\mail.html"));

 

 

URL url = new URL("http://192.168.1.100:8080/myweb/mail.html");


 

 

BufferedReader   bufIn

 

=

 

new

 

BufferedReader(new


InputStreamReader(url.openStream()));

 

 

//2,对读取的数据进行规则的匹配。从中获取符合规则的数据.

String mail_regex = "\\w+@\\w+(\\.\\w+)+";

 

314


 

 

 

 

List list = new ArrayList();

 

 

Pattern p = Pattern.compile(mail_regex);

 

 

String line = null;

 

 

while((line=bufIn.readLine())!=null){

Matcher m = p.matcher(line);

while(m.find()){

//3,将符合规则的数据存储到集合中。

list.add(m.group());

}

}

return list;

}

 

 

public static List getMails() throws IOException{

 

 

//1,读取源文件。

BufferedReader bufr = new BufferedReader(new FileReader("c:\\mail.html"));

 

 

//2,对读取的数据进行规则的匹配。从中获取符合规则的数据.

String mail_regex = "\\w+@\\w+(\\.\\w+)+";

 

 

List list = new ArrayList();

 

 

Pattern p = Pattern.compile(mail_regex);

 

 

String line = null;

 

 

while((line=bufr.readLine())!=null){

 

 

Matcher m = p.matcher(line);

while(m.find()){

//3,将符合规则的数据存储到集合中。

list.add(m.group());

}

}

return list;

}

}

 

第十三章:HTML语言

 

 

 

 

315


 

 

 

 

 

201.

 

 

 

 

列表标签演示:


 

<html>

<head>

<meta http-equiv="Content-Type" content="text/html; charset=GBK">

<title>Untitled Documenttitle>

head>

<body>

 

 

<dl>

<dt>上层项目内容dt>

<dd>下层项目内容dd>

<dd>下层项目内容dd>

<dd>下层项目内容dd>

dl>

 

 

<hr/>

<ul type="square">

<li>无序项目列表li>

<li>无序项目列表li>

<li>无序项目列表li>

<li>无序项目列表li>

ul>

 

 

<ol type="a">

<li>有序的项目列表li>

<li>有序的项目列表li>

<li>有序的项目列表li>

<li>有序的项目列表li>

ol>

 

 

316


 

 

 

body>

html>


 

 

202.

图片标签演示:


 

<html>

<head>

<meta http-equiv="Content-Type" content="text/html; charset=GBK">

<title>Untitled Documenttitle>

head>

<body>

 

 

<img src="imgs\1.jpg" height=350 width=500 border=10 alt="啊,美女!" />

body>

html>


 

 

203.

表格标签演示:


 

<html>

<head>

<meta http-equiv="Content-Type" content="text/html; charset=GBK">

<title>Untitled Documenttitle>

head>

<body>

 

 

<table border=1 bordercolor="#0000EE" cellpadding=10 cellspacing=0 width=500>

<tbody>

<tr>

<th rowspan=2>个人信息th>

<td>张三td>

tr>

<tr>

<td>30td>

tr>

tbody>

table>

 

 

 

<hr/>

 

 

<table border=1 bordercolor="#0000EE" cellpadding=10 cellspacing=0 width=500>

<tr>

<th colspan=2>个人信息th>

tr>

 

317


 

 

 

<tr>

<td>张三td>

<td>30td>

tr>

table>

<hr/>

<table border=1 bordercolor="#0000EE" cellpadding=10 cellspacing=0 width=500>

<caption>表格标题caption>

<tr>

<th>姓名:th>

<th>年龄:th>

tr>

<tr>

<td>张三td>

<td>39td>

tr>

table>

 

 

body>

html>


 

 

204.

超链接标签演示:


 

演示1:

<html>

<head>

<meta http-equiv="Content-Type" content="text/html; charset=GBK">

<title>Untitled Documenttitle>

head>

<body>

<a href="http://www.sohu.com.cn" target="_blank">新浪网站a>

<hr/>

<a href="imgs/1.jpg">美女图片a>

<hr/>

<a href="mailto:[email protected]">联系我们a>

<hr/>

 

 

<a href="http://www.xunlei.com/movies/fczlm.rmvb">复仇者联盟a><br/>

 

318


 

 

 

<a href="thunder://wertyuioasdfghjklwertyuio==">复仇者联盟a>

 

 

<a href="javascript:void(0)" onclick="alert('我弹')">这是一个超链接a>

 

body>

html>

注意:取消默认点击超链接效果方法是javascript:void(0).

演示2:

<html>

<head>

<meta http-equiv="Content-Type" content="text/html; charset=GBK">

<title>Untitled Documenttitle>

head>

<body>

 

 

<a name=top>顶部位置a>

<hr/>

<img src="111.jpg" height=900 width=400 border=10/>

<hr/>

<a name=center>中间位置a>

<hr/>

 

 

<img src="111.jpg" height=900 width=400 border=10/>

<a href="#top">回到顶部位置a>

<a href="#center">回到中间位置a>

 

body>

html>


 

 

205.

框架标签演示:


 

<html>

<head>

<meta http-equiv="Content-Type" content="text/html; charset=GBK">

<title>Untitled Documenttitle>

head>

 

 

 

 

<frameset rows="30%,*">

<frame src="top.html" name="top" />

 

319


 

 

 

 

 

 

<frameset cols="30%,*">

<frame src="left.html" name="left" />

<frame src="right.html" name="right"/>

frameset>

 

 

 

frameset>

 

 

<body>

 

 

body>

html>

 

 

左边:

<html>

<head>

<meta http-equiv="Content-Type" content="text/html; charset=GBK">

<title>Untitled Documenttitle>

head>

<body>

<H3>左边栏

你可能感兴趣的:(jishu,http协议,脚本语言)