跳转至

todo

Overview 部分

  • 垃圾回收机制

  • 保留字

  • 内存分布

  • 访问权限符

  • Package

  • 继承

  • 类内类的顺序

容器

  • Vector

Exception

  • fw

IO

Others

  • regex

Java

约 3847 个字 587 行代码 15 张图片 预计阅读时间 27 分钟

Tip

  • jshell 可以使用与 python 类似的交互

概念

  • SDK

  • JDK

    功能齐全的 Java 开发工具包。包含JRE和Java Development Tools

  • JVM

    Java 虚拟机(Java Virtual Machine, JVM)是运行 Java 字节码的虚拟机。JVM 有针对不同系统的特定实现(Windows,Linux,macOS),目的是使用相同的字节码,它们都会给出相同的结果。字节码和不同系统的 JVM 实现是 Java 语言“一次编译,随处可以运行”的关键所在。

  • JRE 是运行已编译 Java 程序所需的环境,主要包含以下两个部分:

    1. JVM : 也就是我们上面提到的 Java 虚拟机。

    2. Java 基础类库(Class Library):一组标准的类库,提供常用的功能和 API(如 I/O 操作、网络通信、数据结构等)。

overview

  • Compiled Language
  • All objects should be constructed in runtime and be stored in heap.
  • 单根结构(除了C++的所有OOP语言) Every class in Java is a descendant of one class: Object
  • Java 中的输入输出,Scanner 容易超时
//import java.util.*;  
import java.util.Scanner;  
public class Main {  
    public static void main(String[] args) {  
       Scanner sc = new Scanner(System.in);  
       int a = sc.nextInt();  
       System.out.println(a);  
    }  
}
  • Java 中的保留字
  • Identifier

和 C++ 不同的语言特性

  • Java 中的常量使用关键字 final 而不是 const. final int a = 5;

  • Java 中类型推导关键字为 var var s = new StringBuffer();

  • Java 的 reference 更像是 C++中的指针,不可计算. 任何对象变量都是指针

    • 对象变量的赋值

Example

将这里理解为类似指针,是操作权指向的赋予,不会像 C++ 一样做赋值

var a = new A(10);  

var b = new A(20);  

System.out.printf("%d %d\n", a.i, b.i);  

a = b;  

b.i = -20;  

System.out.printf("%d %d\n", a.i, b.i);
  • Auto Memory Management -- 单根结构

  • index check

    • Primitive 可以存放在栈,对象全部都放在堆(heap 只能通过 new 获得)

Dynamic

  • Java 中的一个文件只能含有一个 public 类,其名称与文件名称相同

Java 的运行

image.png

内存布局

Package

  • Java 的 Package 依赖于目录
    • 之后的代码属于这个 package
    • 这部分代码必须放在一个名为 <Package name> 的文件夹中
package <Package name>
  • 设置寻找 Package 的地址
    • 环境变量
    • 通过 -cp 指定寻找路径, java -cp java hello.Hello

Package的作用域

包在 Java 的访问控制中起着重要作用,主要涉及以下访问修饰符:

  • public:公共访问,任何类都可以访问。
  • protected:受保护的访问,允许同一包内的类和不同包中的子类访问。

    • 跟C++不同
  • 默认(包私有):如果不指定访问修饰符,默认为包私有,只有同一包内的类可以访问。

  • private:私有访问,仅在同一类内可见

Static Import

double r = Math.cos(Math.PI * theta); 
import static java.lang.Math.PI; 
import static java.lang.Math.*; 
double r = cos(PI * theta);

String API

  • 所有字符类型都是 unicode,就像下面的中文也会是一个字节
  • 结尾没有\0
对长度的判断
String a = "hello你好";  // len = 7
String b = "hello";     // len = 5
  • String 为不可变类型。

Methods

得到属性 - s.length () - s.charAt (int index)

切片

  • s.substring (int , int )

比较

  • s.compareTo ()
  • s.equals ()
  • s.equalsIgnoreCase ()
  • Boolean startsWith (String str)
  • Boolean endsWith (String str)

变换

  • String trim () 删除前导、后导空格
  • String replace (char c 1, char c 2) 替换
  • public String replaceAll(String regex, String replacement) 可使用正则表达式进行替换。
  • 正则表达式 与其他的值相互转化
String piStr = "3.14159";
Float pi = Float.ValueOf (piStr);
Float pi2 = Float.ParseFLOAT (piStr);

比较

public static void main(String[] args) {
        String s = "abc";  // 指向字符串常量池
        String t = "abc";
        String u = new String("abc");   // new的时候一定会创建一个新的对象在heap
        System.out.println(s == t);             //  true
        System.out.println(s == u); // false
}
String str = "Person";
String str1 = new String("Person");
System.out.println(str == "Person");  // true
System.out.println(str1 == "Person"); // false

Basic

参数传递 - Passing value - 对象传递也是类似指针,无法阻止函数内部对对象的改变 Relation - 整型的 wrapper 在[-128,127]之间是一个固定的对象(类似于字符串常量池的概念)

Integer n1 = new Integer(47);
Integer n2 = new Integer(47);
System.out.println(n1 == n2);  // false

Integer n3 = 47;  
Integer n4 = 47;
System.out.println(n3 == n4);  // true

Integer n1 = 147;// new Integer(47);
Integer n2 = 147;// new Integer(47);
System.out.println(n1 == n2);  // false

垃圾回收 GC

深入理解 JVM 的垃圾回收机制 | 二哥的Java进阶之路 (javabetter.cn)

使用大端编码

Class

Note

  • 类在内存中也是一个对象属于 Class

  • Java 会对 new 的对象的内存清空为 0

  • 定义初始化,在构造函数之前,初始化顺序与在 class 中定义的顺序相关

  • 一个类的定义,可以继承一个父类的同时,再实现多个接口

    class MathTest extends Student implements BaseTest

    • 可以在Student类中实现一部分接口的函数,在MathTest中就无需实现这部分

代理构造

Java 的代理构造和调用父类的构造函数都要放在构造函数开始的位置。

public Rectangle(int width, int length) {
    this.width = width;
    this.length = length;
}
public Rectangle() {
    this(0, 0);
}

初始化顺序

Static member is to be initialized in the loading of the class

graph LR
1[Load 调用static函数] --> 2[定义初始化] --> 3[调用构造函数]
  • Java的初始化会在Heap上申请内存,把这一块内存初始化为空
    • 对于Primitive会赋初值如0

静态初始化

static { ... }
  • 在类被装载的时候运行且只会执行一次
  • 类的状态是运行时装载,所以可能一个类未被装载

定义初始化

{ ... }
  • ==每次创建实例==都会调用

构造初始化

ClassName(){}

一个例子

class A {
    {
        System.out.println("class A : instance initializer");
    }
    public int i = baz();
    {
        System.out.println("class A : i = " + i);
    }

    public int baz() {
        System.out.print("class A : baz\n");
        return 0;
    }
}

public class B extends A {

    public int baz() {
        System.out.print("class B baz : B\n");
        return 10;
    }

    public int i = baz();

    public static void main(String[] args) {
        B b = new B();
    }
}
  • 输出
    class A : instance initializer
    class B baz : B
    class A : i = 10
    class B baz : B
    

函数的绑定

Case 2 Shape. Java

Java 默认为动态绑定,

  • Static binding: call the function as the code
  • Dynamic binding: call the function of the object

override

Final

Abstract & Interface

Abstract

  • 一个抽象类中可以没有 abstract 函数,但含有 abstract 函数一定要是抽象类

Interface

  • 不能拥有构造函数
  • All data members in interface are public static final.
  • All methods in interface are public.

    • 这里实现的时候一定要注意权限的问题,因为默认的权限是public的,所以在实现函数的时候要加上public
      interface I {
          void f();
      }
      
      class C {
          void f() {
          };
      }
      
      interface A extends I {
          void f();
      }
      
      class B extends C implements I { 
          // ERROR : 正在尝试分配更低的访问权限; 以前为public,现在为default
      }
      
  • implements interface 可以实现多个接口

interface Instrument5 {
    // Compile-time constant:
    int i = 5; // static & final
    // Cannot have method definitions:
    void play(); // Automatically public
    String what();
    void adjust();
}
  • 它前面的修饰符只能是abstract / public
abstract interface BaseTest

Note

接口引入了新的方法类型:default 方法、static 方法和 private 方法。这些方法让接口的使用更加灵活。

  • Java 8 引入的default 方法用于提供接口方法的默认实现,可以在实现类中被覆盖。这样就可以在不修改实现类的情况下向现有接口添加新功能,从而增强接口的扩展性和向后兼容性。
public interface MyInterface {

default void defaultMethod() {

System.out.println("This is a default method.");

}

}
  • static 方法无法在实现类中被覆盖,只能通过接口名直接调用( MyInterface.staticMethod()),类似于类中的静态方法。static 方法通常用于定义一些通用的、与接口相关的工具方法,一般很少用。
public interface MyInterface {

static void staticMethod() {

System.out.println("This is a static method in the interface.");

}

}

  • Java 9 允许在接口中使用 private 方法。private方法可以用于在接口内部共享代码,不对外暴露。
public interface MyInterface {

// default 方法

default void defaultMethod() {

commonMethod();

}



// static 方法

static void staticMethod() {

commonMethod();

}



// 私有静态方法,可以被 static 和 default 方法调用

private static void commonMethod() {

System.out.println("This is a private method used internally.");

}



// 实例私有方法,只能被 default 方法调用。

private void instanceCommonMethod() {

System.out.println("This is a private instance method used internally.");

}

}

Enum 枚举类

  • 也是类,相当于构建了匿名子类,不是像C一样的宏(会进行类型检查int i = 1; i == ONE报错)

    • ==equals 效果相同,是比较特殊的引用类型

      • equals会在null的时候报错
      • enum类型的每个常量在JVM中只有一个唯一实例,所以可以直接用==比较
    • 无法通过new创建实例

    • Suit.values()直接遍历访问整个enum类的所有
public enum Rank { DEUCE, THREE, FOUR, FIVE, SIX,
        SEVEN, EIGHT, NINE, TEN, JACK, QUEEN, KING, ACE }

public enum Suit { CLUBS, DIAMONDS, HEARTS, SPADES }
static {
    for (Suit suit : Suit.values())
        for (Rank rank : Rank.values())
            protoDeck.add(new Card(rank, suit));
}
  • 可以存在同名的函数与变量

  • 解决switch case的思路之一

  • 枚举常量必须在字段和构造函数之前定义

    public enum Planet {
        MERCURY(3.303e+23, 2.4397e6),
        VENUS(4.869e+24, 6.0518e6),
        EARTH(5.976e+24, 6.37814e6),
        MARS(6.421e+23, 3.3972e6),
        JUPITER(1.9e+27, 7.1492e7),
        SATURN(5.688e+26, 6.0268e7),
        URANUS(8.686e+25, 2.5559e7),
        NEPTUNE(1.024e+26, 2.4746e7),
        PLUTO(1.27e+22, 1.137e6);
    
        private final double mass; // in kilograms
        private final double radius; // in meters
    
        Planet(double mass, double radius) {
            this.mass = mass;
            this.radius = radius;
        }
    
        public double mass() {
            return mass;
        }
    
        public double radius() {
            return radius;
        }
    
        // universal gravitational constant (m3 kg-1 s-2)
        public static final double G = 6.67300E-11;
    
        public double surfaceGravity() {
            return G * mass / (radius * radius);
        }
    
        public double surfaceWeight(double otherMass) {
            return otherMass * surfaceGravity();
        }
    
        public static void main(String[] args) {
            double earthWeight = Double.parseDouble(args[0]);
            double mass = earthWeight / EARTH.surfaceGravity();
            for (Planet p : Planet.values())
                System.out.printf("Your weight on %s is %f%n",
                        p, p.surfaceWeight(mass));
        }
    }
    

继承与多态

class BoardGame extends Game {
    BoardGame(int i) {
        super(i);
        System.out.println("BoardGame constructor");
    }
}
  • Java 没有 name hide 的问题,子类的一个函数 override 不会覆盖父类的同名函数

  • 静态初始化的顺序问题
  • Upcast

内部类

  • 装载是分开做的,只有在使用的时候才会进行装载。
  • 函数中的内部类
    • 会在前面加上数字编号以区分不同的函数中的类

成员内部类

  • 成员内部类可以调用外部的变量

局部内部类


匿名类

  • 构造匿名子类
    • 可以访问外部的内容 but Argument must be final to use inside anonymous inner class
    • 闭包 与它相关的本地变量不会被回收,(保存当时的外部环境)
    • 定义初始化块可以充当它的初始化函数
// 函数中的匿名类
public Contents cont() {
    return new Contents() {
        private int i = 11;
        public int value() {
            return i;
        }
    }; // Semicolon required in this case
}

public class ThreadDemo {
    public static void main(String[] args) {
        Thread t = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());
            }
        });
        t.start();
    }
}

Generic Containers | 泛型容器

img

Var Args

static void f(Object[] x) {

for (int i = 0; i < x.length; i++)

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

}

}



f(new Object[] { new A(), new A(), new A() });

使用 Object 可以接受这种类型的数组,但是对于Array就会失效

  • 与C++相同,同时使用index和iterator访问时,如果使用index删除某个元素,可能导致iterator的混乱
Iterator<A> it = v.iterator();
while (it.hasNext()) {
    System.out.println(it.next());
}

ArrayList

ArrayList<Integer> a = new ArrayList<Integer>(Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10));

// error

不应该使用int赋给Integer,ArrayList<Integer> a = new ArrayList<>(Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10));

Array

int [] a = new int[10];

for (var x: a){
    x += 1;   // 不会改变它的值,相当于使用值(尽管Array本身是指针)
}

int [] b = a; // 指针
b[0] = 16;    // a[0] = 16
  • 对象的数组中放的实质是指针
Value [] a = new Value[10];
for (var x: a) // 这里x为指针,通过x访问它的元素然后改变

Collection

  • Java没有实现重载,访问某个元素只能使用get不能使用[]
  • Java的泛型不支持Primitive,只能使用wrapper类

共有操作

  • add
  • addAll(Collection)
  • toArray

Colletion

List

ArrayList<Content_Type> 存储方式为Array


  • add一个类的对象,放的还是指针
ArrayList<Value> list = new ArrayList<>;
...
Value v1 = list.get(0);

Set

Map


使用自己定义的类作为键值对

  • must override both hashCode( ) and equals( ),

Generic

  • A generic type declaration is compiled once and for all, and turned into a single class file. 这一点和C++不同

subtype

image-20241024164037656

- Vector

WildCards

void printCollection(Collection<?> c) { 
        for (Object e : c) { 
            System.out.println(e); 
        }
}
// 限定shape子类
public void drawAll(List<? extends Shape> shapes) 
{ ... }
  • 带边界的泛型可以是一个类或多个类的子类。例如,<T extends A & B>表示T必须是A和B的子类。如果用户不清楚这一点,可能会误解如何使用泛型。

Exception & IO

  • System.out.println(true ? Integer.valueOf(1) : Double.valueOf(2.0));
    • 编程语言的格式对齐,一条表达语句的结果只能存在一种情况
  • Java中throw的对象一定是Throwable (the exception root class) object

匹配机制

  • 和C++相同,父类放在最后

异常声明

  • 未处理的异常必须要声明throws ... void f() throws TooBig, TooSmall, DivZero{ }
  • 一个函数中存在异常
    • 调用的函数抛出异常
    • 自己抛出异常

类型

image-20241024192448751

  • Error 是编译时检查的错误

限制

  • When you override a method, you can throw only the exceptions that have been specified in

    the base-class version of the method.

  • Classs

    • 构造
      • 仍然存在C++资源泄露的问题(文件)
    • 非构造
  • 对于多来源的类,子类的能抛出的异常是父类的交集(可能被当做任何一种类,is-a)

Stream

img

这里不同种类的IO不是相互替代的关系

version 1

  • inputStream

  • outputStream

一些需要注意的类

  • media
  • filter Stream

    - DataInputStream 实现读写基本数据类型 读取xx个byte并转换

System.in.read 读取裸数据

Bridge

version 2

  • Reader
  • Writer

在binary和/R/W之间搭建桥梁

File Class

序列化

public class Employee implements Serializable {
    public static final long serialVersionUID = 1L; // 用于检测是否是同一个版本,判断是否可用于反串行化
    public String name;
    public String address;
    public transient int age; // transient瞬态修饰成员,不会被序列化
}
  • statictransient 修饰的字段是不会被序列化的。
  • 读写之后的对象不是同一个对象
    • 但是写进去的时候构造函数不会被调用,直接把值放进去
  • 只有在父类也实现Serializable才能保存其数据,也可以直接由父类实现,而子类不实现

transient

  • 只能修饰字段,而不能修饰方法和类

自定义

private void writeObject(ObjectOutputStream oos) throws IOException {
    oos.defaultWriteObject(); // 使用默认序列化
    oos.writeObject(encrypt(password)); // 对敏感数据加密后序列化
}

private void readObject(ObjectInputStream ois) throws IOException, ClassNotFoundException {
    ois.defaultReadObject(); // 使用默认反序列化
    password = decrypt((String) ois.readObject()); // 反序列化后解密
}

GUI

Note

MVC

MVVM

Thread

image-20241114165902044

基本实现

  1. public class MyThread extends Thread
    1. Thread t1; t1.setName(""); t1.start();
  2. public class MyRunnable implements Runnable
    1. MyRunnable mr; Thread t1 = new Thread(mr, "name");
    2. 好处:避免了 Java 单继承的局限性,Java 不支持多重继承,因此如果我们的类已经继承了另一个类,就不能再继承 Thread 类了。

函数

  • wait

  • public final void wait() throws InterruptedException

    • 调用wait需要考虑后面的exception
  • 进入等待队列前,会把所有线程的 key "归还"

  • 回来的条件

    • 被nofity

    • 能够获得key

      synchronized (theSender) {
          theSender.isValid = false;
          // 
          theSender.notify();
      }
      // 
      

  • The wait(), notify(), and notifyAll() methods must be called in a synchronized method or a synchronized block on the calling object of these methods.

  • sleep

  • 和wait抛出的异常相同

  • 一个静态函数Thread.sleep(<time>),针对对当前线程
  • 使得休眠时间 >= 指定时间

  • yield

    • 放弃当前这个时间片,直接进入ready queue

    • 可能导致资源被浪费:CPU 占用率居高不下

  • join

  • 等待

Group

  • 介绍
  • 每个线程属于一个group
  • Tree的架构
  • 优先级问题
    • 默认和创建线程的线程优先级相同

synchronized | 对共享变量的访问控制

  • 保证一个线程的变化(主要是共享数据的变化)被其他线程所看到
  • zhi'neng
  • 阻止同时访问某一段代码
  • Synchronized Method | 不需要在函数内部使用关键字
synchronized void sync() {
    System.out.println(Thread.currentThread().getName() + "in sync");
}

image-20241202171112719

线程交互

自嵌套

  • java可以保证再次使用key的正确性,不会导致死锁

DeadLock

Pipe

out - > in

Producer - Consumer

Explicit Lock

  • synchronized 是一个内置的同步机制(内部锁 | 隐式锁),它依赖于JVM来管理锁的获取和释放。
  • 显式锁
ReentrantReadWriteLock rwLock = new ReentrantReadWriteLock();
Lock readLock = rwLock.readLock();
Lock writeLock = rwLock.writeLock();

// 获取读锁
readLock.lock();
try {
// 执行读操作
} finally {
    readLock.unlock();
}

// 获取写锁
writeLock.lock();
try {
    // 执行写操作
} finally {
        writeLock.unlock();
}

并行和并发

线程安全类

- ArrayList、HashSet 线程不安全

volatile

  • mark 一个主存中的变量
  • 不保证原子性(相邻原子操作之间不是原子)

    • increment操作可能失效
    • Any operation over double and long is not an atomic one, unless the variable is declared as volatile
      • 对于一个
  • 作为 fence (和OS中的指令作用sfence.vma zero, zero很相似,用来刷新之前的写的值到memory/cache)

    • volatile object
      • Student S s the pointer is volatile, not the whole object

Socket & JDBC

SQL

  • PK 的选择
    • 一般整数
  • 数据字典
    • {1 : "Name"}
  • 数据库迁移
    • 数据类型(某些特殊的)

TCP

  • ServerSocket
    • 在一个线程中一直等待连接,一旦来了一个连接就创建一个新的线程来处理(创建一个socket)
  • Socket
    • 具体的连接都是使用这个类

NIO

  • Channels: Connections to files, sockets etc that support non-blocking reads

  • Buffers: Array-like objects that can be directly read or written by Channels

    • 不同的数据类型
  • Selectors: Tell which of a set of Channels have IO events
    • 向 selector 注册 channel
    • 阻塞,
  • SelectionKeys: Maintain IO event status and bindings

Lambda 表达式

  • 没有变量必须存在()
    • btn.addActionListener(event -> System.out.println("OK"));
  • 存在一个的情况下可以不加()
  • Lambda expression needs an explicit target-type
  • 从lambda 表达式引用的本地变量必须是最终变量或实际上的最终变量
    • 内部新定义的值可以不是 final var
  • Lambda expression's local variable i cannot redeclare another local variable defined in an enclosing scope. 表达式内部使用的值不能覆盖外部的值
  • 闭包 closure
BinaryOperator<Long> add = (x,y) -> x+y;
System.out.println(add.apply(100L, 200L));
BinaryOperator<Long> addExp = (Long x,Long y) -> x+y;
System.out.println(addExp.apply(100L, 200L));
  • 赋值
    • 如果某个类/接口中只有一个函数,可以直接把lambda表达式赋值 | Functional Interface
interface Func {
    void ff();
}
public class FuncInterface {
    public void app(Func f) {
        f.ff();
    }
    public static void main(String[] args) {
        FuncInterface fi = new FuncInterface();
        fi.app(()->System.out.println("Hello"));
    }
}
  • Predicate
Predicate<Integer> atLeast5 = x-> x>5;
public inteface Predicate<T> {
    boolean test(T t);
}
  • Binary Operator
BinaryOperator<Integer> addInt = (x,y) -> x+y;

Stream | 流式计算

设计模式

单例模式 | Singleton

补充知识

命名规范

5 分钟编码,1 小时命名,笑 | 二哥的Java进阶之路 (javabetter.cn)

Java 正则表达式

Pattern pattern = Pattern.compile("your_regex_pattern"); // 创建正则表达式
Matcher matcher = pattern.matcher("your_input_string");  // 创建匹配器对象

Pattern

Matcher类

Matcher 类 - Dev.java - Java 中文 (java-lang.cn)

boolean matches():整个输入字符串是否完全匹配正则表达式。

boolean find():扫描输入字符串,查找与正则表达式匹配的下一个子序列。

String group():返回上一次匹配的子序列。

int start():返回上一次匹配的起始索引。

int end():返回上一次匹配的结束索引(不包括)。

String replaceAll(String replacement):替换所有匹配的子字符串。

String replaceFirst(String replacement):替换第一个匹配的子字符串。

中文处理

处理输入输出

练习题

Week 1

Week 2


Question

Q 1

About access control in Java, which statement below is correct?

A. A member without any access modifier is default to private

B. A member without any access modifier is default to public

C. A member without any access modifier is default to protected

D. Classes in the same package can access members without any access modifier.

Note

Java 中的访问权限是 public、protected、default 和 private。其中 default 具有包访问权


Question

Q 2

image.png

Week 3 | Class

  • What will happen if you try to compile and execute B’s main() method? ( )

    class A {
        int i;
        A(int i) {    this.i = i * 2;    }
    }
    class B extends A {
        public static void main(String[] args) {
            B b = new B(2);
        }
        B(int x) {
            System.out.println(x);
        }
    }
    

    D. This code will not compile

Note

class A中只有一个带参数的初始化函数,必须要显式调用!(如果删除则可以通过编译)


Week 5 容器

Question

Given list an object of ArrayList, which code below for //todo delete can remove an element in the list correctly and safely?

Iterator it = list.iterator();

int index = 0;

while (it.hasNext()){

Object obj = it.next();

if (needDelete(obj)) { // returns Boolean for removing or not

//todo delete

}

index ++;

}

A. list.remove(obj);

B. list.remove(index);

C. list.remove(it.next());

D. it.remove();

Question

For code below:

ArrayList<Integer> a = new ArrayList<Integer>();

ArrayList<Double> b = new ArrayList<Double>();

Which statement below is NOT correct?

A. a.getClass().equals(b.getClass()) is true

B. a.getClass() == b.getClass() is true

C. a instanceof ArrayList is true

D. a.getClass() == b.getClass() is false

Exception

Question

Suppose there is no file Hello.txt in the current directory. Run the program:

import java.io.*;

public class ABC {

public static void main(String argv[]) throws Exception {

ABC m=new ABC();

System.out.println(m.ff());

}



public int ff() {

try {

FileInputStream dis=new FileInputStream("Hello.txt");

} catch (FileNotFoundException fne) {

System.out.print("No such file found, ");

throw fne;

} finally {

System.out.print("Doing finally, ");

}

return 0;

}

}

A.

No such file found,

B.

No such file found ,0

C.

No such file found, Doing finally,

D.

No such file found, Doing finally, 0

Note

IO

image-20241108225943443

image-20241108225918666

image-20241108230027881

  • read() 返回 int,因为需要返回 -1 表示文件结束。

Week 10 | Thread & 同步问题

image-20241202163347637

image-20241121234950928

image-20241121231604850

image-20241121234400292

  1. 广播的实现
  2. Send 的效率问题
    1. 消息队列
      1. 经典的consumer - producer
import java.io.*;
import java.net.*;
import java.util.concurrent.*;
import java.nio.channels.*;
import java.nio.*;
import java.util.*;

public class Server {
    // 使用CopyOnWriteArrayList保证线程安全,适合读多写少的场景
    private final CopyOnWriteArrayList<ClientHandler> clients = new CopyOnWriteArrayList<>();
    // 使用线程池处理客户端连接和消息
    private final ExecutorService executorService = Executors
            .newFixedThreadPool(Runtime.getRuntime().availableProcessors());
    // 使用NIO的Selector实现非阻塞IO
    private Selector selector;
    // 消息队列,用于异步处理消息广播
    private final BlockingQueue<String> messageQueue = new LinkedBlockingQueue<>();

    class ClientHandler {
        private SocketChannel socketChannel;
        private String clientId;

        public ClientHandler(SocketChannel socketChannel) {
            this.socketChannel = socketChannel;
            this.clientId = UUID.randomUUID().toString();
        }

        public void sendMessage(String message) {
            try {
                ByteBuffer buffer = ByteBuffer.wrap(message.getBytes());
                socketChannel.write(buffer);
            } catch (IOException e) {
                removeClient(this);
            }
        }
    }

    public void start(int port) {
        try {
            // 创建ServerSocketChannel
            ServerSocketChannel serverChannel = ServerSocketChannel.open();
            serverChannel.bind(new InetSocketAddress(port));
            serverChannel.configureBlocking(false);

            // 创建Selector
            selector = Selector.open();
            serverChannel.register(selector, SelectionKey.OP_ACCEPT);

            // 启动消息处理线程
            startMessageProcessor();

            while (true) {
                selector.select();
                Set<SelectionKey> selectedKeys = selector.selectedKeys();
                Iterator<SelectionKey> iter = selectedKeys.iterator();

                while (iter.hasNext()) {
                    SelectionKey key = iter.next();
                    iter.remove();

                    if (key.isAcceptable()) {
                        handleAccept(serverChannel);
                    } else if (key.isReadable()) {
                        handleRead(key);
                    }
                }
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    private void handleAccept(ServerSocketChannel serverChannel) throws IOException {
        SocketChannel clientChannel = serverChannel.accept();
        clientChannel.configureBlocking(false);
        clientChannel.register(selector, SelectionKey.OP_READ);

        ClientHandler client = new ClientHandler(clientChannel);
        clients.add(client);
    }

    private void handleRead(SelectionKey key) throws IOException {
        SocketChannel channel = (SocketChannel) key.channel();
        ByteBuffer buffer = ByteBuffer.allocate(1024);
        int numRead = -1;

        try {
            numRead = channel.read(buffer);
        } catch (IOException e) {
            key.cancel();
            channel.close();
            return;
        }

        if (numRead == -1) {
            key.cancel();
            channel.close();
            return;
        }

        buffer.flip();
        byte[] data = new byte[buffer.remaining()];
        buffer.get(data);
        String message = new String(data);

        // 将消息放入队列异步处理
        messageQueue.offer(message);
    }

    private void startMessageProcessor() {
        executorService.submit(() -> {
            while (true) {
                try {
                    String message = messageQueue.take();
                    broadcast(message);
                } catch (InterruptedException e) {
                    Thread.currentThread().interrupt();
                    break;
                }
            }
        });
    }

    private void broadcast(String message) {
        for (ClientHandler client : clients) {
            executorService.submit(() -> client.sendMessage(message));
        }
    }

    private void removeClient(ClientHandler client) {
        clients.remove(client);
        try {
            client.socketChannel.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        Server server = new Server();
        server.start(5382);
    }
}