diff --git a/01:概述/01.md b/01:概述/01.md index 665795f..65268c4 100644 --- a/01:概述/01.md +++ b/01:概述/01.md @@ -12,25 +12,24 @@ 如果从C语言出发,需要建立一个结构体来描述圆。 -```c {.line-numbers} +```c struct Circle { - float x; // X坐标 - float y; // Y坐标 - float r; // 半径 + float x; // X坐标 + float y; // Y坐标 + float r; // 半径 }; ``` 上面只是定义了一个结构体(抽象,代表所有圆),但是还不存在一个真正的圆的实例(具体的对象),因此需要定义一个变量,其类型是Circle这个结构体。 -```c {.line-numbers} +```c struct Circle c1; ``` 如果要改变c1这个圆结构变量的值我们有两种方法,首先是直接赋值。 - -```c {.line-numbers} +```c struct Circle c1; c1.x = 0; c1.y = 1; @@ -41,7 +40,7 @@ struct Circle c1; 另外还可以用函数的方式来改变结构体的值。 -```c {.line-numbers} +```c /** * 给圆设定初始值 * @param c 圆结构的指针 @@ -58,7 +57,7 @@ void setCircle(struct Circle *c, float x, float y, float r) { 另外我们一次性的打印整个圆的数据,这样我们最好需要一个函数。 -```c {.line-numbers} +```c /** * 打印 * @param c @@ -70,7 +69,7 @@ void printCircle(struct Circle *c) { 把所有代码放在一起: -```c {.line-numbers} +```c #include #include @@ -117,22 +116,22 @@ int main() { ### 1.2. C语言描述圆的总结 -1. 圆的结构体描述是一个抽象的描述,因为结构体是类型定义,并没有与其对应的变量;这儿描述和后面的Java类的定义很像。 -2. 声明一个结构体变量后就有一个真实的内存变量来描述一个特定的圆了,这个变量是圆(抽象描述的)一个实例(实体),也叫做对象;这个概念和Java的对象非常相似; -3. 通过点(.)或者(->)可以操作结构体变量(对象)内部的成员,这点和Java对象的操作也非常相识,只不过Java都是用点(.); +1. 圆的结构体描述是一个抽象的描述,因为结构体是类型定义,并没有与其对应的变量;这儿描述和后面的Java类的定义很像。 +2. 声明一个结构体变量后就有一个真实的内存变量来描述一个特定的圆了,这个变量是圆(抽象描述的)一个实例(实体),也叫做对象;这个概念和Java的对象非常相似; +3. 通过点(.)或者(->)可以操作结构体变量(对象)内部的成员,这点和Java对象的操作也非常相识,只不过Java都是用点(.); 这里我们发现,操作结构体的函数是独立定义的,并且需要把结构体变量作为参数进行传递。圆这个结构体是死的,需要外部的函数来对其进行操作。如果把圆这个结构体变量和生活中的对象联系起来,会有些有趣的事情。 例如,人这个对象有固有的特性(性别、年龄、身高、体重。。。)这些和圆的坐标和半径对应,都表示一个对象的特性;但是人可以说话,可以运动(动作、行为,相当于函数);而结构体变量的圆不能自己动作,只能依靠外部的函数进行动作。如果圆可以自己动作,例如让圆这个对象自己打印输出,这样不就可以和生活中的对象联系起来了? -```c {.line-numbers} +```c struct Circle { - float x; // X坐标 - float y; // Y坐标 - float r; // 半径 + float x; // X坐标 + float y; // Y坐标 + float r; // 半径 function void print(){ - printf("The position x=%f, y=%f, radius=%f!\n", c->x, c->y, c->r); + printf("The position x=%f, y=%f, radius=%f!\n", c->x, c->y, c->r); } }; struct Circle c1; @@ -151,7 +150,7 @@ c1.print(); // C语言无法做到 接下来我们使用java来实现上述的功能,这里主要是要把相关的函数作为结构体的一个部分。 -```c {.line-numbers} +```c /** * 定义Circle类 * @@ -159,19 +158,19 @@ c1.print(); // C语言无法做到 * */ class Circle { - float x; - float y; - float r; - - public void setCircle(float x, float y, float r) { - this.x = x; - this.y = y; - this.r = r; - } - - public void printCircle() { - System.out.printf("The position x=%f, y=%f, radius=%f!\n", x, y, r); - } + float x; + float y; + float r; + + public void setCircle(float x, float y, float r) { + this.x = x; + this.y = y; + this.r = r; + } + + public void printCircle() { + System.out.printf("The position x=%f, y=%f, radius=%f!\n", x, y, r); + } } /** @@ -182,28 +181,28 @@ class Circle { */ public class Circle_Test { - public static void main(String[] args) { - Circle circle = new Circle(); - circle.x=0; - circle.y=1; - circle.r=11.2F; - circle.printCircle(); - circle.setCircle(1, 1, 3.3F); - circle.printCircle(); - } + public static void main(String[] args) { + Circle circle = new Circle(); + circle.x=0; + circle.y=1; + circle.r=11.2F; + circle.printCircle(); + circle.setCircle(1, 1, 3.3F); + circle.printCircle(); + } } ``` 看有什么区别? -1. Java定义一个类是如何的? -2. Java中13~16行中的this代表什么? -3. Java的语法和C有什么区别? -4. 注意C中33行和Java中32行代码,代表什么意思? -5. 注意C中给结构体指针成员赋值与Java中对象成员赋值的区别。 -6. Java中类包含了函数,这是C中结构体不具备的能力。 -7. C和Java都有主函数Main,他们的写法有什么不一样? +1. Java定义一个类是如何的? +2. Java中13~16行中的this代表什么? +3. Java的语法和C有什么区别? +4. 注意C中33行和Java中32行代码,代表什么意思? +5. 注意C中给结构体指针成员赋值与Java中对象成员赋值的区别。 +6. Java中类包含了函数,这是C中结构体不具备的能力。 +7. C和Java都有主函数Main,他们的写法有什么不一样? #### 1.3.1. 编译和运行Java @@ -211,11 +210,10 @@ public class Circle_Test { 命令行中执行: -```bash {.line-numbers} +```bash javac Circle_Test.java ``` - 发现生成了两个文件 ![](img/20230302094127.png) @@ -223,13 +221,13 @@ javac Circle_Test.java 接下来使用命令行 -```bash {.line-numbers} +```bash java Circle_Test ``` 运行刚刚编译后的代码,得到结果 -```bash {.line-numbers} +```bash The position x=0, y=1, radius=11.200000! The position x=1, y=1, radius=3.300000! ``` @@ -270,9 +268,9 @@ The position x=1, y=1, radius=3.300000! #### 2.3.1. JDK、API、IDE -1. API: application programming interface,应用编程接口。Java提供了一套最基本的类,包括文件操作、网络操作、数学运算、输出输出、数据结构等,用来简化你的开发。这些其实是最基本的数据结构和操作函数,只不过Java是按照类(class)的方式来组织,的;类比C语言中的标准函数库是按照头文件和C文件来组织的。 -2. JDK: Java Developing Kit,是Java的开发套件。类比C语言,可以看作是编译器、链接器这些命令行工具。不同的是,Java没有链接,只有编译工具(Javac);另外Java编译后的代码(.class文件,有叫做bytecode)不能直接运行,需要使用java虚拟机(java)运行。 -3. IDE: Integrated Development Environment,这是集成开发环境。一般人不会用写字板和命令行行(Jdk)来开发Java,因为效率太低了。  +1. API: application programming interface,应用编程接口。Java提供了一套最基本的类,包括文件操作、网络操作、数学运算、输出输出、数据结构等,用来简化你的开发。这些其实是最基本的数据结构和操作函数,只不过Java是按照类(class)的方式来组织,的;类比C语言中的标准函数库是按照头文件和C文件来组织的。 +2. JDK: Java Developing Kit,是Java的开发套件。类比C语言,可以看作是编译器、链接器这些命令行工具。不同的是,Java没有链接,只有编译工具(Javac);另外Java编译后的代码(.class文件,有叫做bytecode)不能直接运行,需要使用java虚拟机(java)运行。 +3. IDE: Integrated Development Environment,这是集成开发环境。一般人不会用写字板和命令行行(Jdk)来开发Java,因为效率太低了。  ## 3. JAVA开发环境 @@ -312,8 +310,8 @@ Eclipse 是一个开放的框架,基于Eclipse框架的有很多不同语言 在弹出的对话框中主要是: -1. Project name:项目名称,建议使用英文,命名规则可以参考C语言的变量命名规则; -2. Use an execution environment JRE:Java虚拟机的版本,请一定选择1.8这个版本,因为我们书中的例子是按照1.8版本的。 +1. Project name:项目名称,建议使用英文,命名规则可以参考C语言的变量命名规则; +2. Use an execution environment JRE:Java虚拟机的版本,请一定选择1.8这个版本,因为我们书中的例子是按照1.8版本的。 ==项目名称最好使用英文,符合C语言变量名命名规则。因为项目名称直接对应到Workspace文件夹中的一个子文件夹,中文可能会造成意想不到的问题。== @@ -345,40 +343,41 @@ Eclipse 是一个开放的框架,基于Eclipse框架的有很多不同语言 ![](img/20230302095740.png) -1. Name:类名,上面名称的规则。在Java中,一个类**一般**是一个文件(一个文件可以包含多个类,一般不建议)。一个类的定义如同C中结构体类型的定义。Java推荐类名的首字母大写; -2. Modifiers:可视等级,这个以后解释,这里选择 public 就可以了; -3. Superclass:父类,这个以后解释,保持缺省; -4. public static void main(String[] args):如果C语言需要一个main的主函数作为启动入口,如果你需要,请勾选,会自动给你建立一个空的main函数。因为我们需要这个类可以运行,因此勾选。 +1. Name:类名,上面名称的规则。在Java中,一个类**一般**是一个文件(一个文件可以包含多个类,一般不建议)。一个类的定义如同C中结构体类型的定义。Java推荐类名的首字母大写; +2. Modifiers:可视等级,这个以后解释,这里选择 public 就可以了; +3. Superclass:父类,这个以后解释,保持缺省; +4. public static void main(String[] args):如果C语言需要一个main的主函数作为启动入口,如果你需要,请勾选,会自动给你建立一个空的main函数。因为我们需要这个类可以运行,因此勾选。 完成后,会打开一个新的类: -```java {.line-numbers} +```java package first; public class Hello { - public static void main(String[] args) { - // TODO Auto-generated method stub + public static void main(String[] args) { + // TODO Auto-generated method stub - } + } } ``` 接下来我们修改这个类框架,主要是修改main函数部分,结果如下: -```java {.line-numbers} +```java package first; public class Hello { - public static void main(String[] args) { - System.out.println("Hello world!"); + public static void main(String[] args) { + System.out.println("Hello world!"); - } + } } ``` + 保存(ctl+s)后,可以运行看看结果: ![](img/20230302095908.png) @@ -406,13 +405,14 @@ package first; ```java public class Hello { - public static void main(String[] args) { - System.out.println("Hello world!"); + public static void main(String[] args) { + System.out.println("Hello world!"); - } + } } ``` + 注意,第一行并没有 package 这个申明。在看看文件系统: ![](img/20230302100109.png) @@ -429,24 +429,25 @@ Hello.java 这个文件的确在src文件夹中。 我们再来回顾以下上面最简单的Hello world: -```java {.line-numbers} +```java package first; public class Hello { - public static void main(String[] args) { - System.out.println("Hello world!"); + public static void main(String[] args) { + System.out.println("Hello world!"); - } + } } ``` -1. package:包描述,该源代码在那个包,对应src文件夹中的字文件夹。如果有多个层级的包(src子文件夹中的子文件夹),使用点(.)来分割包名。尝试一下在包下面建立子包后package是如何描述的。package不是必须的,如果java文件在src中,package可以省略,这时表示缺省包。 -2. 一般来说,一个Java文件包含一个类(Class)结构,如果有多个并列的Class结构,只能有一个描述成public。关于public的含义后面会讲到。 -3. 类名(或者描述成public的类名),这个例子是Hello,必须和主文件名一致。 -4. Java中的变量、常量、函数等结构都属于一个类(Class),在类外面定义非类(Class)的结构都是非法的。 -5. 如果要代码运行,需要一个主函数main,且这个函数的写法是固定的(上面代码第5行),且main函数所在的类必须是public修饰的。 -6. 一个工程中可以有多个类,多个主函数,这个和C是有区别的。C中一个工程只能有一个主函数。 + +1. package:包描述,该源代码在那个包,对应src文件夹中的字文件夹。如果有多个层级的包(src子文件夹中的子文件夹),使用点(.)来分割包名。尝试一下在包下面建立子包后package是如何描述的。package不是必须的,如果java文件在src中,package可以省略,这时表示缺省包。 +2. 一般来说,一个Java文件包含一个类(Class)结构,如果有多个并列的Class结构,只能有一个描述成public。关于public的含义后面会讲到。 +3. 类名(或者描述成public的类名),这个例子是Hello,必须和主文件名一致。 +4. Java中的变量、常量、函数等结构都属于一个类(Class),在类外面定义非类(Class)的结构都是非法的。 +5. 如果要代码运行,需要一个主函数main,且这个函数的写法是固定的(上面代码第5行),且main函数所在的类必须是public修饰的。 +6. 一个工程中可以有多个类,多个主函数,这个和C是有区别的。C中一个工程只能有一个主函数。 后续的学习中会学到更多关于语法方面的内容,这里只做一个初步的了解。 @@ -454,9 +455,9 @@ public class Hello { 这里讨论指针比较早,C中的指针是一个非常重要的概念,在Java中关于类似C指针的概念有区别,这里需要进行说明,请大家先有个印象。 -1. 首先Java中不存在C中指针的术语,与之对应的是引用(reference)。引用和指针其实是相似的; -2. Java中有动态分配内存(如使用 new 操作符),但是不需要回收,Java虚拟机会自动回收; -3. Java中有引用的操作(类似C中取指针地址)也是使用(&)操作符。 +1. 首先Java中不存在C中指针的术语,与之对应的是引用(reference)。引用和指针其实是相似的; +2. Java中有动态分配内存(如使用 new 操作符),但是不需要回收,Java虚拟机会自动回收; +3. Java中有引用的操作(类似C中取指针地址)也是使用(&)操作符。 关于引用的具体内容我们放在函数和类中进行讲解。 @@ -468,9 +469,9 @@ public class Hello { 语法错误,编译的时候发现,例如: -```java {.line-numbers} +```java public class ShowSyntaxErrors { - public static main(String[] args) { + public static main(String[] args) { System.out.println("Welcome to Java); } } @@ -480,11 +481,11 @@ public class ShowSyntaxErrors { 运行时错误,运行时产生的错误,例如空对象操作、0作为除数、文件不存在等。运行时错误是我们需要特别关注的,因为会导致程序的流程异常,产生意想不到的结果。例如下面这个例子: -```java {.line-numbers} +```java public class ShowRuntimeErrors { - public static void main(String[] args) { - System.out.println(1 / 0); - } + public static void main(String[] args) { + System.out.println(1 / 0); + } } ``` @@ -494,12 +495,12 @@ public class ShowRuntimeErrors { 逻辑错误,简单说来就是语法没错、也没有运行时错误(异常),但是结果错误了。通常是编程者的算法错了,例如: -```java {.line-numbers} +```java public class ShowLogicErrors { - public static void main(String[] args) { - System.out.println("Celsius 35 is Fahrenheit degree "); - System.out.println((9 / 5) * 35 + 32); - } + public static void main(String[] args) { + System.out.println("Celsius 35 is Fahrenheit degree "); + System.out.println((9 / 5) * 35 + 32); + } } ``` @@ -513,22 +514,22 @@ public class ShowLogicErrors { 代码如下: -```java {.line-numbers} +```java public class Debug { - public static void main(String[] args) { - String outStr = ""; - for (int i = 0; i < 10; i++) { - outStr += i + ","; - System.out.println(outStr); - } - } + public static void main(String[] args) { + String outStr = ""; + for (int i = 0; i < 10; i++) { + outStr += i + ","; + System.out.println(outStr); + } + } } ``` 输出结果是: -``` {.line-numbers} +```{.line-numbers} 0, 0,1, 0,1,2, @@ -553,15 +554,15 @@ eclipse的调试界面是单独的布局,第一次运行调试会提示进入 ![](img/20230302100457.png) -1. 中断行:高亮显示,表示当前需要执行的行(还未执行); -2. 调试窗口:当前的变量值;断电设置;表达式计算等; -3. 输出窗口:和运行是一样的,不过这个输出要更具代码的具体执行位置进行输出; -4. 运行按钮:控制代码执行的按钮: - 1. Step over(F6):单步执行(跳过),运行到当前下一行代码,如果有函数,不进入函数; - 2. Step into(F5):如果高亮代码有函数,进入函数执行(单步执行); - 3. Step return(F7):从函数中返回到被调用位置(单步执行); - 4. Resume(F8):执行直到下一个个中断位置; - 5. Terminate(Ctl+F2):终止运行,无论是否运行完成,立即终止并退出。 +1. 中断行:高亮显示,表示当前需要执行的行(还未执行); +2. 调试窗口:当前的变量值;断电设置;表达式计算等; +3. 输出窗口:和运行是一样的,不过这个输出要更具代码的具体执行位置进行输出; +4. 运行按钮:控制代码执行的按钮: + 1. Step over(F6):单步执行(跳过),运行到当前下一行代码,如果有函数,不进入函数; + 2. Step into(F5):如果高亮代码有函数,进入函数执行(单步执行); + 3. Step return(F7):从函数中返回到被调用位置(单步执行); + 4. Resume(F8):执行直到下一个个中断位置; + 5. Terminate(Ctl+F2):终止运行,无论是否运行完成,立即终止并退出。 尝试不断按F8,观察调试窗口、输出窗口的变化。 @@ -595,12 +596,12 @@ eclipse的调试界面是单独的布局,第一次运行调试会提示进入 ## 6. 本章节重点 -1. JDK、API、IDE: P11-1.6 -2. Java语言的基本结构:P12-1.7 -3. 建立、编译、执行Java:P15-1.8 -4. 错误类型:P20-1.10 +1. JDK、API、IDE: P11-1.6 +2. Java语言的基本结构:P12-1.7 +3. 建立、编译、执行Java:P15-1.8 +4. 错误类型:P20-1.10 补充内容: -1. 包和文件夹的关系;不考,但是掌握后有帮助。 -2. Eclipse的使用,特别是调试(Debug);不考,但是非常有用。 \ No newline at end of file +1. 包和文件夹的关系;不考,但是掌握后有帮助。 +2. Eclipse的使用,特别是调试(Debug);不考,但是非常有用。 \ No newline at end of file diff --git a/02:Java语言基础/01.md b/02:Java语言基础/01.md index 93b4e3b..382c603 100644 --- a/02:Java语言基础/01.md +++ b/02:Java语言基础/01.md @@ -2,7 +2,6 @@ Java基础(第二章) ## 1. 如何打印输出 - **这一章中,我们使用面向过程(C语言的方式)来了解Java的基本语法。虽然说是和面向对象无关,但是我们需要一个函数入口(main)函数,这个函数上一章讲过,有固定的格式,且这个函数必须定义在一个Public类中。** ### 1.1. ComputeArea.java @@ -11,21 +10,21 @@ Java基础(第二章) ComputeArea.java -```java {.line-numbers} +```java public class ComputeArea { - public static void main(String[] args) { - double radius = 0.0; // Declare radius - double area; // Declare area + public static void main(String[] args) { + double radius = 0.0; // Declare radius + double area; // Declare area - // Assign a radius - radius = 20.0; // New value is radius + // Assign a radius + radius = 20.0; // New value is radius - // Compute area - area = radius * radius * 3.14159; + // Compute area + area = radius * radius * 3.14159; - // Display results - System.out.println("The area for the circle of radius " + radius + " is " + area); - } + // Display results + System.out.println("The area for the circle of radius " + radius + " is " + area); + } } ``` @@ -51,15 +50,15 @@ public class ComputeArea { 再总结一下我们学的的: -1. main函数的固定写法; -2. main函数必须在一给public类中定义; -3. 字符串的表示和连接; -4. 如何使用打印函数。 +1. main函数的固定写法; +2. main函数必须在一给public类中定义; +3. 字符串的表示和连接; +4. 如何使用打印函数。 ### 1.2. 问题 -1. main函数前面的static是什么意思? -2. main函数的参数 String\[\] args 是什么意思? +1. main函数前面的static是什么意思? +2. main函数的参数 String\[\] args 是什么意思? 后面回讲到上述两个问题,你可以先研究一下。 @@ -71,18 +70,18 @@ C使用scanf函数配合一套字符串规则实现各种数据类型的输入 ### 2.1. ComputeAreaWithConsoleInput.java -```java {.line-numbers} +```java import java.util.Scanner; // Scanner is in the java.util package public class ComputeAreaWithConsoleInput { public static void main(String[] args) { // Create a Scanner object Scanner input = new Scanner(System.in); - + // Prompt the user to enter a radius System.out.print("Enter a number for radius: "); double radius = input.nextDouble(); - + // Compute area double area = radius * radius * 3.14159; // Display result @@ -91,6 +90,7 @@ public class ComputeAreaWithConsoleInput { } } ``` + 注意第一行的 import ..... 这个语句,这是把Scanner所在的包引入进来,相当于C的include语句。如果不写第一行,Eclipse会在地6行的 Scanner位置错误(红色下划线),这是只需要把鼠标移动到 Scanner 上面,会出现quike fix 的建议,如下图。这时只需要点击:Import 'Scanner' (java.util) 就可以了,Eclipse会自动加上import语句。 ![](img/20230303153514.png) @@ -113,14 +113,13 @@ public class ComputeAreaWithConsoleInput { ## 3. 标识符、变量、常量 - ### 3.1. 与C语言一致的 -1. Identifiers:标识符 -2. Variables:变量 -3. Declaring Variables:变量声明 -4. Assignment Statements:赋值语句 -5. Declaring and Initializing in One Step:变量声明与赋值用一行语句 +1. Identifiers:标识符 +2. Variables:变量 +3. Declaring Variables:变量声明 +4. Assignment Statements:赋值语句 +5. Declaring and Initializing in One Step:变量声明与赋值用一行语句 上面这些和C保持一致,我们只看看常量的声明。 @@ -132,7 +131,7 @@ final datatype CONSTANTNAME = VALUE;   例如: -```java {.line-numbers} +```java final int PI = 3.14; final String ERROR="ERROR"; ``` @@ -141,23 +140,23 @@ final String ERROR="ERROR"; 其实只要是符合标识符的规则就是合法的变量名、常量名、函数名、类名;但是为了阅读方便,大家有个对命名大小写的共识,一般是这样: -1. 常量:全部大写; -2. 变量:首字母小写; -3. 类名(接口名):首字母大写,如果有多个单词,每个单词的首字母大写; -4. 函数名:首字母小写。 +1. 常量:全部大写; +2. 变量:首字母小写; +3. 类名(接口名):首字母大写,如果有多个单词,每个单词的首字母大写; +4. 函数名:首字母小写。 ## 4. 基本数据类型 ### 4.1. 数值类型 -| 类型 | 存储大小(bit) | 说明 | -|--------|-----------|----| -| byte | 8 | | -| short | 16 | | -| int | 32 | | -| long | 64 | | -| float | 32 | | -| double | 64 | +| 类型 | 存储大小(bit) | 说明 | +| ------ | --------- | --- | +| byte | 8 | | +| short | 16 | | +| int | 32 | | +| long | 64 | | +| float | 32 | | +| double | 64 | | C语言中有uint、ulong等带u开头的整形类型,Java中有吗? @@ -175,14 +174,15 @@ Scanner对象有相应读取这些数据类型的函数,请自己尝试一下 字面量的含义是语句中直接写入整形数值或者是浮点数值的时候,这个数值表达的数据类型或者是赋值规则。例如: -```java {.line-numbers} - byte a = 100; - byte b = 200; - int c = 300; - int d = 5000; - float e = 33.3; - double f = 33.3; +```java + byte a = 100; + byte b = 200; + int c = 300; + int d = 5000; + float e = 33.3; + double f = 33.3; ``` + 上述代码中,第二行代码是非法的;第五行代码也是非法的;为什么? 规则: @@ -190,9 +190,10 @@ Scanner对象有相应读取这些数据类型的函数,请自己尝试一下 1. 对于整形赋值,只要字面量表达的数值不超过变量类型的取值范围就是合法的; 2. 对于带小数的字面量,缺省的类型是double,因此如果赋值给float的类型的变量会出错,正确的做法是在字面量后面加上F。 -```java {.line-numbers} +```java float e = 33.3F; ``` + 科学计数法:例如:1.23456e-2 ### 4.4. 扩展赋值语法:Augmented Assignment Operators @@ -215,9 +216,9 @@ float e = 33.3F; ## 5. 本章重点 -1. 如何打印输出:有很多同名的打印函数(函数重载overload);print,println,printf函数;与C有什么区别; -2. 字符串与其他类型连接通过+号; -3. 如何读取键盘输入,以及赋值。注意不同的类型使用不同的函数; -4. 如何判断是否还有没有读取的数据(hasNext函数XXX); -5. 基本数据类型:全部都是小写,以后会学习对应大写的包装类型;都是有符号的; -6. 如何声明常量(final);7. 类型转换以及规则。 +1. 如何打印输出:有很多同名的打印函数(函数重载overload);print,println,printf函数;与C有什么区别; +2. 字符串与其他类型连接通过+号; +3. 如何读取键盘输入,以及赋值。注意不同的类型使用不同的函数; +4. 如何判断是否还有没有读取的数据(hasNext函数XXX); +5. 基本数据类型:全部都是小写,以后会学习对应大写的包装类型;都是有符号的; +6. 如何声明常量(final);7. 类型转换以及规则。 diff --git a/02:Java语言基础/02.md b/02:Java语言基础/02.md index 9c41525..003f0c8 100644 --- a/02:Java语言基础/02.md +++ b/02:Java语言基础/02.md @@ -28,18 +28,17 @@ Java的布尔类型和C保持一致,基本的运算符也和C保持一致。 例如:0B 和 0A 代表两个byte,转换成位是: -| 字节1(0x08) | 字节2(0x07) | 说明 | -| ----------- | ----------- | ------------ | -| 0000-1000 | 0000-0111 | 对应的二进制 | +| 字节1(0x08) | 字节2(0x07) | 说明 | +| --------- | --------- | ------ | +| 0000-1000 | 0000-0111 | 对应的二进制 | -| 位操作 | 结果 | 说明 | -| ------------ | --------- | ------------------------------ | +| 位操作 | 结果 | 说明 | +| ------------ | --------- | --------------- | | 0x0A & 0x0B | 0000-1000 | 按照每一位对应进行与运算的结果 | | 0x0A \| 0x0B | 0000-1111 | 按照每一位对应进行或运算的结果 | **位操作作为了解。** - ### 1.4. 选择语句 略 @@ -54,7 +53,7 @@ Java的布尔类型和C保持一致,基本的运算符也和C保持一致。 这个方式在书上没有讲,在考试中可能会遇到这种情况。 -```java {.line-numbers} +```java import java.util.Scanner; /** @@ -64,20 +63,20 @@ import java.util.Scanner; * */ public class LoopRead { - public static void main(String[] args) { - Scanner scanner = new Scanner(System.in); - System.out.println("Please input a sequence of integer, divided by space."); - int sum = 0; - while (true) { - if (scanner.hasNextInt()) { - sum += scanner.nextInt(); - } else { - break; - } - } - scanner.close(); - System.out.println("The sum is: " + sum); - } + public static void main(String[] args) { + Scanner scanner = new Scanner(System.in); + System.out.println("Please input a sequence of integer, divided by space."); + int sum = 0; + while (true) { + if (scanner.hasNextInt()) { + sum += scanner.nextInt(); + } else { + break; + } + } + scanner.close(); + System.out.println("The sum is: " + sum); + } } ``` @@ -93,9 +92,9 @@ public class LoopRead { 本章的内容中的大部分技术要点都与C保持一致,因此之需要掌握几个例子就可以了: -1. AdditionQuiz -2. SubtractionQuiz -3. ComputeAndInterpretBMI -4. ComputeTax -5. ChineseZodiac -6. 本页的补充例子,循环读取输入 \ No newline at end of file +1. AdditionQuiz +2. SubtractionQuiz +3. ComputeAndInterpretBMI +4. ComputeTax +5. ChineseZodiac +6. 本页的补充例子,循环读取输入 \ No newline at end of file diff --git a/02:Java语言基础/03.md b/02:Java语言基础/03.md index 95886cc..e0e2f2c 100644 --- a/02:Java语言基础/03.md +++ b/02:Java语言基础/03.md @@ -6,28 +6,28 @@ 对标准库而言,我们最关心的是什么?一般是三个问题: -1. 基本数据结构 -2. 常量 -3. 函数 +1. 基本数据结构 +2. 常量 +3. 函数 这一章暂时不设计基数据结构,数据结构在另外的章节专门讲解。这里的重点是: -1. 常量 PI -2. 取整函数 -3. 随机数函数:Rounding Methods -4. min, max, abs, and random Methods +1. 常量 PI +2. 取整函数 +3. 随机数函数:Rounding Methods +4. min, max, abs, and random Methods ### 1.1. 取整函数 -1. double ceil(double x) x round up to its nearest integer. This integer is  returned as a double value. -2. double floor(double x) x is rounded down to its nearest integer. This integer is  returned as a double value. -3. double rint(double x) x is rounded to its nearest integer. If x is equally close to two integers, the even one is returned as a double. -4. int round(float x) Return (int)Math.floor(x+0.5). -5. long round(double x) Return (long)Math.floor(x+0.5).   +1. double ceil(double x) x round up to its nearest integer. This integer is  returned as a double value. +2. double floor(double x) x is rounded down to its nearest integer. This integer is  returned as a double value. +3. double rint(double x) x is rounded to its nearest integer. If x is equally close to two integers, the even one is returned as a double. +4. int round(float x) Return (int)Math.floor(x+0.5). +5. long round(double x) Return (long)Math.floor(x+0.5).   实例: -```java {.line-numbers} +```java Math.ceil(2.1) returns 3.0 Math.ceil(2.0) returns 2.0 Math.ceil(-2.0) returns –2.0 @@ -50,20 +50,20 @@ Math.round(-2.6) returns -3 ### 1.2. min, max, and abs -1. max(a, b)and min(a, b) Returns the maximum or minimum of two parameters. -2. abs(a) Returns the absolute value of the parameter. -3. random() Returns a random double value in the range \[0.0, 1.0).  - +1. max(a, b)and min(a, b) Returns the maximum or minimum of two parameters. +2. abs(a) Returns the absolute value of the parameter. +3. random() Returns a random double value in the range \[0.0, 1.0).  实例: -```java {.line-numbers} +```java Math.max(2, 3) returns 3 Math.max(2.5, 3) returns 3.0 Math.min(2.5, 3.6) returns 2.5 Math.abs(-2) returns 2 Math.abs(-2.1) returns 2.1 ``` + ### 1.3. The random Method `Math.random();` 函数生成一个0~1之间的double类型的数,包含0,但是不包含1。 @@ -78,22 +78,24 @@ char类型是有序的(类似整形),整形可以使用的运算符都可 ### 2.1. char和整形的转换 -```java {.line-numbers} +```java int i = 'a'; // Same as int i = (int)'a'; char c = 97; // Same as char c = (char)97; ``` ### 2.2. char的比较 -```java {.line-numbers} - if (ch >= 'A' && ch <= 'Z') - System.out.println(ch + " is an uppercase letter"); - else if (ch >= 'a' && ch <= 'z') - System.out.println(ch + " is a lowercase letter"); - else if (ch >= '0' && ch <= '9') - System.out.println(ch + " is a numeric character"); +```java + if (ch >= 'A' && ch <= 'Z') + System.out.println(ch + " is an uppercase letter"); + else if (ch >= 'a' && ch <= 'z') + System.out.println(ch + " is a lowercase letter"); + else if (ch >= '0' && ch <= '9') + System.out.println(ch + " is a numeric character"); ``` + ### 2.3. 字符类型的相关函数 + ![](img/20230303171401.png) 上述这些函数不能直接调用,而是在 Character 这个类中,因此如果你要调用上述的函数,需要使用 Character.xxx 这样的方式。所有的函数都是静态函数,静态函数,关于静态函数我们会在学习类的时候进行学习,目前我们用到的都是静态函数。 @@ -104,33 +106,36 @@ char c = 97; // Same as char c = (char)97; 字符串与C完全不一样,Java中的字符串的类型是 String 。注意,以前在Java中出现的类型都是小写开头的,这是第一个大写开头的,因此这个是一个类的类型,而不是普通类型。前面讲到过,类是抽象描述(类似结构体指针),如果来实例化(成为可操作的对象)需要使用new 操作生成一个对象。这里我们看看字符串是如何生成的。 -```java {.line-numbers} +```java public class Main { - public static void main(String[] args) { - String s1 = new String("Hello"); - System.out.println(s1); - } + public static void main(String[] args) { + String s1 = new String("Hello"); + System.out.println(s1); + } } ``` + 第三行代码是生成一个字符串对象,第四行是打印这个字符串。等等,这里和Hello world程序好像有所不同?对的,标准的对象生成应该是按照第三行的方式生成,不过因为String在程序中使用得太频繁了,因此如果写入字符串的字面量,编译器会自动构造一个对象,下面的代码也是正确的: -```java {.line-numbers} +```java public class Main { - public static void main(String[] args) { - String s1 = "Hello"; - System.out.println(s1); - } + public static void main(String[] args) { + String s1 = "Hello"; + System.out.println(s1); + } } ``` + 或者直接使用字面量作为参数: -```java {.line-numbers} +```java public class Main { - public static void main(String[] args) { - System.out.println("Hello"); - } + public static void main(String[] args) { + System.out.println("Hello"); + } } ``` + ### 3.2. 字符串操作 既然字符串是一个对象,那么就可以用对象的方式来操作字符串。在上述例子中,试试在 s1 这个变量后面输入一个点(.),Eclipse将提示该对象上面的属性和方法(有括号的是方法\\函数,没有括号的是属性\\对象中的变量)。 @@ -141,60 +146,64 @@ public class Main { 上述这些函数是在数据库对象上面进行调用的,例如: -```java {.line-numbers} +```java String sA = "Hello world!"; int len = sA.length(); char c = sA.charAt(1); String sB = sA.toUpperCase(); String sC = sA.trim(); ``` + 这里不可能解释所有字符串对象的函数,请参考在线文档。 #### 3.2.2. 字符串连接 字符串连接可以使用算术运算符(+),也可以使用对象函数: -```java {.line-numbers} - String sA = "Hello"; - String sB = "World"; - System.out.println(sA.concat(" ").concat(sB).concat("!")); - System.out.println(sA + " " + sB + "!"); +```java + String sA = "Hello"; + String sB = "World"; + System.out.println(sA.concat(" ").concat(sB).concat("!")); + System.out.println(sA + " " + sB + "!"); ``` + **注意,concat 函数返回的是字符串对象,因此concat函数后面可以使用任何的字符串对象的函数。** 另外,字符串可以和数值类型的变量或者是字面量进行连接: -```java {.line-numbers} +```java public class Main { - public static void main(String[] args) { - String sA = "Hello"; - System.out.println(sA + ":" + 3.14 + ":" + 1); - } + public static void main(String[] args) { + String sA = "Hello"; + System.out.println(sA + ":" + 3.14 + ":" + 1); + } } ``` + **其实,字符串可以和几乎所有类型的变量或者字面量进行连接,这一点我们在后面还会遇到。更多的原理要我们学了类的继承才能理解。** #### 3.2.3. 从控制台读取字符串 前面学习了如何从控制台读取数值类型的值,这里我们看看如何读取字符串/字符。 -```java {.line-numbers} +```java import java.util.Scanner; public class Main { - public static void main(String[] args) { - Scanner input = new Scanner(System.in); - - String s1 = input.next(); - String s2 = input.next(); - String s3 = input.next(); - - System.out.println("s1 is " + s1); - System.out.println("s2 is " + s2); - System.out.println("s3 is " + s3); - } + public static void main(String[] args) { + Scanner input = new Scanner(System.in); + + String s1 = input.next(); + String s2 = input.next(); + String s3 = input.next(); + + System.out.println("s1 is " + s1); + System.out.println("s2 is " + s2); + System.out.println("s3 is " + s3); + } } ``` + 运行一下这个代码,输入三个个字符串,字符串间需要使用空格分割,看看结果是什么? next() 是以空格区分字符串的,如果我们需要读取一段字符串,本身包含空格,这就需要使用 nextLine() 这个函数。试试把next() 改成 nextLine() 会是什么结果。nextLine()是以回车进行区分的。 @@ -225,16 +234,17 @@ next() 是以空格区分字符串的,如果我们需要读取一段字符串 这时我们需要使用到相应数值类型的类类型(包装类型),而不是简单的类型了。关于包装类型我们后面会讲到,这里先了解一下。 -```java {.line-numbers} +```java public class Main { - public static void main(String[] args) { - int a = Integer.parseInt("1"); - int b = Integer.valueOf("2"); - float c = Float.parseFloat("2.2"); - float d = Float.valueOf("2.2"); - } + public static void main(String[] args) { + int a = Integer.parseInt("1"); + int b = Integer.valueOf("2"); + float c = Float.parseFloat("2.2"); + float d = Float.valueOf("2.2"); + } } ``` + 这里 Interger 或者是 Float 是一个类,而非对象。我们把在类上面可以直接调用的方法(不需要实例化成对象),叫做静态方法。关于静态方法,后面学到类的时候还会讲解。 **parseXXX 方法和 valueOf 方法都可以返回相应字符串对应的数值类型(看你是在哪个类型上调用的),区别是 valueOf 返回的是包装类型,parseXXX返回的是基本类型。** @@ -245,52 +255,55 @@ public class Main { 有很多方式,最简单的是加上一个空字符串(没有内容的字符串),例如: -```java {.line-numbers} +```java String sA = 100 + ""; ``` + #### 3.2.8. 格式化字符串 格式化字符串如同C语言的printf方法,甚至连占位符和格式方法都一模一样。如果要输出一个字符串,可以使用 System.out.printf 方法。 -```java {.line-numbers} +```java public class Main { - public static void main(String[] args) { - System.out.printf("Hello, my name is %s, and %d years old.\n", "Danny", 20); - } + public static void main(String[] args) { + System.out.printf("Hello, my name is %s, and %d years old.\n", "Danny", 20); + } } ``` + 当然,如果不想输出字符串,而需要对中间的结果进行格式化应该如何? -```java {.line-numbers} +```java public class Main { - public static void main(String[] args) { - String a = String.format("Hello, my name is %s, and %d years old.\n", "Danny", 20); - System.out.print(a); - } + public static void main(String[] args) { + String a = String.format("Hello, my name is %s, and %d years old.\n", "Danny", 20); + System.out.print(a); + } } ``` + 下面是占位符列表: ![](img/20230303172243.png) ## 4. 本章重点 -1. 数学类:PI常量、取整、随机、最大、最小、绝对值等常用静态函数; -2. 字符类型: - 1. 字符类型声明 - 2. unicode(了解) - 3. 字符与整形的关系(与C一致) -3. 字符类型操作函数(静态函数) -4. 字符串(字符串是对象,不是基本类型) - 1. 声明与使用 - 2. 基本函数: - 1. length() - 2. charAt(index) - 3. concat(s1) - 4. toUpperCase() - 5. toLowerCase() - 6. trim() - 3. 字符串比较函数 - 4. 获取子串 - 5. 查找字符串 - 6. 字符串与数值的转换 \ No newline at end of file +1. 数学类:PI常量、取整、随机、最大、最小、绝对值等常用静态函数; +2. 字符类型: + 1. 字符类型声明 + 2. unicode(了解) + 3. 字符与整形的关系(与C一致) +3. 字符类型操作函数(静态函数) +4. 字符串(字符串是对象,不是基本类型) + 1. 声明与使用 + 2. 基本函数: + 1. length() + 2. charAt(index) + 3. concat(s1) + 4. toUpperCase() + 5. toLowerCase() + 6. trim() + 3. 字符串比较函数 + 4. 获取子串 + 5. 查找字符串 + 6. 字符串与数值的转换 \ No newline at end of file diff --git a/02:Java语言基础/05.md b/02:Java语言基础/05.md index 08900d0..d81d8a3 100644 --- a/02:Java语言基础/05.md +++ b/02:Java语言基础/05.md @@ -1,76 +1,369 @@ Java的函数和 C 类似,但因为Java是面向对象的语言,所以又有很多扩展的地方,我们这里主要对其特殊的地方进行讲解。 -## 静态函数与非静态函数 +**广义上讲,函数(function)与过程(procedure)的含义是一样的。在有些语言中(例如Pascal),函数是指有返回值的;过程是指没有返回值的。** +## 1. 函数 -到目前为止,我们学习的大多是静态函数。什么是静态函数?简单说来,静态函数是不依附对象就可以直接运行的函数。这个怎么解释?要了解静态与非静态函数的区别,首先需要知道什么是对象。前面说过,类是抽象描述的,如同类型定义;对象是类的一个真实变量(分配了具体的内存)。这种关系如果我们第一章的结构体指针类型和一个分配了存储空间的结构体指针变量。 +函数的定义和数学上的定义相似,可以理解为函数是一个抽象的算法,当传入一定的参数后,函数可以有一定的返回结果。当然函数不一定都需要返回,但是一定是包含一定的功能和算法,并且产生一定的效果。 -### 如何计算字符串的长度 +从更广泛的意义上说,一个应用程序或者是APP都可以说是一个复杂的函数,它们接受输入,通过计算产生一定的效果(输出)。 -接下来看看C的函数的语义。函数是具备高度抽象的计算功能,需要传入参数后返回或者不返回结果(对数据进行修改)。例如C语言如果需要确定字符串的长度,这个函数的原型(签名)是这样的: +### 1.1. 空函数 -```C {.line-numbers} +空函数在语法上将是存在的,例如下面一个Java的空函数: + +```java +void nullFunxtion(){ + // 没有算法,不返回,没有任何功能 +} +``` + +这个函数虽然在语法上没有问题,也可以通过编译,也能被其他的代码所调用,但是这个函数本身是没有意义的。因为没有算法,没有任何的效果。**这里我们不讨论空函数。** + +### 1.2. 没有返回值的函数 + +函数可以没有返回值,但是需要有算法,并且一定产生某些效果(影响)。 + +例如: + +```java +public class Test { + + static int sum = 0; + + public static void main(String[] args) { + add(10); + add(11); + System.out.print(sum); + } + + static void add(int i) { + sum = sum + i; + } +} +``` + +上面的函数 `static void add(int i)` 这个函数就是没有返回值的,但是函数中的算法有效果:对变量sum进行累加。当然,其他的一些没有返回的值的函数可能产程其他的效果:例如打印、文件输出、网络连接... 这些都是函数产生的影响,也是函数存在的意义。 + +### 1.3. 有返回值的函数 + +如果函数有返回值,需要确定返回值的类型,这一点和C语言也是一致的。任何的系统预定义类型,和自定义类型(Java中主要是类),都可以作为函数的返回类型。下面是一个简单的例子: + +```java +public class Test { + + public static void main(String[] args) { + System.out.print(add(1, 10)); + } + + static int add(int i, int j) { + return i + j; + } +} +``` + +print也是一个函数,它的输入参数刚好是add这个函数的返回值,因此这里的结果是11。 + +### 1.4. 函数返回多个值 + +上面两个例子说明函数有没有返回值都可能产生一定的效果。在某些情况下,一个函数可能会产生多种效果,例如C语言的fwrite(写文件)函数,他的效果是向文件写入一些数据,同时其返回值是表示写入的数据是否成功。也就是说这个函数同有两种结果(返回成功与否,同时写入数据)。 + +那一个函数是否可以有多个返回值?在C中,一个函数只能有一个返回值,但是可以通过一些技巧使其具有返回多个值的不同效果,例如: + +1. 传递参数的指针,这样在函数内部可以改变传入参数的值,这是指针传递; + +2. 使用多个全局变量,在函数内是可以改变这些全局变量的值; + +3. 使用结构体返回数据。 + +当然还有其他很多方法,使得函数可以返回多个值。 + +**注意:Java中没有指针的概念,所以也没有指针传递参数的用法。但是Java有引用,这是后面我们要讨论的问题。** + +Java没有指针,但是有加强的结构体(类-对象)。类与对象我们放在后面来讲,这里我们看一个通过“全局变量”返回多个值的例子: + +```java +public class Test { + + static int a = 0; + static int b = 0; + + public static void main(String[] args) { + System.out.printf("a=%d and b=%d\n", a, b); + changeAB(); + System.out.printf("a=%d and b=%d\n", a, b); + } + + static void changeAB() { + a = 100; + b = 200; + } +} +``` + +Java并没有真正意义上的“全局变量”,每个变量都被局限到一个类中,类好比一个盒子,把变量(目前是静态变量)、常量(常量都是静态的)和函数(目前是静态函数)包装起来了(封装特性)。在类(这个盒子中)可以自由调用这些静态变量、常量、静态函数。如果在这个盒子外边调用这个类的静态变量、常量、静态函数,就需要先找到这些静态成员的盒子(类),再调用,例如前面使用到的`Math.PI` 常量;`Math.abs(a,b)` 静态函数;上个例子中的 变量 a 和 b(静态变量)。 + +上个例子中的函数 changeAB 因为和静态变量在同一个空间(一个类中),因此在前面不用加上类名。当然加上类名也不错,例如: + +```java +public class Test { + + static int a = 0; + static int b = 0; + + public static void main(String[] args) { + System.out.printf("a=%d and b=%d\n", a, b); + changeAB(); + System.out.printf("a=%d and b=%d\n", a, b); + } + + static void changeAB() { + Test.a = 100; + Test.b = 200; + } +} +``` + +## 2. 静态与非静态(实例化) + +到目前为止,我们学习的大多是静态函数。什么是静态函数?简单说来,静态函数是不依附对象就可以直接运行的函数。这个怎么解释?要了解静态与非静态函数的区别,首先需要知道什么是对象。前面说过,类是抽象描述的,像C语言中的类型定义;对象是类的一个真实变量(分配了具体的内存)。这种关系我们第一章的结构体指针类型和一个分配了存储空间的结构体指针变量中进行了分析。 + +同样,静态变量也是不依赖对象存在的,在程序运行的时候就存在,而不需要创建和依附一个对象存在。 + +简单的理解,静态函数相当于C语言中的函数;静态变量相当于C语言中的全局变量,只不过Java没有全局变量,都是封装在类这个盒子当中。 + +### 2.1. 静态函数与实例函数 + +#### 2.1.1. C语言中如何计算字符串的长度 + +接下来看看C的函数的语义。例如C语言如果需要确定字符串的长度,这个函数的原型是这样的: + +```C int strlen(const char *str) ``` -注意,C是非面向对象的,因此它需要知道需要计算哪个字符串的长度。 +这个函数可以直接调用,有一个参数告诉这个函数是计算哪个字符串的长度。 + +#### 2.1.2. Java中如何计算字符串的长度 + +注意,C是非面向对象的,因此它需要知道需要计算哪个字符串(数据)的长度。 接下来,我们看看Java字符串长度的函数: -```java {.line-numbers} +```java String s = "Hello world!"; int size = s.length(); ``` -看到区别了吗?Java计算字符串长度的意思是计算自己本身字符串的长度;C是的含义是计算谁的字符串长度。这个就是最根本的区别。 -对象是相关数据与在相关数据上操作函数的集合。 +看到区别了吗?Java计算字符串长度的意思是计算自己本身对象字符串的长度;C是的含义是计算谁的字符串长度。这个就是最根本的区别。因为Java中的字符串是一个对象,有数据,因此在计算的时候就知道数据在哪里,因此不再需要传入一个字符串的数据。这里的s是一个字符串对象的引用变量,这样`s.length()`的语义是s这个字符串对象的长度(s中本来就就数据),计算自己字符串的长度。 + +**对象是相关数据与在相关数据上操作函数的集合。** 显然,s这个对象(String类型的实例),本身含有数据(字符串内容),以及相关的操作函数(例如 length 这个函数)。因此length()并不需要传入字符串的参数,因为length()在 s 这个对象上执行,s本身包含数据。 一般说来,需要在对象上执行的函数叫做非静态函数;而不需要对象,直接可以运行的函数叫做静态函数。其实,静态变量也是一样的,后面会做解释。 -在前面几章中我们大多使用静态函数例如对基本类型(小写开头的类型,非类类型,非包装类型)操作的函数大多是静态函数。想一想为什么?因为基本类型尽量做到与C兼容,不是对象,需要静态函数来对基本类型进行操作。后面会学到基本类型的包装类型(warp类型),本生就是对象,因此可以直接在对象上进行操作。 +String对象的length()函数是一个实例函数,如果你们看这个函数的代码,发现其并没有static进行修饰。实例函数是只能在实例(对象)上面进行调用,不能在类上面进行调用。很明显,String的类在没有实例化前是一个抽象的模板,并没有保存任何的数据,因此,使用String.length()是没有意义的,也是非法的。 + +#### 2.1.3. String中的静态函数 + +String中一个很有用的函数,就是字符串格式化,我们看看下面这个例子: + +```java +public class Test { + + public static void main(String[] args) { + String string = String.format("Hello %s !\n", "world"); + System.out.print(string); + } + +} +``` + +这个函数和`System.printf`函数一样,都是负责代码格式化,只不过,`String.format`是返回一个格式化后的字符串,`System.printf` 是打印格式化后的字符串。 + +**一个类中可以同时拥有静态、或者是实例函数;同样可以同时拥有静态变量和实例变量。** + +### 2.2. 静态变量 + +有static修饰的变量就是静态变量。Java中的静态变量和C的全局变量很相似,不过是封装在一个类中的,在本页面的1.3中的例子讲了静态变量的使用。非动态变量(依附于对象存在的变量)将在类和对象的相关知识点中讲解。 + +## 3. 函数签名与函数重载 + +在java中,一个函数肯能看起来很复杂,不仅有返回值,还有前面的public和static等修饰(这些修饰是可选的)。函数签名的概念是一个函数的函数名连同其形式参数列表叫做函数签名。 + +在一个类中,可能出现两个同名的函数,只要他们的参数列表不同就可以,这种方式叫做函数重载,对比C语言,是没有函数重载的。函数重载避免了C语言中,因为功能相似,而参数名不同需要用多个不同的函数名的问题。大家可以看到C语言的函数名大多很长,其实是因为C的所有函数都是全局存储的(Java是把函数放在盒子Class里,不同的Class中可以存在同名且同参数的函数),另外就是C没有函数重载的机制。 + +```java +public class TestMethodOverloading { + /** Main method */ + public static void main(String[] args) { + // Invoke the max method with int parameters + System.out.println("The maximum of 3 and 4 is " + max(3, 4)); + + // Invoke the max method with the double parameters + System.out.println("The maximum of 3.0 and 5.4 is " + max(3.0, 5.4)); + + // Invoke the max method with three double parameters + System.out.println("The maximum of 3.0, 5.4, and 10.14 is " + max(3.0, 5.4, 10.14)); + } + + /** Return the max of two int values */ + public static int max(int num1, int num2) { + if (num1 > num2) + return num1; + else + return num2; + } + + /** Find the max of two double values */ + public static double max(double num1, double num2) { + if (num1 > num2) + return num1; + else + return num2; + } + + /** Return the max of three double values */ + public static double max(double num1, double num2, double num3) { + return max(max(num1, num2), num3); + } +} +``` + +上面这个例子说明了函数重载的使用,大家可能跟踪调试一下。当调用一个函数的时候,是通过参数类型和参数的个数来确定最终调用哪个函数的。上个例子中的语义并不存在歧义。 + +需要注意的是,参数类型不一样的函数重载不是总是成功,例如: + +```java +public class Test { + + public static void main(String[] args) { + int rst1 = add(1, 100); + long a = 100; + long b = 200; + int rst2 = add(a, b); + } + + public static int add(int a, int b) { + System.out.println("add version 1 is invoked!"); + return a + b; + } -想一想,前面学过那些静态函数? + public static long add(long a, long b) { + System.out.println("add version 2 is invoked!"); + return a + b; + } -前面章节也有例外,这些大写的类型都是类,一般都会实例化一个对象进行使用,例如: +} +``` + +上面这个例子中我们希望 `int rst2 = add(a,b)` 调用第二个版本的add函数,但是编译会出错,这是Java的限制,大家需要根据实际情况来判断。 + +## 4. 局部变量的访问范围 -1. String 类型 -2. Scanner 类型 +> A local variable: a variable defined inside a method. +> Scope: the part of the program where the variable can be referenced. +> The scope of a local variable starts from its declaration and continues to the end of the block that contains the variable. A local variable must be declared before it can be used. -### Java与C语言像吗? +> You can declare a local variable with the same name multiple times in different non-nesting blocks in a method, but you cannot declare a local variable twice in nested blocks. + +上面说明了什么叫做局部变量(就是函数中的变量,这个和C的定义是一致的),以及局部变量的访问范围(作用域)。一句话,局部变量的作用域在其定义的时候开始,一致到该语法块结束。 + +```java +public class Test { -我们看看一个例子: + public static void main(String[] args) { + correctMethod(); + } -```java {.line-numbers} -public class Main { - static int size; + public static void correctMethod() { + int x = 1; + int y = 1; + // i is declared + for (int i = 1; i < 10; i++) { + x += i; + } + // i is declared again + for (int i = 1; i < 10; i++) { + y += i; + } - public static void main(String[] args) { - String s = "Hello world!"; - size = s.length(); + System.out.printf("X = %d ; Y = %d", x,y); + } - System.out.printf("String:'%s', size is %d.", s, size); - } } ``` -上面这个例子,如果我们把类定义去掉后变成什么样子? +这个例子中,局部变量 x,y 在for循环中是可以访问到的。 ```java - static int size; + public static void incorrectMethod() { + int x = 1; + int y = 1; + for (int i = 1; i < 10; i++) { + int x = 0; //这里的x定义非法! + x += i; + } + } +``` + +上面这个例子会显示一个编译错误,提示for循环中定义的x变量与函数中定义的x变量重名了,这时候需要考虑修改一个变量的名字。 + +## 5. 访问其他类中的静态函数 + +直接借用书上的例子: - public static void main(String[] args) { - String s = "Hello world!"; - size = s.length(); +主类 - System.out.printf("String:'%s', size is %d.", s, size); - } +```java +public class TestRandomCharacter { + /** Main method */ + public static void main(String args[]) { + final int NUMBER_OF_CHARS = 175; + final int CHARS_PER_LINE = 25; + + // Print random characters between 'a' and 'z', 25 chars per line + for (int i = 0; i < NUMBER_OF_CHARS; i++) { + char ch = RandomCharacter.getRandomLowerCaseLetter(); + if ((i + 1) % CHARS_PER_LINE == 0) + System.out.println(ch); + else + System.out.print(ch); + } + } +} ``` -看看上面的例子,结构是不是非常像C语言? +使用 `RandomCharacter.getRandomLowerCaseLetter()` 这种方式来访问静态函数,`getRandomLowerCaseLetter()` 这个函数被装在 `RandomCharacter` 这个小盒子中。 -思考一个问题:Java有全局变量吗?如果没有,如何实现全局变量? -如何调用一个静态的常量或者是变量?包是如何组织的? +被访问函数所在的类 -其实,Java没有C那样的全局变量。因为Java所有的代码必须在一个类定义中(一个类在一个包中)。这样调用一个变量(必须是静态的),可以使用 xxx.yyy.Zzz.var 的方式进行调用。其中:xxx.yyy 是包定义,Zzz是类名,var是该类中的一个静态变量。同理,静态函数的调用也可以使用这种方式。这种方式在Java中叫做命名空间(name space),有什么好处?优势是所有的函数、变量、常量等都在一个空间中,不像C语言是全局的;这样即使是同名的变量、常量、函数,只要不再同一个空间中(也不可能在同一个空间中),那么就是合法的。通过空间的方式,相当于多了一个限定,不必给变量、常量、函数取很长的名字来进行区分。反观C语言,你们会看看很长的命名,因为都在全局中,需要保证命名的唯一性。 +```java +public class RandomCharacter { + /** Generate a random character between ch1 and ch2 */ + public static char getRandomCharacter(char ch1, char ch2) { + return (char) (ch1 + Math.random() * (ch2 - ch1 + 1)); + } + + /** Generate a random lowercase letter */ + public static char getRandomLowerCaseLetter() { + return getRandomCharacter('a', 'z'); + } + + /** Generate a random uppercase letter */ + public static char getRandomUpperCaseLetter() { + return getRandomCharacter('A', 'Z'); + } + + /** Generate a random digit character */ + public static char getRandomDigitCharacter() { + return getRandomCharacter('0', '9'); + } + + /** Generate a random character */ + public static char getRandomCharacter() { + return getRandomCharacter('\u0000', '\uFFFF'); + } +} +``` diff --git a/02:Java语言基础/06.md b/02:Java语言基础/06.md index e69de29..5294d6a 100644 --- a/02:Java语言基础/06.md +++ b/02:Java语言基础/06.md @@ -0,0 +1,351 @@ +数组的定义和使用大多和C是一样的。数组是存放一类数据的一种数据结构。Java中的数组(Array)是一种特殊的对象(通过类实例化),也有特殊的语法结构。在学习Array的时候,请注意其语法,以及与后续类和对象的区别。 + +## 1. 声明与使用 + +### 1.1. 声明与使用数组 + +下面这个例子说明了数组的基本使用。 + +```java +public class ArrayTest { + + public static void main(String[] args) { + int[] myList; + + myList = new int[2]; + myList[0] = 0; + myList[1] = 1; + + System.out.println(myList[1]); + + myList = new int[3]; + + System.out.println(myList[1]); + + } +} +``` + +Java中数组的声明与使用与C类似,`int[] myList` 定义了一个可以存放int数据类型的数组,但是此时还没有为这个数组分配内存。这一点和C的指针类是,其实myList这个变量叫做引用变量,也和指针类似。 + +要为myList这个数组分配内存,使用 `new int[2]` 这种类似的语法。该语法返回一个数组对象的引用地址,数组的容量是2,并赋值给引用变量 myList,然后就可以通过引用变量 myList对数组进行操作了。 + +可以重复对一个数组引用变量使用new语句分配内存,例如,例子中第6行和第12行。与C不同的是,Java不用对动态分配的内存进行回收。 + +  + +**特别注意这里new的语法结构,后面是数组定义时的类型,大小设置不是圆括号,而是方括号!** + +### 1.2. 声明和分配内存在一行中 + +```java +public class ArrayTest2 { + + public static void main(String[] args) { + int[] myList = new int[2]; + + myList[0] = 0; + myList[1] = 1; + + System.out.println(myList[1]); + } + +} +``` + +### 1.3. 在声明的同时初始化数组 + +这个语法结构和C类似: + +```java +public class ArrayTest3 { + + public static void main(String[] args) { + int[] myList = {1,3,5,7}; + System.out.println(myList[1]); + } + +} +``` + +**注意,大括号赋初值的语法只能在数组声明中使用,不能单独使用。** + +### 1.4. 数组元素的缺省值 + +When an array is created, its elements are assigned the default value of + +0 for the numeric primitive data types, +'\u0000' for char types, and +false for boolean types. + +## 2. 数组操作 + +### 2.1. 数组对象上的操作 + +数组本质上是一个对象,因此直接可以在数组的引用变量上对数组进行操作。例如,取得数组的长度: + +```java +public class ArrayTest2 { + + public static void main(String[] args) { + int[] myList = {1,3,5,7}; + System.out.println(myList.length); + } + +} +``` + +myList 作为一个对象,还有很多实例函数,只需要在IDE中写入这个引用变量,并在后面打一个点,就会出现这个对象的所有操作(可能包括变量、函数和常量)。 + +![](img/2023-03-04-13-50-32-image.png) + +有兴趣的可以研究一下这些函数的作用,其实从这些函数名就可以知道大约的作用。 + +### 2.2. 数组元素的初始值 + +When an array is created, its elements are assigned the default value of + +0 for the numeric primitive data types, +'\u0000' for char types, and +false for boolean types. + +### 2.3. 加强的for循环 + +和C一样,Java数组的下标也是从0开始的;结合数组的长度,可以对数组进行遍历。 + +```java +public class Test { + + public static void main(String[] args) { + int arrayInt[] = { 1, 3, 5, 7 }; + for (int i = 0; i < arrayInt.length; i++) { + System.out.println(arrayInt[i]); + } + } + +} +``` + +Java还提供了一种更方便的for循环: + +```java +public class ArrayTest2 { + + public static void main(String[] args) { + int[] myList = { 1, 3, 5, 7 }; + for (int i : myList) { + System.out.println(i); + } + } + +} +``` + +这个加强版的for循环例子如上,myList是一个数组(更准确的说是一个数据集),每次取得一个数据集的元素并保存在变量i中(注意这个临时变量i的类型必须和数组中存放数据的类型一致),直到所有的元素被遍历。 + +**注意,这个for循环在以后会大量用到。** + +### 2.4. 数组变量是引用变量(指针) + +![](img/20230304124145.png) + +对数组赋值,会改变引用变量的内存指向,这和C的指针是一个概念。 + +**注意:Java中不存在手动回收内存,不存在指针悬空的情况。** + +### 2.5. 使用函数来复制数组 + +可以使用循环来复制数组: + +```java +public class Test { + + public static void main(String[] args) { + int[] sourceArray = { 2, 3, 1, 5, 10 }; + int[] targetArray = new int[sourceArray.length]; + + for (int i = 0; i < sourceArray.length; i++) + targetArray[i] = sourceArray[i]; + } + +} +``` + +Java在System这个类中提供了一个更加通用的数组复制函数`arraycopy`,注意这个函数是静态函数。 + +```java +public class ArrayTest2 { + + public static void main(String[] args) { + int[] myList1 = { 1, 3, 5, 7 }; + int[] myList2 = new int[10]; + System.arraycopy(myList1, 0, myList2, 0, 3); + for (int i : myList2) { + System.out.println(i); + } + } +} +``` + +注意 System.arraycopy 这个函数的参数代表什么意思。 + +### 2.6. 数组作为函数参数 + +```java +public class ArrayTest3 { + + public static void main(String[] args) { + int[] iA = { 5, 6, 7, 8 }; + System.out.println("The original array:"); + printArray(iA); + + changeVal(iA); + System.out.println("After change item value in function:"); + printArray(iA); + + changeArray(iA); + System.out.println("After change array in function:"); + printArray(iA); + } + + public static void printArray(int[] a) { + for (int i : a) { + System.out.println(i); + } + } + + public static void changeVal(int[] a) { + for (int i = 0; i < a.length; i++) { + a[i] = i; + } + } + + private static void changeArray(int[] a) { + a = new int[3]; + for (int i = 0; i < a.length; i++) { + a[i] = i + 100; + } + } + +} +``` + +**注意:可以在函数中改变数组内部元素的值,但是无法改变数值本身的引用(例如changeArray函数),虽然数组变量是引用变量(类似指针)。** + +### 2.7. 对数组元素排序 + +Arrys这个类中提供对数组进行排序的函数: + +```java +public class Test { + + public static void main(String[] args) { + + double[] numbers = { 6.0, 4.4, 1.9, 2.9, 3.4, 3.5 }; + java.util.Arrays.sort(numbers); + for (double item : numbers) + System.out.print(item + " "); + } + +} +``` + +排序后,数组中的元素按照从小到达排列。 + +### 2.8. 数组元素的查找 + +我们可以使用循环来查找数组中某个元素的位置,其实Java已经给我们提供了一个数组查找的函数`binarySearch`。注意这里的binary是指二分法排序,而不是指二进制。如果采用二分法查找,那么数组必须是有序的。如果无法确保数组是有序的,使用前面的`java.util.Arrays.sort`函数先对数组进行排序。 + +```java +public class Test { + + public static void main(String[] args) { + int[] list = { 2, 4, 7, 10, 11, 45, 50, 59, 60, 66, 69, 70, 79 }; + int pos = -1; + pos = java.util.Arrays.binarySearch(list, 11); + System.out.println("Index is " + pos); + } + +} +``` + +这个函数就是binarySearch,这个函数指使用注意这个函数的使用,前面的一窜`java.util.Arrays` 不是都指类的名字;最有最后的 `Arrays` 是类名,而 `java.util` 是命名空间。通常情况下,可以简单的理解命名空间就是目录的层级关系。 + +关于`java.util.Arrays`这个类还有很多其他函数,可以在IDE中通过输入点(.)查看: + +![](img/2023-03-04-15-49-20-image.png) + +有兴趣的同学可以自己去研究一下。 + +### 2.9. 补充 + +这些内容可能考试不会直接考到,但是会方便你的编程,希望你们可以掌握。 + +关于Sring的操作,通常会用到把一个字符串转换成数组。例如一个以空格分隔的字符串,我们希望把每一段切取出来,形成一个数组,这时我们使用: + +```java + String string = "AAA BBB CCC"; + string.split(" "); +``` + +splite的参水是一个正则表达式(这个我们可以不用掌握),这里如果字符串是空格分隔,写入一个空格的字符串就可以了,标识拆分的字符串。 + +函数返回是一个 String\[\] 类型的数组,你得到的是 {"AAA", "BBB", "CCC" } ,注意,不包含分隔字符串。 + +相对的,可以使用 join 函数把一个字符串数组以一定的分隔符连接成一个新的字符串。 + +split 和 join 函数都是实例函数,必须在对象上面调用。 + +## 3. 读取命令行参数 + +细心的同学们可能注意到了,在主函数中有一个字符串的数组,这个是做什么的?其实这是传递程序运行是输入的命令行参数设计的,我们看看下面这个例子: + +```java +public class Calculator { + /** Main method */ + public static void main(String[] args) { + // Check number of strings passed + if (args.length != 3) { + System.out.println("Usage: java Calculator operand1 operator operand2"); + System.exit(0); + } + + // The result of the operation + int result = 0; + + // Determine the operator + switch (args[1].charAt(0)) { + case '+': + result = Integer.parseInt(args[0]) + Integer.parseInt(args[2]); + break; + case '-': + result = Integer.parseInt(args[0]) - Integer.parseInt(args[2]); + break; + case '.': + result = Integer.parseInt(args[0]) * Integer.parseInt(args[2]); + break; + case '/': + result = Integer.parseInt(args[0]) / Integer.parseInt(args[2]); + } + + // Display result + System.out.println(args[0] + ' ' + args[1] + ' ' + args[2] + " = " + result); + } +} +``` + +主函数`void main(String[] args)` 中有有一个args 的字符串数组。编译这个程序,然后在命令行下执行,如果不记得命令行中如何编译和运行java,请参考第一章中的内容。 + +![](img/2023-03-04-15-41-18-image.png) + +在命令行中运行编译好的代码 `java Calculator 1 + 21` 其中 java是指虚拟机,后面的 `Calculator` 是Java编译后的可执行代码;`1 + 21` 是命令行参数。这个参数会传递给Calculator的主程序,通过main函数的args这个字符串数组解析出来。如同Scanner 对象一样,解析这个数组是按照空格区分的,因此,这个例子中我们得到的args数组是: + +| 数组元素 | 值 | 说明 | +| ------- | --- | --- | +| args[0] | 1 | | +| args[1] | + | | +| args[2] | 21 | | + +后面的代码就很简单了。 + +   diff --git a/02:Java语言基础/img/2023-03-04-13-50-32-image.png b/02:Java语言基础/img/2023-03-04-13-50-32-image.png new file mode 100644 index 0000000..19c0fe2 Binary files /dev/null and b/02:Java语言基础/img/2023-03-04-13-50-32-image.png differ diff --git a/02:Java语言基础/img/2023-03-04-15-41-18-image.png b/02:Java语言基础/img/2023-03-04-15-41-18-image.png new file mode 100644 index 0000000..bb013a5 Binary files /dev/null and b/02:Java语言基础/img/2023-03-04-15-41-18-image.png differ diff --git a/02:Java语言基础/img/2023-03-04-15-49-20-image.png b/02:Java语言基础/img/2023-03-04-15-49-20-image.png new file mode 100644 index 0000000..400bd07 Binary files /dev/null and b/02:Java语言基础/img/2023-03-04-15-49-20-image.png differ diff --git a/02:Java语言基础/img/20230304124145.png b/02:Java语言基础/img/20230304124145.png new file mode 100644 index 0000000..88eb5e2 Binary files /dev/null and b/02:Java语言基础/img/20230304124145.png differ diff --git a/README.md b/README.md index 215f950..f50a54b 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,3 @@ - ## 面向对象程序设计(JAVA) 1. [第一章:概述](01%EF%BC%9A%E6%A6%82%E8%BF%B0/01.md)