Java学习笔记 - 第021天

每日要点

折半查找(二分查找)

例子1:

    public static <T extends Comparable<T>> int binarySearch(T[] array, T key) {
        int start = 0;
        int end = array.length - 1;
        while (start <= end) {
            int mid = (end - start) / 2 + start;
            if (array[mid].equals(key)) {
                return mid;
            }
            else if (array[mid].compareTo(key) > 0) {
                end = mid - 1;
            }
            else {
                start = mid + 1;
            }
        }
        return -1;
    }
    
    public static <T> int binarySearch(T[] array, T key, Comparator<T> comp) {
        int start = 0;
        int end = array.length - 1;
        while (start <= end) {
        //  int mid = (start + end) / 2;  // 有溢出的风险
        //  int mid = (end - start) / 2 + start;
            int mid = (start + end) >>> 1;
            if (array[mid].equals(key)) {
                return mid;
            }
            else if (comp.compare(array[mid], key) > 0) {
                end = mid - 1;
            }
            else {
                start = mid + 1;
            }
        }
        return -1;
    }
        int a = 1300000000;
        int b = 1000000000;
        System.out.println(a + b); //溢出 (超出了int类型的表示范围)

Collections工具类

例子1:

        List<String> list = new ArrayList<>();
        list.add("banana");
        list.add("grape");
        list.add("banana");
        list.add("apple");
        list.add("");
        list.add("zoo");
        list.add("bear");
        Collections.shuffle(list);
    //  Collections.fill(list, "apple");
        for (String str : list) {
            System.out.println(str);
        }
        System.out.println("============");
        // Lambda表达式 ---> 闭包(closure)
        Collections.sort(list, (o1, o2) -> {
            return o2.length() - o1.length();
        });
        for (String str : list) {
            System.out.println(str);
        }
        List<String> list2 = Collections.unmodifiableList(list);
        System.out.println(list2.size());
        list2.remove("apple");
        list2.add("strawberry");

遗留容器

遗留容器 (自己写代码不要用但是可能在项目早期版本中可能遇到的容器)
所谓遗留容器就是在新版本中已经被淘汰的容器 - 设计烂、效率低、使用麻烦
BitSet ---> 二进制位集合
Vector --> ArrayList的线程安全版本
Stack ---> 栈 - FILO的容器 设计烂 不要继承工具类
Dictionary --> 字典 - 键值对容器
Hashtable ---> HashMap的线程安全版本
Properties ---> 键和值的类型都是String的HashMap
Enumeration ---> 枚举器 (对遗留容器进行迭代的迭代器)
例子1:

        // 获得系统属性
        Properties sysProps = System.getProperties();
        for (Object key : sysProps.keySet()) {
            System.out.println(key + ":" + sysProps.get(key).toString());
        }
        // 获得环境变量
        Map<String, String> map = System.getenv();
        for (String key : map.keySet()) {
            System.out.println(key + "--" + map.get(key));
        }
        Properties props = new Properties();
        try {
            // 从文件中加载键值对到容器中
            props.load(Test03.class.getClassLoader().getResourceAsStream("hello.ini"));
        } catch (IOException e) {
            e.printStackTrace();
        }
        System.out.println(props.size());
        for (Object key : props.keySet()) {
            System.out.println(key + "--->" + props.getProperty((String) key));
        }
        Enumeration<Object> e = props.keys();
        while (e.hasMoreElements()) {
            String key = (String) e.nextElement();
            System.out.println(key + "--->" + props.getProperty(key));
        }

异常捕捉机制

try - 把可能在运行时出状况的代码放到try块中保护起来执行
如果出现了异常状况就通过后面的catch来捕获异常对象并进行相应的处理

FileInputStream的构造器用throws关键字声明了FileNotFoundException异常
而且该构造器的代码中判定如果文件无效就用throw关键字抛出异常

一个try块后面可以跟多个catch用于捕获不同的异常类型
但是在书写上要保证先捕获子类型异常再捕获父类型异常
如果捕获的异常之间没有父子关系那么捕获顺序随意

从Java 7开始如果多个catch代码是一致的可以合并为一个catch

打印异常栈轨迹 - 对于调试程序比较有帮助的一个方法
e.printStackTrace();

finally 总是执行代码块 - 不管程序正常异常此处都要执行
此处最适合释放程序中打开的各种外部资源(文件、数据库、网络)

例子1:

        InputStream in = null;
        // try - 把可能在运行时出状况的代码放到try块中保护起来执行
        // 如果出现了异常状况就通过后面的catch来捕获异常对象并进行相应的处理
        try {
            // FileInputStream的构造器用throws关键字声明了FileNotFoundException异常
            // 而且该构造器的代码中判定如果文件无效就用throw关键字抛出异常
            in = new FileInputStream("c:/Java基础.jpg");
            int data;
            while ((data = in.read()) != -1) {
                System.out.printf("%x ", data);
            }
        }
        // 一个try块后面可以跟多个catch用于捕获不同的异常类型
        // 但是在书写上要保证先捕获子类型异常再捕获父类型异常
        // 如果捕获的异常之间没有父子关系那么捕获顺序随意
        // 从Java 7开始如果多个catch代码是一致的可以合并为一个catch
        catch (FileNotFoundException e) {
            // 打印异常栈轨迹 - 对于调试程序比较有帮助的一个方法
            e.printStackTrace();
        } 
        catch (IOException e) {
            e.printStackTrace();
        }
        // 总是执行代码块 - 不管程序正常异常此处都要执行
        // 此处最适合释放程序中打开的各种外部资源(文件、数据库、网络)
        finally {
            if (in != null) {
                try {
                    in.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }

TWR语法

TWR - try-with-resources
从Java 7开始可以使用TWR语法将需要释放的外部资源直接放在try后的圆括号中
这样的话不管正常离开还是异常离开try块 外部资源会自动关闭
实现了AutoCloseable接口的外部资源对象可以使用TWR语法自动关闭
try (....) {}

输入、输出

IO.png

Unix - IO操作被简化成了文件
C++/Java - IO操作被抽象成了流 读数据 - 输入流 - read()
               写数据 - 输出流 - write()
               只有一个方向一个维度

字节输入、输出流

InputStream - 字节流(以字节为单位进行读取) - 二进制文件

三种方式输入:

  • 第一种:
    如果频繁的进行I/O操作会使用CPU的利用率非常低下
    因为I/O操作会导致CPU产生I/O中断 CPU不能满负荷工作
    例子1:
    //  try (InputStream in = new FileInputStream("c:/Java基础.jpg")) {
        try (InputStream in = new FileInputStream("c:/git命令.txt")) {    
            // 如果频繁的进行I/O操作会使用CPU的利用率非常低下
            // 因为I/O操作会导致CPU产生I/O中断 CPU不能满负荷工作
            int data;
            while ((data = in.read()) != -1) {
                System.out.printf("%x ", data);
            }
  • 第二种:
    例子2:
    创建一个字节数组作为输入缓冲区 将文件中的数据直接读入缓冲区中
    这种做法可以减少I/O中断的次数从而保证CPU能够不被频繁中断
    文件太大不好
            // 创建一个字节数组作为输入缓冲区 将文件中的数据直接读入缓冲区中
            // 这种做法可以减少I/O中断的次数从而保证CPU能够不被频繁中断
            // 文件太大不好
            byte[] buffer = new byte[in.available()];
            if (in.read(buffer) != -1) {
                in.read(buffer);
                for (byte b : buffer) {
                    System.out.printf("%02x ", b & 0xff);
                } 
            }
  • 第三种:
    例子3:
            byte[] buffer = new byte[1024];
            int totalBytes; // 记录read操作总共读了多少字节
            while ((totalBytes = in.read(buffer)) != -1) {
            //  System.out.println("\n读到了" + totalBytes + "个字节");
            //  for (int i = 0; i < totalBytes; i++) {
            //      System.out.printf("%02x ", buffer[i] & 0xff);
            //  } 
            System.out.print(new String(buffer, 0, totalBytes));
            }
        }
        catch (IOException e) {
            e.printStackTrace();
        }

字符输入输出流

Reader - 字符流(以字符为单位进行读取) - 文本文件
例子1:

        File file = new File("c:/git命令.txt");
        try (Reader reader = new FileReader(file)){
/*          int ch;
            while ((ch = reader.read()) != -1) {
                System.out.print((char) ch);
                Thread.sleep(200);
            }*/
            
/*          char[] buffer = new char[(int) file.length()];
            int totalChars = reader.read(buffer);
            System.out.print(new String(buffer, 0, totalChars));*/
            
            char[] buffer = new char[512];
            int totalChars;
            while ((totalChars = reader.read(buffer)) != -1) {
                System.out.print(new String(buffer, 0, totalChars));
            //  Thread.sleep(1000);
            }
        } 
        catch (IOException e) {
            e.printStackTrace();
        }

输出流:
FileWriter构造器的第一个参数是文件名或者File对象
第二个参数用来指定是否启动追加模式(在原来文件的末尾输出新的内容)
Writer writer = new FileWriter("c:/abc.txt", true)
例子2:

        String s1 = "Hello, world!";
        String s2 = "你好, 世界!";
        // FileWriter构造器的第一个参数是文件名或者File对象
        // 第二个参数用来指定是否启动追加模式(在原来文件的末尾输出新的内容)
        try (Writer writer = new FileWriter("c:/abc.txt", true)) {
            writer.write(s1);
            writer.write("\r\n");
            writer.write(s2);
        } 
        catch (IOException e) {
            e.printStackTrace();
        }
        System.out.println("程序结束.");

作业

  • 1.九九乘法表文件输出
        try (Writer writer = new FileWriter("c:/qwe.txt")) {
            for (int i = 1; i <= 9; i++) {
                for (int j = 1; j <= i; j++) {
                    writer.write(i + "*" + j + "=" + i * j + "\t");
                }
                writer.write("\r\n");
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
  • 2.写一个方法实现文件拷贝功能
    public static void fileCopy(String source, String target) {
        try (InputStream in = new FileInputStream(source);
            OutputStream out = new FileOutputStream(target)){
            byte[] buffer = new byte[1024];
            int totalBytes;
            while ((totalBytes = in.read(buffer)) != -1 ) {
                out.write(buffer, 0, totalBytes);
            }
        } 
        catch (IOException e) {
            e.printStackTrace();
        }   
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,602评论 18 399
  • 二分法 1、 2、 遗留容器 IO Unix - IO操作被简化成文件 C++ /Java IO操作被抽成了流读数...
    你大爷在奔腾阅读 193评论 0 0
  • 面向对象主要针对面向过程。 面向过程的基本单元是函数。 什么是对象:EVERYTHING IS OBJECT(万物...
    sinpi阅读 1,046评论 0 4
  • (一)Java部分 1、列举出JAVA中6个比较常用的包【天威诚信面试题】 【参考答案】 java.lang;ja...
    独云阅读 7,088评论 0 62
  • 在周末的清晨醒来,晨曦从窗扉斜斜地打进书房,温柔明媚的光影斑驳地散落一地。落在古红色的木质地板上,落在书架那一本本...
    云婵苑阅读 178评论 0 1