Java开发入门与实战!能让程序员头痛不已

3.金额数值计算精度的坑

看下这个浮点数计算的例子吧:

public class DoubleTest {
    public static void main(String[] args) {
        System.out.println(0.1+0.2);
        System.out.println(1.0-0.8);
        System.out.println(4.015*100);
        System.out.println(123.3/100);

        double amount1 = 3.15;
        double amount2 = 2.10;
        if (amount1 - amount2 == 1.05){
            System.out.println("OK");
        }
    }
}

运行结果:

0.30000000000000004
0.19999999999999996
401.49999999999994
1.2329999999999999

可以发现,结算结果跟我们预期不一致,其实是因为计算机是以二进制存储数值的,对于浮点数也是。对于计算机而言,0.1无法精确表达,这就是为什么浮点数会导致精确度缺失的。因此,金额计算,一般都是用BigDecimal 类型

对于以上例子,我们改为BigDecimal,再看看运行效果:

System.out.println(new BigDecimal(0.1).add(new BigDecimal(0.2)));
System.out.println(new BigDecimal(1.0).subtract(new BigDecimal(0.8)));
System.out.println(new BigDecimal(4.015).multiply(new BigDecimal(100)));
System.out.println(new BigDecimal(123.3).divide(new BigDecimal(100)));

运行结果:

0.3000000000000000166533453693773481063544750213623046875
0.1999999999999999555910790149937383830547332763671875
401.49999999999996802557689079549163579940795898437500
1.232999999999999971578290569595992565155029296875

发现结果还是不对,其实,使用 BigDecimal 表示和计算浮点数,必须使用字符串的构造方法来初始化 BigDecimal,正例如下:

public class DoubleTest {
    public static void main(String[] args) {
        System.out.println(new BigDecimal("0.1").add(new BigDecimal("0.2")));
        System.out.println(new BigDecimal("1.0").subtract(new BigDecimal("0.8")));
        System.out.println(new BigDecimal("4.015").multiply(new BigDecimal("100")));
        System.out.println(new BigDecimal("123.3").divide(new BigDecimal("100")));
    }
}

在进行金额计算,使用BigDecimal的时候,我们还需要注意BigDecimal的几位小数点,还有它的八种舍入模式哈

4. static静态变量依赖spring实例化变量,可能导致初始化出错

之前看到项目有类似的代码。静态变量依赖于spring容器的bean。

 private static SmsService smsService = SpringContextUtils.getBean(SmsService.class);

这个静态的smsService有可能获取不到的,因为类加载顺序不是确定的,正确的写法可以这样,如下:

 private static SmsService  smsService =null;

 //使用到的时候采取获取
 public static SmsService getSmsService(){
   if(smsService==null){
      smsService = SpringContextUtils.getBean(SmsService.class);
   }
   return smsService;
 }

5. FileReader默认编码导致乱码问题

public class FileReaderTest {
    public static void main(String[] args) throws IOException {

        Files.deleteIfExists(Paths.get("jay.txt"));
        Files.write(Paths.get("jay.txt"), "你好,捡田螺的小男孩".getBytes(Charset.forName("GBK")));
        System.out.println("系统默认编码:"+Charset.defaultCharset());

        char[] chars = new char[10];
        String content = "";
        try (FileReader fileReader = new FileReader("jay.txt")) {
            int count;
            while ((count = fileReader.read(chars)) != -1) {
                content += new String(chars, 0, count);
            }
        }
        System.out.println(content);
    }
}

运行结果:

系统默认编码:UTF-8
���,�����ݵ�С�к�

从运行结果,可以知道,系统默认编码是utf8,demo中读取出来,出现乱码了。为什么呢?

FileReader 是以当前机器的默认字符集来读取文件的,如果希望指定字符集的话,需要直接使用 InputStreamReader 和 FileInputStream。

正例如下:

public class FileReaderTest {
    public static void main(String[] args) throws IOException {

        Files.deleteIfExists(Paths.get("jay.txt"));
        Files.write(Paths.get("jay.txt"), "你好,捡田螺的小男孩".getBytes(Charset.forName("GBK")));
        System.out.println("系统默认编码:"+Charset.defaultCharset());

        char[] chars = new char[10];
        String content = "";
        try (FileInputStream fileInputStream = new FileInputStream("jay.txt");
             InputStreamReader inputStreamReader = new InputStreamReader(fileInputStream, Charset.forName("GBK"))) {
            int count;
            while ((count = inputStreamReader.read(chars)) != -1) {
                content += new String(chars, 0, count);
            }
        }
        System.out.println(content);
    }
}

6. Integer缓存的坑

public class IntegerTest {

    public static void main(String[] args) {
        Integer a = 127;
        Integer b = 127;
        System.out.println("a==b:"+ (a == b));

        Integer c = 128;
        Integer d = 128;
        System.out.println("c==d:"+ (c == d));
    }
}

运行结果:

a==b:true
c==d:false

为什么Integer值如果是128就不相等了呢?编译器会把 Integer a = 127 转换为 Integer.valueOf(127)。 我们看下源码。

public static Integer valueOf(int i) {
      if (i >= IntegerCache.low && i <= IntegerCache.high)
          return IntegerCache.cache[i + (-IntegerCache.low)];
      return new Integer(i);
 }

可以发现,i在一定范围内,是会返回缓存的。

默认情况下呢,这个缓存区间就是[-128, 127],所以我们业务日常开发中,如果涉及Integer值的比较,需要注意这个坑哈。还有呢,设置 JVM 参数加上 -XX:AutoBoxCacheMax=1000,是可以调整这个区间参数的,大家可以自己试一下哈

7. 使用ThreadLocal,线程重用导致信息错乱的坑

使用ThreadLocal缓存信息,有可能出现信息错乱的情况。看下这个例子吧。

private static final ThreadLocal<Integer> currentUser = ThreadLocal.withInitial(() -> null);

@GetMapping("wrong")
public Map wrong(@RequestParam("userId") Integer userId) {
    //设置用户信息之前先查询一次ThreadLocal中的用户信息
    String beforeUser  = Thread.currentThread().getName() + ":" + currentUser.get();
    //设置用户信息到ThreadLocal
    currentUser.set(userId);
    //设置用户信息之后再查询一次ThreadLocal中的用户信息
    String afterUser  = Thread.currentThread().getName() + ":" + currentUser.get();
    //汇总输出两次查询结果
    Map map = new HashMap();
    map.put("before", beforeUser);
    map.put("after", afterUser);
    return map;
}

按理说,每次获取的beforeUser应该都是null,但是呢,程序运行在 Tomcat 中,执行程序的线程是 Tomcat 的工作线程,而 Tomcat 的工作线程是基于线程池的。

线程池会重用固定的几个线程,一旦线程重用,那么很可能首次从 ThreadLocal 获取的值是之前其他用户的请求遗留的值。这时,ThreadLocal 中的用户信息就是其他用户的信息。

把tomcat的工作线程设置为1

server.tomcat.max-threads=1

用户1,请求过来,会有以下结果,符合预期:

用户2请求过来,会有以下结果,不符合预期

因此,使用类似 ThreadLocal 工具来存放一些数据时,需要特别注意在代码运行完后,显式地去清空设置的数据,正例如下:

@GetMapping("right")
public Map right(@RequestParam("userId") Integer userId) {
    String beforeUser  = Thread.currentThread().getName() + ":" + currentUser.get();
    currentUser.set(userId);
    try {
        String afterUser = Thread.currentThread().getName() + ":" + currentUser.get();
        Map map = new HashMap();
        map.put("before", beforeUser);
        map.put("after", afterUser);
        return map;
    } finally {
        //在finally代码块中删除ThreadLocal中的数据,确保数据不串
        currentUser.remove();
    }
}

8. 疏忽switch的return和break

这一点严格来说,应该不算坑,但是呢,大家写代码的时候,有些朋友容易疏忽了。

/*
 * 关注公众号:
 * 捡田螺的小男孩
 */
public class SwitchTest {

    public static void main(String[] args) throws InterruptedException {
        System.out.println("testSwitch结果是:"+testSwitch("1"));
    }

    private static String testSwitch(String key) {
        switch (key) {
            case "1":
                System.out.println("1");
            case "2":
                System.out.println(2);
                return "2";
            case "3":
                System.out.println("3");
            default:
                System.out.println("返回默认值");
                return "4";
        }
    }
}

输出结果:

测试switch
1
2
testSwitch结果是:2

switch 是会沿着case一直往下匹配的,知道遇到return或者break。 所以,在写代码的时候留意一下,是不是你要的结果。

9. Arrays.asList的几个坑

9.1 基本类型不能作为 Arrays.asList方法的参数,否则会被当做一个参数。
public class ArrayAsListTest {
    public static void main(String[] args) {
        int[] array = {1, 2, 3};
        List list = Arrays.asList(array);
        System.out.println(list.size());
    }
}

运行结果:

1

Arrays.asList源码如下:

public static <T> List<T> asList(T... a) {
    return new ArrayList<>(a);
}

9.2 Arrays.asList 返回的 List 不支持增删操作。
public class ArrayAsListTest {
    public static void main(String[] args) {
        String[] array = {"1", "2", "3"};
        List list = Arrays.asList(array);
        list.add("5");
        System.out.println(list.size());
    }
}

运行结果:

Exception in thread "main" java.lang.UnsupportedOperationException
	at java.util.AbstractList.add(AbstractList.java:148)
	at java.util.AbstractList.add(AbstractList.java:108)
	at object.ArrayAsListTest.main(ArrayAsListTest.java:11)

Arrays.asList 返回的 List 并不是我们期望的 java.util.ArrayList,而是 Arrays 的内部类ArrayList。内部类的ArrayList没有实现add方法,而是父类的add方法的实现,是会抛出异常的呢。

9.3 使用Arrays.asLis的时候,对原始数组的修改会影响到我们获得的那个List
public class ArrayAsListTest {
    public static void main(String[] args) {
        String[] arr = {"1", "2", "3"};
        List list = Arrays.asList(arr);
        arr[1] = "4";
        System.out.println("原始数组"+Arrays.toString(arr));
        System.out.println("list数组" + list);
    }
}

运行结果:

原始数组[1, 4, 3]
list数组[1, 4, 3]

从运行结果可以看到,原数组改变,Arrays.asList转化来的list也跟着改变啦,大家使用的时候要注意一下哦,可以用new ArrayList(Arrays.asList(arr))包一下的。

10. ArrayList.toArray() 强转的坑
public class ArrayListTest {
    public static void main(String[] args) {
        List<String> list = new ArrayList<String>(1);
        list.add("公众号:捡田螺的小男孩");
        String[] array21 = (String[])list.toArray();//类型转换异常
    }
}

因为返回的是Object类型,Object类型数组强转String数组,会发生ClassCastException。解决方案是,使用toArray()重载方法toArray(T[] a)

String[] array1 = list.toArray(new String[0]);//可以正常运行

11. 异常使用的几个坑

11.1 不要弄丢了你的堆栈异常信息
public void wrong1(){
    try {
        readFile();
    } catch (IOException e) {
        //没有把异常e取出来,原始异常信息丢失  
    }
}

public void wrong2(){
    try {
        readFile();
    } catch (IOException e) {
        //只保留了异常消息,栈没有记录啦
        log.error("文件读取错误, {}", e.getMessage());
    }
}

正确的打印方式,应该酱紫

public void right(){
    try {
        readFile();
    } catch (IOException e) {
        //把整个IO异常都记录下来,而不是只打印消息
        log.error("文件读取错误", e);
    }
}

11.2 不要把异常定义为静态变量
public void testStaticExeceptionOne{
    try {
        exceptionOne();
    } catch (Exception ex) {
        log.error("exception one error", ex);
    }
    try {
        exceptionTwo();
    } catch (Exception ex) {
        log.error("exception two error", ex);
    }
}

private void exceptionOne() {
    //这里有问题
    throw Exceptions.ONEORTWO;
}

private void exceptionTwo() {
    //这里有问题
    throw Exceptions.ONEORTWO;
}

exceptionTwo抛出的异常,很可能是exceptionOne的异常哦。正确使用方法,不是静态变量,而是应该new 一个出来。

private void exceptionTwo() {
    throw new BusinessException("业务异常", 0001);
}

11.3 生产环境不要使用e.printStackTrace();
public void wrong(){
    try {
        readFile();
    } catch (IOException e) {
       //生产环境别用它
        e.printStackTrace();
    }
}

因为它占用太多内存,造成锁死,并且,日志交错混合,也不易读。正确使用如下:

//放弃使用e.printStackTrace();
log.error("异常日志正常打印方式",e);

11.4 线程池提交过程中,出现异常怎么办?
public class ThreadExceptionTest {

    public static void main(String[] args) {
        ExecutorService executorService = Executors.newFixedThreadPool(10);

        IntStream.rangeClosed(1, 10).forEach(i -> executorService.submit(()-> {
                    if (i == 5) {
                        System.out.println("发生异常啦");
                        throw new RuntimeException("error");
                    }
                    System.out.println("当前执行第几:" + Thread.currentThread().getName() );
                }
        ));
        executorService.shutdown();
    }
}

运行结果:

当前执行第几:pool-1-thread-1
当前执行第几:pool-1-thread-2
当前执行第几:pool-1-thread-3
当前执行第几:pool-1-thread-4
发生异常啦
当前执行第几:pool-1-thread-6
当前执行第几:pool-1-thread-7
当前执行第几:pool-1-thread-8
当前执行第几:pool-1-thread-9
当前执行第几:pool-1-thread-10

可以发现,如果是使用submit方法提交到线程池的异步任务,异常会被吞掉的,所以在日常发现中,如果会有可预见的异常,可以采取这几种方案处理:

  • 1.在任务代码try/catch捕获异常
  • 2.通过Future对象的get方法接收抛出的异常,再处理
  • 3.为工作者线程设置UncaughtExceptionHandler,在uncaughtException方法中处理异常
  • 4.重写ThreadPoolExecutor的afterExecute方法,处理传递的异常引用
11.5 finally重新抛出的异常也要注意啦
public void wrong() {
    try {
        log.info("try");
        //异常丢失
        throw new RuntimeException("try");
    } finally {
        log.info("finally");
        throw new RuntimeException("finally");
    }
}

一个方法是不会出现两个异常的呢,所以finally的异常会把try的异常覆盖。正确的使用方式应该是,finally 代码块负责自己的异常捕获和处理

public void right() {
    try {
        log.info("try");
        throw new RuntimeException("try");
    } finally {
        log.info("finally");
        try {
            throw new RuntimeException("finally");
        } catch (Exception ex) {
            log.error("finally", ex);
        }
    }
}

12.JSON序列化,Long类型被转成Integer类型!

public class JSONTest {
    public static void main(String[] args) {

        Long idValue = 3000L;
        Map<String, Object> data = new HashMap<>(2);
        data.put("id", idValue);
        data.put("name", "捡田螺的小男孩");

        Assert.assertEquals(idValue, (Long) data.get("id"));
        String jsonString = JSON.toJSONString(data);

        // 反序列化时Long被转为了Integer
        Map map = JSON.parseObject(jsonString, Map.class);
        Object idObj = map.get("id");
        System.out.println("反序列化的类型是否为Integer:"+(idObj instanceof Integer));
        Assert.assertEquals(idValue, (Long) idObj);
    }
}

运行结果:

Exception in thread "main" 反序列化的类型是否为Integer:true
java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.Long
	at object.JSONTest.main(JSONTest.java:24)

注意啦,序列化为Json串后,Josn串是没有Long类型呢。而且反序列化回来如果也是Object接收,数字小于Interger最大值的话,给转成Integer啦!

13. 使用Executors声明线程池,newFixedThreadPool的OOM问题

 ExecutorService executor = Executors.newFixedThreadPool(10);
        for (int i = 0; i < Integer.MAX_VALUE; i++) {
            executor.execute(() -> {
                try {
                    Thread.sleep(10000);
                } catch (InterruptedException e) {
                    //do nothing
                }
            });
        }

IDE指定JVM参数:-Xmx8m -Xms8m :

运行结果:

我们看下源码,其实newFixedThreadPool使用的是无界队列!

public static ExecutorService newFixedThreadPool(int nThreads) {
    return new ThreadPoolExecutor(nThreads, nThreads,
                                  0L, TimeUnit.MILLISECONDS,
                                  new LinkedBlockingQueue<Runnable>());
}

public class LinkedBlockingQueue<E> extends AbstractQueue<E>
        implements BlockingQueue<E>, java.io.Serializable {
    ...
    /**
     * Creates a {@code LinkedBlockingQueue} with a capacity of
     * {@link Integer#MAX_VALUE}.
     */
    public LinkedBlockingQueue() {
        this(Integer.MAX_VALUE);
    }
...
}

newFixedThreadPool线程池的核心线程数是固定的,它使用了近乎于无界的LinkedBlockingQueue阻塞队列。当核心线程用完后,任务会入队到阻塞队列,如果任务执行的时间比较长,没有释放,会导致越来越多的任务堆积到阻塞队列,最后导致机器的内存使用不停的飙升,造成JVM OOM。

14. 直接大文件或者一次性从数据库读取太多数据到内存,可能导致OOM问题

如果一次性把大文件或者数据库太多数据达到内存,是会导致OOM的。所以,为什么查询DB数据库,一般都建议分批。

读取文件的话,一般文件不会太大,才使用Files.readAllLines()。为什么呢?因为它是直接把文件都读到内存的,预估下不会OOM才使用这个吧,可以看下它的源码:

public static List<String> readAllLines(Path path, Charset cs) throws IOException {
    try (BufferedReader reader = newBufferedReader(path, cs)) {
        List<String> result = new ArrayList<>();
        for (;;) {
            String line = reader.readLine();
            if (line == null)
                break;
            result.add(line);
        }
        return result;
    }
}

如果是太大的文件,可以使用Files.line()按需读取,当时读取文件这些,一般是使用完需要关闭资源流的哈

15. 先查询,再更新/删除的并发一致性问题

再日常开发中,这种代码实现经常可见:先查询是否有剩余可用的票,再去更新票余量。

if(selectIsAvailable(ticketId){	
    1、deleteTicketById(ticketId)	
    2、给现金增加操作	
}else{	
    return “没有可用现金券”	
}

如果是并发执行,很可能有问题的,应该利用数据库更新/删除的原子性,正解如下:

if(deleteAvailableTicketById(ticketId) == 1){	
    1、给现金增加操作	
}else{	
    return “没有可用现金券”	
}

16. 数据库使用utf-8存储, 插入表情异常的坑

低版本的MySQL支持的utf8编码,最大字符长度为 3字节,但是呢,存储表情需要4个字节,因此如果用utf8存储表情的话,会报SQLException: Incorrect string value: '\xF0\x9F\x98\x84' for column,所以一般用utf8mb4编码去存储表情。

17. spring事务未生效的坑

日常业务开发中,我们经常跟事务打交道,事务失效主要有以下几个场景:

  • 底层数据库引擎不支持事务
  • 在非public修饰的方法使用
  • rollbackFor属性设置错误
  • 本类方法直接调用
  • 异常被try…catch吃了,导致事务失效。

其中,最容易踩的坑就是后面两个,注解的事务方法给本类方法直接调用,伪代码如下:

public class TransactionTest{
  public void A(){
    //插入一条数据
    //调用方法B (本地的类调用,事务失效了)
    B();
  }



### 最后

**经过日积月累, 以下是小编归纳整理的深入了解Java虚拟机文档,希望可以帮助大家过关斩将顺利通过面试。**
由于整个文档比较全面,内容比较多,篇幅不允许,下面以截图方式展示 。如有需要获取资料文档的朋友,[可以点击这里免费获取](https://gitee.com/vip204888/java-p7)
![](https://img-blog.csdnimg.cn/img_convert/bbe3aba6b784f08071bc751ff1018526.png)
![](https://img-blog.csdnimg.cn/img_convert/a26c3bdb78e211bad7bf29c48509abe2.png)
![](https://img-blog.csdnimg.cn/img_convert/2c6164bc188da85d6cdb0222017b9cb6.png)
![](https://img-blog.csdnimg.cn/img_convert/27f28f98347c78f085d12176698291cf.png)
![](https://img-blog.csdnimg.cn/img_convert/34e14979d162e52b189593dd01bf5db5.png)
![](https://img-blog.csdnimg.cn/img_convert/1b06cd007537f2ed7270a610f7e96365.png)
![](https://img-blog.csdnimg.cn/img_convert/dc4be0e24b9558de6cb96a9296675506.png)

**主要有以下几个场景:

*   底层数据库引擎不支持事务
*   在非public修饰的方法使用
*   rollbackFor属性设置错误
*   本类方法直接调用
*   异常被try...catch吃了,导致事务失效。

其中,最容易踩的坑就是后面两个,**注解的事务方法给本类方法直接调用**,伪代码如下:

public class TransactionTest{
public void A(){
//插入一条数据
//调用方法B (本地的类调用,事务失效了)
B();
}

最后

经过日积月累, 以下是小编归纳整理的深入了解Java虚拟机文档,希望可以帮助大家过关斩将顺利通过面试。
由于整个文档比较全面,内容比较多,篇幅不允许,下面以截图方式展示 。如有需要获取资料文档的朋友,可以点击这里免费获取
[外链图片转存中…(img-7e5v26Jn-1628133116821)]
[外链图片转存中…(img-9CODty7u-1628133116822)]
[外链图片转存中…(img-s60Sy7js-1628133116823)]
[外链图片转存中…(img-wVhDv9Ec-1628133116824)]
[外链图片转存中…(img-2AqwcakF-1628133116824)]
[外链图片转存中…(img-S1URUTgf-1628133116825)]
[外链图片转存中…(img-HTgiyBIj-1628133116826)]

由于篇幅限制,文档的详解资料太全面,细节内容太多,所以只把部分知识点截图出来粗略的介绍,每个小节点里面都有更细化的内容!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 首先,恭喜你决定学习 Java 编程!在开始学习之前,我建议你先了解一些基础知识,包括计算机基础知识、编程基础知识以及 Java 的基础概念。 在学习过程中,你可以按照以下步骤来学习 Java 编程: 1. 了解 Java 的基础概念,包括 Java 的历史、安装和运行环境、Java 的基本语法和数据类型等。 2. 学习 Java 编程的基本操作,包括如何编写、编译和运行 Java 程序、如何使用变量、常量和运算符、如何使用循环和分支结构等。 3. 学习 Java 面向对象编程的基本概念和技术,包括类、对象、封装、继承、多态等。 4. 学习 Java 的常用类库和框架,包括 Java 核心类库、Java 文件输入输出、Java 网络编程、Java 数据库编程、Java GUI 编程等。 5. 学习 Java 应用程序的开发、测试和部署流程,包括如何使用版本控制工具、如何使用调试工具、如何使用测试工具、如何打包和发布 Java 应用程序等。 学习 Java 编程时,可以通过阅读 ### 回答2: 作为入门程序员,学习Java是一个不错的选择。以下是一个可行的Java学习开发路线: 1. 学习基本编程概念:首先,你需要掌握编程的基本概念,如变量、数据类型、条件语句和循环结构等。了解这些概念可以帮助你构建基本的程序逻辑。 2. 学习Java语言基础:接下来,你需要学习Java语言的基础知识,包括语法、关键字、运算符和面向对象编程等。你可以通过参加在线课程、阅读相关书籍或参考Java官方文档来学习。 3. 学习Java核心类库:Java提供了丰富的核心类库,包括输入输出、集合框架、多线程和异常处理等。理解这些类库的使用方法对于开发Java应用程序非常重要。 4. 学习数据库开发:学习使用Java与数据库进行交互,可以使用JDBC(Java Database Connectivity)技术实现。了解如何连接数据库、执行SQL语句和处理查询结果等内容。 5. 学习Web开发:掌握JavaWeb开发是非常有用的。学习使用Servlet和JSP进行动态Web页面开发,了解MVC(模型-视图-控制器)架构和常用的Web开发框架,如Spring和Hibernate。 6. 学习开发工具:选择一个适合的集成开发环境(IDE),如Eclipse或IntelliJ IDEA,以提高开发效率。学习使用版本控制工具,如Git,可以帮助你管理代码和与团队进行协作。 7. 实战项目:尝试通过参与实际项目来应用你所学到的知识。可以参与开源项目或者自己创建一个小项目,以锻炼自己的编码能力和解决问题的能力。 总结来说,入门程序员要学习Java,你需要掌握基本的编程概念和Java语言基础,了解核心类库和数据库交互,掌握Web开发以及使用开发工具和版本控制工具。通过不断实践和参与项目,你可以提升自己的编程能力和经验。 ### 回答3: 作为一个入门程序员,建议你按照以下路线学习和开发Java技术。 1. 学习Java基础知识:首先学习Java的语法、控制结构、数据类型、面向对象编程等基础知识。可以通过阅读Java教程、参加在线课程或找一本入门级的Java教材来学习。 2. 编写简单的命令行程序:使用学到的知识,编写一些简单的命令行程序,如计算器、学生成绩管理系统等。这些练习可以帮助你巩固所学知识,熟悉Java的编程环境。 3. 学习Java核心库:学习并了解Java标准库中的常用类和方法,如字符串处理、文件操作、集合框架等。这些库将会在日后的开发中经常使用到。 4. 学习面向对象设计原则:理解面向对象编程的基本原则,如封装、继承、多态等,学习如何设计和组织Java程序的类结构。 5. 学习常用开发工具和集成开发环境(IDE):了解常用的开发工具,如Eclipse、IntelliJ IDEA等。掌握基本的调试和测试技巧,提高开发效率。 6. 学习常用的Web开发技术:学习HTML、CSS、JavaScript等前端技术,了解Java Web开发框架,如Spring、Struts等。尝试编写简单的Web应用程序。 7. 学习数据库技术:了解关系型数据库和SQL语言,学习使用Java连接和操作数据库,如MySQL、Oracle等。 8. 持续学习和项目实践:加入一个开源项目或自己动手开发一个小项目,通过实践提升编程能力,不断学习新的技术和框架。 这是一个基本的Java学习和开发路线,通过不断练习和实践,你将逐渐提升自己的编程能力,并成为一名合格的Java程序员。最重要的是保持学习的热情和持续投入。祝你学习顺利!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值