异常
异常分类
运行时的异常RuntimeException
可以不处理,交给虚拟机处理。
编译的异常Exception
必须要处理。
异常产生的过程
JVM会根据产生的原因创建一个异常对象,异常对象包括了(内容,原因,位置
但是在方法之中没有try-catch,JVM会把对象给调用此方法的main去处理
main方收到对象,但是main方法没有异常的处理逻辑,所以会继续把对象抛出给main方法的调用者JVM来处理。
throws关键字
作用:
当方法内部出现异常的时候,用throws丢出给方法的调用着处理(自己不处理),最终是交给JVM处理——中断处理
注意:
throws必须写在方法的声明处
抛出的异常是Exception或者是其子类
如果丢出了,我们必须处理,用try处理,如果无就给JVM中断处理。
如果抛出了编译异常,必须处理这个异常,也可以在方法体后面再次声明,再次抛出,最后抛到主函数(主函数也能继续抛出)。
try-catch
catch(异常类)里面接收的参数,子类一定要父类的最后,最后一个用Exception异常来捕捉其他异常。
3个处理异常的方法
String getMessage() 返回此throwable的简单描述
String toString() 返回详细的字符串
void printStackTrace () //堆栈跟踪,JVM打印异常对象,最为全面
finally代码块
无论是否异常都会执行
因为性质1,finally一般用于资源的释放和回收。
继承时父子抛出异常注意
父类抛出了多个异常,子类重写父类方法时候,抛出的异常必须小于等于父类(不能比父类大,可以不抛出)
情况:
子类重写父类的方法时候,抛出的异常相同。
子类重写父类的方法时候,抛出父类的异常的子类。
父类不抛出异常,子类也不能抛出异常
自定义异常
注意:
一般以xxException结尾,说明是一个异常类
必须继承Exception或者 RuntimeException
继承Exception 那么就是编译的异常,必须处理,要么throws,要么try...catch
继承RuntimeException,是运行期异常,交给JVM中断处理。
多线程
并发与并行:
并发:交替执行,一个人吃两个馒头
并行:同时发生,两个人吃两个馒头
进程与线程
进程:硬盘进入到内存的程序(内存:临时存储RAM)
线程:属于进程,是进程之中的执行单元,一个进程之中至少有一个线程。
线程调度
分时调度
所有线程轮流使用,时间平均
抢占式调度
优先级高,cup用的时间多。多线程不能提高程序的运行速度,但是可以提高程序的运行效率,让cpu使用的效率更高。
主线程
JVM会执行main方法,main方法会进入栈内存,JAVM会找操作系统开辟一条main方法通向cpu的执行路径,cpu可以通过这个路径执行main方法,这个路径有个名字就叫main(主线程)线程
总结:就是栈内存和cpu之间的路线
Thread
调用Thread的run方法,并且此线程不能重复执行。
多线程的原理
方法:在栈里执行
new对象:在堆里
多线程:每次新线程的增加就会开辟一个新的栈空间,执行run方法,对于cpu可以随便执行任意一个方法。
内存原理:
Thread常用的方法
设置线程名称:setName 或者 父类构造
获取线程名称getName
返回一个线程的 静态方法currentThread
start
sleep(ms)暂停线程,其本身就有InterruptedException,用try或者 throw处理
Runnable接口实现方法
实现Runnable 接口 重写Run方法
在主线程创建Runnable的实现类
将Runnable作为参数传到Thread类里面
用Thread的类的run方法调用Runnable
Runnable创建多线程接口的好处(尽量使用Runnable)
避免单继承的局限性
增加了程序的拓展性,降低了程序的耦合性,它把设置线程任务和开启新的线程进行了分离,实现类之中,我们重写了run方法设置线程任务,创建Thread对象,调用start的方法调用。传递不同的实现类,实现不同的需求
线程安全
概念
多线程访问了共享的数据,会产生线程的安全问题。比如三个人在一起卖票,就会出现负数票的情况
同步代码块
同步代码块:synchronized关键字可以用于方法中的某个区块之中,表示只对这个区块的资源实行互斥访问,当有线程已经在执行这块代码了,其他线程会进行等待。
synchronized(同步锁){
可能出现线程安全问题的代码
}
原理:多个线程一起抢cpu的执行权,谁抢到了就会执行run方法,当遇到synchronized代码块这时会检查synchronized代码块是否有锁对象,有的话,就获取锁对象进入代码块。当其他线程运行时,发现没有锁对象,因此会进入阻塞状态,直到其他线程归还锁对象。
总结:
1.同步中没有执行完毕不会释放锁对象,没有锁对象进不来。
2.此外,锁对象一定要唯一,但是程序频繁判断锁,程序效率会有所降低。
同步方法
同步方法:将同步关键字加在函数声明上,直接调用即可。
原理:实际上也是同步代码快,代码块的锁对象就是this(对象)本身。
静态同步方法
唯一值得一提的是,锁对象不是this,因为this是创建对象的时候生成的,而静态方法优先于对象,锁对象是本类的class属性——class文件对象(反射),实际上就是。
synchronized(xxx类.class)
{
}
使用方法:和同步方法一样
Lock锁
Lock锁:Lock实现提供比使用synchronized方法和语句可以获得的更广泛的锁定操作
使用方法
利用多态写法,创建Lock的实现类ReentrantLock。
在可能出现安全问题的代码之前加lock();
代码结束之后用unlock();
注意
在try语句的里,将unlock放在finally里,这样无论程序是否都出现异常都可以把锁释放。
线程状态的概述
线程状态
导致状态发生的条件
NEW
刚被创建等待start
Runnable
可以在jvm运行的状态,不一定正在运行,取决操作系统处理器
Waiting
无线等待,需要手动启动
Timed Waiting
计时醒来
Teminated
run方法正常退出而死亡,或者是没有捕获异常JVM终止了run方法而死亡
线程池
利用Executors静态方法获取ExecutorService实现类创建线程个数
利用submit添加线程
shutDown 结束线程
注意 :
没有shutDown销毁线程,程序就不会自己结束,他会一直等待新的线程添加。
线程池使用方法如下:
ExecutorService es= Executors.newFixedThreadPool(5);//创建线程个数
es.submit(new RunnableImpl());//添加线程进去 然后直接执行
es.submit(new RunnableImpl());
//如果没有销毁 程序不会结束
es.shutdown();
Lambda表达式
介绍
Lambda表达式:可以用来代替匿名内部类,在参数里完成接口的实现。
使用条件:接口的需要实现的函数有且仅有一个。
语法规则:(参数列表)->{方法体}
不同情况示例
函数有参
arr.sort((Person o1,Person o2)->{
return o1.age - o2.age;
});
函数无参
new Thread( ()->{
System.out.println("无参Lambda");
}).start();
Lambda的省略形式
参数类型可省
Method(1,2,(a,b)->{
System.out.println(a+b);// 1 2 是 a 和 b , 1 和2 默认是传进去就是int
});
省略括号(void方法体只有一行)
Method(1,2,(a,b)->System.out.println(a+b));
省略括号(非void方法体只有一行,缩写必须是返回值)
arr.sort((o1,o2)->o1.age-o2.age);//简化版Lambda表达式
总结
Lambda格式固定 三段式 ()->{}
省略括号只能写一行,并且如果是有返回值函数会将其的值作为返回值。
File类
File类的静态方法
pathSeparator 系统默认路径分隔符,windows 封号,linux 冒号。
separator系统默认文名称分隔符
值得一提的是,在不同的系统下,文件名称分隔符是不同的,比如C:\Windows\123 需要改写成"C\:windows"+File.separator+"123" ,这样就在不同的系统下正常运作了。
绝对路径与相对路径
绝对路径:以盘符开始的路径
相对路径:默认是当前项目的根目录
File类的构造方法
File(String pathname)
参数注意:
pathname 字符串的路径名称
路径可以是文件/文件夹结尾
路径可以是不存在的路径
File(String parent ,String child )父路径和子路径
父路径和子路经可以单独书写,使用起来更加灵活。
File("C:\\","apple.tXt")= "C:\\apple.txt"
File(File parent ,String child )
路径更加灵活
父路径是File路径,可以使用File类的方法对路径进行操作,再用路径创新对象,直接放入就是合并。
常用方法
获得:
public String getAbsolutePath():获取绝对路径,
public String getPath():获取绝对/相对路径,实际上重写toString就是调用了这个方法
public String getName():获取末尾部分文件/文件夹的名称
public long length()此File表示的文件长度,获取的是构造方法指定文件的大小,以字节为长度,
注意:
如果此文件不存在,那么length返回0。
文件夹是没有大小的概念的,不能获取文件夹的大小。
public Files[] listFiles()
public Files[] listFiles(FileFliter)获取此目录内的所有文件,如果此File是文件,会抛出空指针异常。
判断:
pbulic boolean exists()表示文件是否真实存在
public boolean isDirectory()表示是否为目录
public boolean isFile()表示是否为目录,与上面互斥,并且两个方法如果路径不存在,那么会返回false
操作:
public boolean createNewFile():会先去找文件,如果文件不存在就会创建文件,如果路径不正确就会抛出异常,路径不正确和文件已经存在就会返回false,反之返回true。
public boolean makdir():创建文件夹
public boolean makdirs():创建多级文件夹,路径可以不存在,不存在就会自动创建多个文件夹。
public boolean delete():删除文件或者目录。如果删除的是目录,则目录为空才能删除。
public class FileCreateDelete {
public static void main(String[] args) throws IOException {
File f = new File("aaa.txt");
System.out.println("是否存在:"+f.exists()); // false
System.out.println("是否创建:"+f.createNewFile()); // true
System.out.println("是否存在:"+f.exists()); // true
// 目录的创建
File f2= new File("newDir");
System.out.println("是否存在:"+f2.exists());// false
System.out.println("是否创建:"+f2.mkdir()); // true
System.out.println("是否存在:"+f2.exists());// true// 创建多级目录
File f3= new File("newDira\\newDirb");
System.out.println(f3.mkdir());// false
File f4= new File("newDira\\newDirb");
System.out.println(f4.mkdirs());// true// 文件的删除
System.out.println(f.delete());// true// 目录的删除
System.out.println(f2.delete());// true
System.out.println(f4.delete());// false
}
}
综合案例
文件搜索:
递归实现
public File
//如果是文件夹 就输出里面的所有文件 如果里面的文件是目录就继续递归
public static void show01(File f) {
if(f.isFile()){
if(f.getName().endsWith(".java"))
System.out.println(f.getName());
}
else{
File[] r = f.listFiles();
for(var i:r){
show01(i);
}
}
}
文件过滤器FileFilter
java.io.FileFilter: 是一个接口,是File的过滤器。 该接口的对象可以传递给File类的 listFiles(FileFilter) 作为参数, 接口中只有一个方法。
boolean accept(File pathname):测试pathname是否应该包含在当前File目录中,符合则返回true。
文件过滤器接口的使用
File[] r = f.listFiles(new FileFilter() {
@Override
public boolean accept(File pathname) {
//如果是文件夹或者是我们想要的文件 那么就放进去
return pathname.isDirectory() || pathname.getName().endsWith(".java");
}
});
文件过滤器Lambda表达式实现
File[] r = f.listFiles(file->file.isDirectory()|| file.getName().endsWith(".java"));
简化规则:
只有一个方法需要重写
只有一行return,return可以不写,但同时要把括号去掉
参数类型可以不写
参数只有一个的时候,参数的括号也可以不写
(标准形式): (参数)->{} => (简化形式) 参数->返回值
IO流
字节流
Java中I/O操作主要是指使用java.io 包下的内容,进行输入、输出操作。输入也叫做读取数据,输出也叫做作写 出数据。
OutputStream
OutputStream:此抽象类是所有输出字节流的超类。
常用方法:
``
FileOutputStream
FileOutputStream:文件字节输出流
作用:内存数据写入到硬盘的文件中
构造方法:
FileOutputStream(String name)
FileOutputStream(File name)
参数:写入数据的目的地
File:目的地是一个文件
String:目的地是文件路径
字节输出流使用的步骤:
创建FileOutputStream对象,构造方法中传递写入数据的目的地
调用write(byte[])把数据写入到文件中
释放资源
追加写数据:
FileOutputStream(String name,boolean append)
FileOutputStream(File name,boolean append)
append为true就是代表不新建文件,即可以追加写数据。
换行:
windows :\r\n
linux:/n
amac:/r
write("\r\n".getbytes())
字节输入流
InputStream字节输入流
InputStream,此抽象类为所有输入流的超类
方法:
FileInputStream文件字节输入流
FileInputStream
作用:
硬盘中的文件读取到内存中
构造方法:
FileInputStream(String name)
FileInputStream(File file)
字节输入流的原理(硬盘->内存)
java程序->JVM->OS->OS调用读取数据的方法->读取文件到内存
方法
int read()
读取文件中的一个字节并返回,读取到文件末尾返回-1
循环读取:
while((le=file.read())!=-1)
int read(bytes[])
while((le=file.read(bytes))!=-1){
String s=new String(bytes,0,le)//这样可以避免放入过多的空格
}
字节流练习——文件复制
FileInputStream file=new FileInputStream("img\\ji.jpg");
FileOutputStream file2=new FileOutputStream("img\\jiCopy.jpg",true);
byte []bytes=new byte[1024];
int le=0;
while((le=file.read(bytes))!=-1){
file2.write(bytes,0,le);
}
System.out.println("复制成功");
file.close();
file2.close();
思路:复制A文件到B,边读取边写即可。注意关闭流,释放资源。
字符流
Reader
字符输入流,是字符输入流最顶层的超类,是一个抽象类。
常用方法
read(char[])
read()//读取一个字符并返
FileReader字符输入流
作用:读取文件的字符流
构造方法:
FileReader(String fileName)
FileReader(File file)
优点:相比于字节流,能够更好的读取中文,byte存储的范围是0~127 而字符char能够存储2个字节,能够存储0~65535,比如一个中文的字符编码为263,那么在byte显示下是负数,因此单个字节无法读取中文。
Writer字符输出流
Witer是最顶层的超类,是抽象,
方法:
write(String s)
write(char[] buf)
write(int i)写入单个字符
FileWriter文件字符输出流
使用方法:
构造,用write写入缓冲区,最后用flash/close,数据从内存中写到硬盘之中。如果char数组大于8186,那么就会自动写入硬盘之中,所以不用flash,最后记得close即可。
利用try catch处理字符输出流的异常
在日常开发之中我们需要声明字符输出流,和close,但是我们不能总是抛出异常让jvm处理异常。下面分别是
普通处理
JDK7特性
JDK9特性处理方式
FileWriter f = null;//首先给它附上初始值
try{
f=new FileWriter("123.txt");
f.write("123");
} catch (IOException e ){
System.out.println(e);
}finally{//在finally用来资源释放
if(null!=f) {//解决f可能是空指针的情况
try {
f.close();//close还有可能有异常,需要再次处理
} catch (Exception e) {
System.out.println(e);
}
}
}
//JDK7 优化 在try()里面声明流对象,出了try的作用域会自动释放掉
try(FileWriter file=new FileWriter("123.txt")){
}catch (IOException e ){
System.out.println(e);
}
//JDK9优化 可以把外面申明的流放进
FileWriter file1=new FileWriter("123.txt");
FileWriter file2=new FileWriter("123.txt");
try(file1;file2){
file1.write("12");
file2.write("12");
}catch (IOException e ){
System.out.println(e);
}
Properties属性集
Properties:是唯一一个和IO流相结合的集合。
网络编程
基础入门
软件结构
C/S结构 Client/Server
B/S结构 Browser/Server
网络通信协议
网络通信协议:计算机在网络通信之中需要遵守的一定的规则
TCP/IP协议:传输控制协议/因特网互联协议,结构为4层的分层模型。
应用层:应用程序的协议
传输层:网络通信
网络层:TCP/IP核心层,用来分组和发送。
数据结构层+物理层:底层网络
网络通信协议的分类
UDP:用户数据汇报协议,UDP是无连接通信协议,就是不用确定连接。特点是耗资小,通信效率高。但是偶尔会丢一两个数据包。
TCP:传输控制协议,需要先建立连接。特点是,可靠无差错,每次链接都需要进行三次握手,效率没UDP搞
三次握手
客户端喊服务器
服务器回应客户端
客户端丢绳子,确认连接。
网络编程三要素
协议:
协议:规则
IP地址
互联网协议地址,可以理解成电话号码。
IP地址的分类
IPv4:32位的二进制,表示为a.b.c.d形式。
IPv6:128位地址长度
端口号
软件自带的一个数字,A计算机上的软件QQ,带着ip+端口号(3500) 192.0.0.1:3500通过网络到另外一个计算机,另外一个接受了以后,就能用根据端口号从而用QQ来接受这个信息。
TCP通信程序
Clinet与Server通信
Socket/SocketServer
步骤如下:
服务端:
服务器端口创建SocketServer,并指定接收的客户端的端口号。
用accept进行接收,等待客户端传信号。
接收成功以后,获取套接字,通过套接字获取输入流,输出其内容即可。
客户端:
创建套接字,发送信号,如果没有服务器接收,会抛出Connection refused 的异常。
服务器接收信号,经过三次握手建立连接,获取套接字的输出流,write信息,然后开始等待。
获取服务器发送回来的套接字的输入流,获取信息。
TCP文件上传案例
原理::客户端读取本地的文件,文件上传到服务器,服务器把上传的文件保存到服务器的硬盘上面。