Java.io.File类可以获取文件以及文件夹的一些基本的属性
常用的方法 文件名称,路径,大小,判断是否存在,删除,创建
// 创建一个文件对象(可以是文件,可以是文件夹)
File file = new File("e:/java_text.txt");
// 基本属性
boolean canWriter = file.canWrite();
System.out.println("是否可写:" + canWriter);
boolean canRead = file.canRead();
System.out.println("是否可读:" + canRead);
long size = file.length(); // 常用
System.out.println("文件大小:" + size);
boolean isFile = file.isFile(); // 常用
System.out.println("是否是文件:" + isFile);
boolean isDirectory = file.isDirectory();
System.out.println("是否是文件夹:" + isDirectory);
String filename = file.getName(); // 常用
System.out.println("文件的名称:" + filename);
String absolutePath = file.getAbsolutePath(); // 常用
System.out.println("文件的绝对路径:" + absolutePath);
String filepath = file.getPath();
System.out.println("文件的绝对路径:" + filepath);
boolean isExists = file.exists(); // 常用
System.out.println("是否存在:" + isExists);
boolean isDelete = file.delete(); // 常用
System.out.println("是否已经被删除:" + isDelete);
boolean isCreate = file.createNewFile();
System.out.println("创建了没有:" + isCreate);
File类的listFile获取文件夹下面的所有文件内容,可以通过递归调用的方法把某一个文件夹下的所有的文件查询出来
// 测试文件目录的属性(递归遍历文件夹中所有的文件信息)
public static void testDirectoryDeme(File file) {
if (file.isDirectory()) {
File[] files = file.listFiles();
System.out.println("文件夹"+file.getName()+"有"+files.length+"个文件");
// 利用for遍历所有的文件
for (int i = 0; i < files.length; i++) {
File childFile = files[i];
if (childFile.isFile()) {
// 这是一个文件
System.out.println("\t这是一个文件:" + childFile.getName());
} else {
// 文件夹:继续递归调用
testDirectoryDeme(childFile);
}
}
} else {
// 这是一个文件
System.out.println("\t这是一个文件:" + file.getName());
}
}
文件的创建,删除,判断是否存在,移动,拷贝. 因为提供了静态的方法,所以不需要创建对象直接调用方法即可
// 如果文件不存在复制
if (!Files.exists(Paths.get("e:/a/cart1.jpg"))) {
// java.nio.Files(文件的工具类) Paths(文件路径工具类)
Files.copy(Paths.get("e:/cart1.jpg"), Paths.get("e:/a/cart1.jpg"));
}
Files.move(Paths.get("e:/a/cart1.jpg"), Paths.get("e:/a/b/cart1" + ((int) (Math.random() * 100)) + ".jpg"));
Files.delete(Paths.get("e:/cart1.jpg"));
文件流的分类:
根据功能分为:输入流(读取文件) 和 输出流(写入文件)
根据操作内容:字符流(读取字符数组) 和 字节流(读取字节数组)
字节输入流,字节输出流,字符输入流,字符输出流
使用字节流实现文件的读取
//利用字节输入流实现文件的内容读取(inputStream 接口的 FileInputStream )
public static void testInputStream() throws Exception{
File file = new File("e:/a/file.txt");
InputStream is = new FileInputStream(file);
if(!file.exists()){
System.out.println("文件不存在");
}
//开始读取文件
byte[] temp_bytes = new byte[1024];
int size = 0; //用于记录读取文件的字节个数,如果没有读取任何的内容返回-1
//因为文件不可能一次读取完毕,需要循环读取
do{
size =is.read(temp_bytes);
if(size!=-1){
String info = new String(temp_bytes,0,size,"GBK");
System.out.println("读取的内容是:" + info);
}
}while(size !=-1);
//文件流是必须要关闭的(像水管子一样)
is.close();
利用字节输出流实现文件的写入
//利用字节输出流实现文件内容的写入(OutputStream 接口的FileOutputStream)
public static void testOuputStream() throws Exception{
File file = new File("e:/a/file_new.txt");
if(file.exists()){
file.createNewFile();
}
//文件写入
String info = "这就是我们要写入文件的内容";
//创建文件写入对象
OutputStream os = new FileOutputStream(file);
os.write(info.getBytes());
//写入完毕后,关闭
os.flush(); //清空缓存区
os.close();
}
利用字节输入流和字节输出流实现文件的拷贝
//利用字节输入输入输出流,实现文件的复制,为了防止文件名称重复,需要对文件名称重命名
public static void testCopy(String filepath) throws Exception
{
//创建文件对象
File file = new File(filepath);
//判断文件是否存在,且必须是一个文件而不能是一个文件夹
if(!file.exists())
throw new Exception("文件不存在");
if(file.isDirectory())
throw new Exception("只能拷贝文件,不能拷贝文件夹");
//默认目标地址就是e:/a 文件夹
//开始拷贝
//创建一个文件输入流对象,读取文件的内容
InputStream is = new FileInputStream(file);
//创建一个文件输出流对象,写入文件的内容
String filename = getFileName(file.getName());
String targetpath="e:/a/"+filename;
OutputStream os = new FileOutputStream(targetpath);
//利用循环,边读取内容,边写入内容即可
byte[] temp_info = new byte[1024]; //利用临时数组保存读取的内容
int size = 0; //保存读取的字节个数,如果没有读取到内容返回-1
do{
//先读取
size = is.read(temp_info);
//判断是否读取到了文件的内容
if(size!=-1){
//写入文件
os.write(temp_info, 0, size);
}
}while(size!=-1);
//关闭,先关闭输出流,后关闭输入流
os.flush();
os.close();
is.close();
}
//根据原有的文件名称获取新的文件名称
public static String getFileName(String fileName){
SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddhhmmss");
//abc.mp4 ---abc_20190805164520.mp4
//根据.分别获取文件名称和扩展名
String[] name_infos = fileName.split("\\.");
//获取当前日期的字符串
Date date = new Date();
String dateStr = sdf.format(date);
return name_infos[0]+"_"+dateStr+"."+name_infos[1];
}
重点是熟练使用FileInputStream 和FileOutputStream 的使用
// 利用字节流复制某一个文件夹中的所有文件
public static void testCopyDirectory() throws Exception {
File file = new File("e:/file_source");
// 实现复制
// 创建文件输入输出流对象
InputStream is = null;
OutputStream os = null;
try {
// 遍历这个文件夹下的所有的文件信息
File[] files = file.listFiles();
for (int i = 0; i < files.length; i++) {
File childFile = files[i];
// 如果是一个文件就复制
if (childFile.isFile()) {
is = new FileInputStream(childFile);
// 根据原有的文件名称获取新的文件名称
String newFileName = getNewFileName(childFile.getName());
os = new FileOutputStream("e:/a/b/" + newFileName);
byte[] temp_info = new byte[1024];
int size = -1;
do {
// 先读取
size = is.read(temp_info);
// 后写入(写入的内容多去取决于读取的内容多少)
if (size != -1) {
os.write(temp_info, 0, size);
}
} while (size != -1);
}
}
} catch (Exception ex) {
throw new Exception(ex);
} finally {
if (os != null)
os.close();
if (is != null)
is.close();
}
}
// 根据原有的文件名称,获取新的文件名称
public static String getNewFileName(String oldFileName) {
// oldname :上机作业.docx newnmae : 上机作业_20180222.docx;
// 根据原有的文件名称获取文件名字和文件的类型
int index = oldFileName.lastIndexOf(".");
if (index == -1) {
return oldFileName;
}
// 获取文件名: 上机作业
String name = oldFileName.substring(0, index);
// 获取文件类型: docx
String type = oldFileName.substring(index + 1);
SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddhhmmss");
Date date = new Date();
String formatstr = sdf.format(date);
// 上机作业_20180222.docx;
return name + "_" + formatstr + "." + type;
}
使用对象输入输出流实现对象的序列化和反序列化
public static void testObjectInputStream() throws Exception{
File file = new File("e:/a/stu.txt");
InputStream is = new FileInputStream(file);
ObjectInputStream ois = new ObjectInputStream(is);
List stuList= (List)ois.readObject();
ois.close();
is.close();
for(Student stu: stuList){
System.out.println(stu);
}
}
//对象序列化(实现JAVA对象的保存和读取 ObjectInputStream)
public static void testObjectOutputStream() throws Exception{
Student stu3 = new Student("赵丽丽","女","河北廊坊");
stuList.add(stu3);
//把集合保存到文件中
File file = new File("e:/a/stu.txt");
OutputStream os = new FileOutputStream(file);
ObjectOutputStream oos = new ObjectOutputStream(os);
oos.writeObject(stuList);
oos.flush();
oos.close();
os.close();
}
字符流的读取和写入(只能读取字符文件的信息)
//使用字符流读取文件
public static void testReader() throws Exception{
//创建文件对象
File file = new File("e:/a/file.txt");
//创建字符输入流对象
Reader reader = new FileReader(file);
//创建字符输出流对象
File file2 = new File("e:/a/b/file_3.txt");
Writer writer = new FileWriter(file2);
char[] temp = new char[1000];
int size =0;
//通过循环边度编写
do{
size= reader.read(temp); //读取字符内容
if(size!=-1)
{
writer.write(temp, 0, size);
}
}while(size!=-1);
//关闭
writer.flush();
writer.close();
reader.close();
}
释放资源的新方法
//新的关闭资源的方法
//try( 定义必须要关闭的对象; ){}catch(Exception ex){};
// 创建输出流对象
try( OutputStream os = new FileOutputStream(FILE_PATH);
ObjectOutputStream oos = new ObjectOutputStream(os); ) {
oos.writeObject(stuList);
} catch (Exception e) {
throw new Exception(e);
}
初始状态(创建了对象)
可运行状态(调用了start方法)
运行状态(调用了run方法)
阻塞状态 (调用了seleep,join,wait方法)
终结状态 (运行完毕)
Start() 启动方法 表示线程进入了可运行状态
Seleep(int) 随眠方法,当前线程进入阻塞状态,在一定时间以后苏醒
Join() 方法 被调用者优先执行,执行完毕后当前线程在执行(阻塞的的是当前线程)
Wait() 方法 当前线程进入阻塞状态,一直到对方唤醒自己才可以继续执行
notifyAll()唤醒被阻塞的线程
需要注意:wait(),notify()两个方法是Object类提供的
package com.xja.sxnd.filedemo;
public class ThreadDemo {
public static void main(String[] args) {
MyThread mt = new MyThread();
MyRunable mr = new MyRunable();
Thread thread = new Thread(mr);
mt.start();
thread.start();
for(int i = 0;i< 50;i++){
if(i == 20){
try{
//主线程执行到20的时候,先让第一个线程执行完毕,然后主线程在接着执行
mt.join();
}catch(Exception ex){
ex.printStackTrace();
}
}
System.out.println("主线程中的内容:" +i);
}
}
}
class MyThread extends Thread
{
@Override
public void run() {
for(int i =0;i< 50;i++){
try{
if(i==20)
Thread.sleep(500); //停止运行0.5秒,时间到了以后自动执行
}catch(Exception ex){
ex.printStackTrace();
}
System.out.println("第一个线程的对象:" +i);
}
}
}
class MyRunable implements Runnable{
@Override
public void run() {
for(int i =0;i< 50;i++){
System.out.println("第二个线程的对象:"+i);
}
}
}
.
如果两个线程同时调用对方线程的JOIN方法,互相希望对方先运行就会出现死锁的问题
public class MyRunnable implements Runnable {
public Thread myThread;
@Override
public void run() {
try {
for (int i = 0; i < 100; i++) {
Thread.sleep(10);
if(i == 5){
System.out.println("MyRunnable运行到了5,也休息,让Thread先走");
myThread.join();
}
System.out.println("第二个线程:" + i);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public class MyThread extends Thread {
public Thread myRun;
@Override
public void run() {
try {
for (int i = 0; i < 10; i++) {
Thread.sleep(10);
if(i==5){
System.out.println("Thread 运行到了5 就需要对方先走");
myRun.join();
}
System.out.println("第一个线程的内容:" + i);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) {
// TODO Auto-generated method stub
try {
// 创建一个线程对象
MyThread myThread = new MyThread();
// 创建第二个线程
MyRunnable myRunn = new MyRunnable();
Thread thread = new Thread(myRunn);
// 启动线程 启动线程不能直接调用run 只能调用start
myThread.myRun = thread;
myRunn.myThread = myThread;
myThread.start();
thread.start();
}
StringBuilder和StringBuffer ArrayList 和vector HashMap 和HashTable
当两个线程同时操作同一个对象的时候,因为两个线程互相影响对方的结果,导致数据不统一,这种现象称之为线程不同步
例如:银行卡存取钱的问题。(两种方式操作一个账户)
public class Card {
private int balance =500;
public void takeMoney(int money){
if(balance>money){
System.out.println("取钱之前的余额:" + balance);
int temp_balance = balance - money;
balance = temp_balance;
System.out.println("取钱之后的余额:" + balance);
}
}
public void saveMoney(int money){
System.out.println("存钱之前的余额:" + balance);
int temp_balance = balance + money;
balance = temp_balance;
System.out.println("存钱之后的余额:" + balance);
}
}
public class ThreadDemo {
public static void main(String[] args) {
Card card = new Card();
MyThread mt = new MyThread(card);
MyRunable mr = new MyRunable(card);
Thread thread = new Thread(mr);
mt.start();
thread.start();
System.out.println("程序结束");
}
}
class MyThread extends Thread
{
private Card card;
public MyThread(Card card){
this.card = card;
}
@Override
public void run() {
for(int i =0;i< 10;i++){
card.takeMoney(50);
}
}
}
class MyRunable implements Runnable{
private Card card;
public MyRunable(Card card){
this.card = card;
}
@Override
public void run() {
for(int i =0;i< 10;i++){
card.saveMoney(50);
}
}
}
通过synchronized关键字修饰方法就是线程同步的方法,线程同步的方法要求同一个对象同时只能调用一个方法
public class Card {
private int balance = 500;
public void takeMoney(int money) {
synchronized (this) {
if (balance > money) {
System.out.println("取钱之前的余额:" + balance);
int temp_balance = balance - money;
balance = temp_balance;
System.out.println("取钱之后的余额:" + balance);
}
}
}
public void saveMoney(int money) {
synchronized (this) {
System.out.println("存钱之前的余额:" + balance);
int temp_balance = balance + money;
balance = temp_balance;
System.out.println("存钱之后的余额:" + balance);
}
}
}
public class Card {
private int balance = 500;
public synchronized void takeMoney(int money) throws Exception {
if (balance -money<0) {
System.out.println("家里没有钱了,需要减少消费");
this.wait();
return;
}
System.out.println("取钱之前的余额:" + balance);
int temp_balance = balance - money;
balance = temp_balance;
System.out.println("取钱之后的余额:" + balance);
// 一旦我花费了钱,就可以让生成者继续生成
this.notify();
}
public synchronized void saveMoney(int money) throws Exception {
// 如果挣钱的足够,则需要休息,让消费者消费
if (balance > 700) {
System.out.println("钱以够多了,可以休息一下");
this.wait();
return;
}
System.out.println("存钱之前的余额:" + balance);
int temp_balance = balance + money;
balance = temp_balance;
System.out.println("存钱之后的余额:" + balance);
// 一旦有钱就可以让消费者消费
this.notifyAll();
}
}