[{"data":1,"prerenderedAt":-1},["ShallowReactive",2],{"\u002Fresource\u002Fdocument\u002Flist?undefined":3,"\u002Fresource\u002Fdocument\u002Fquery\u002Fpgevws6w2krkffa4?undefined":462,"\u002Fresource\u002Fadvertise\u002Flist?type=all?undefined":465},{"data":4,"status":460,"success":461},[5,148,202,291,332,370,420],{"books":6,"desc":145,"id":8,"image":146,"title":147},[7,40,63,78,93,105,117],{"cateId":8,"chapters":9,"desc":36,"id":11,"time":37,"title":38,"video":39},1,[10,15,18,21,24,27,30,33],{"bookId":11,"id":12,"indexOrder":13,"name":14},24,"8egfulw98v3h680j",0,"JavaSE 笔记（一）走进Java语言",{"bookId":11,"id":16,"indexOrder":13,"name":17},"pew6po6wrou23pk3","JavaSE 笔记（二）面向过程编程",{"bookId":11,"id":19,"indexOrder":13,"name":20},"eldst1fgrbdkmfs7","JavaSE 笔记（三）面向对象基础",{"bookId":11,"id":22,"indexOrder":13,"name":23},"48zphgkpjto8cath","JavaSE 笔记（四）面向对象高级篇",{"bookId":11,"id":25,"indexOrder":13,"name":26},"6r4llai92yc15j98","JavaSE 笔记（五）泛型程序设计",{"bookId":11,"id":28,"indexOrder":13,"name":29},"k6fmxd6qabgkwm9i","JavaSE 笔记（六）集合类与IO",{"bookId":11,"id":31,"indexOrder":13,"name":32},"qrd0xfttsz32gpqg","JavaSE 笔记（七）多线程与反射",{"bookId":11,"id":34,"indexOrder":13,"name":35},"td5tgn04nqmkrryt","JavaSE 笔记（八）GUI程序开发","基于Java25全新录制的SE课程",2025,"JavaSE 核心内容","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV163GGz2E8c",{"cateId":8,"chapters":41,"desc":59,"id":8,"time":60,"title":61,"video":62},[42,44,46,49,51,53,55,57],{"bookId":8,"id":43,"indexOrder":13,"name":14},"ibeeuwsbbi00undq",{"bookId":8,"id":45,"indexOrder":13,"name":17},"dncxjecdv4wciqcp",{"bookId":8,"id":47,"indexOrder":13,"name":48},"jviyz2hsht9ete5k","JavaSE 笔记（三）面向对象基础篇",{"bookId":8,"id":50,"indexOrder":13,"name":23},"qb9i6q9fap7bg1cc",{"bookId":8,"id":52,"indexOrder":13,"name":26},"hnkrjrkm3hjzeq6s",{"bookId":8,"id":54,"indexOrder":13,"name":29},"erpm32wduoaaqmrx",{"bookId":8,"id":56,"indexOrder":13,"name":32},"lfqtvxr7azumcwja",{"bookId":8,"id":58,"indexOrder":13,"name":35},"qs7gqok56gzc6idr","2022年制作的JavaSE版本",2022,"JavaSE 22年旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1YP4y1o75f\u002F",{"cateId":8,"chapters":64,"desc":75,"id":66,"time":60,"title":76,"video":77},[65,69,72],{"bookId":66,"id":67,"indexOrder":13,"name":68},2,"g96k66kczovvbm1i","JVM 笔记（一）走进JVM",{"bookId":66,"id":70,"indexOrder":13,"name":71},"ydd7n3jg8unc3clg","JVM 笔记（二）内存管理",{"bookId":66,"id":73,"indexOrder":13,"name":74},"r9dq37de0kaeauoi","JVM 笔记（三）类与类加载","了解Java的底层运作机制","Java JVM 虚拟机","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Er4y1r7as\u002F",{"cateId":8,"chapters":79,"desc":90,"id":81,"time":60,"title":91,"video":92},[80,84,87],{"bookId":81,"id":82,"indexOrder":13,"name":83},3,"asncyye9ya18gfar","JUC 笔记（一）再谈多线程",{"bookId":81,"id":85,"indexOrder":13,"name":86},"5tr1sm4ho6ygpt9q","JUC 笔记（二）并发编程核心",{"bookId":81,"id":88,"indexOrder":13,"name":89},"1scf51z5300mzxkh","JUC 笔记（三）并发编程进阶","你也可以成为多线程的主宰者","Java JUC 并发编程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1JT4y1S7K8\u002F",{"cateId":8,"chapters":94,"desc":102,"id":96,"time":60,"title":103,"video":104},[95,99],{"bookId":96,"id":97,"indexOrder":13,"name":98},4,"eedesc445ygiqhil","NIO 笔记（一）基础内容",{"bookId":96,"id":100,"indexOrder":13,"name":101},"ndz9t0uunrmfmv4n","NIO 笔记（二）Netty框架专题","编写畅快的高性能网络服务器","Java NIO 网络编程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1ar4y1J7mC\u002F",{"cateId":8,"chapters":106,"desc":114,"id":108,"time":60,"title":115,"video":116},[107,111],{"bookId":108,"id":109,"indexOrder":13,"name":110},5,"9890i8ofuadpwy2b","[扩展篇] Java 9-17新特性介绍",{"bookId":108,"id":112,"indexOrder":13,"name":113},"tsrkqvb6zpmtwh0n","[扩展篇] JavaSE关键字总结 笔记","精彩仍在继续，不要停止脚步","其他内容","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1tU4y1y7Fg\u002F",{"cateId":8,"chapters":118,"desc":141,"id":120,"time":142,"title":143,"video":144},[119,123,126,129,132,135,138],{"bookId":120,"id":121,"indexOrder":13,"name":122},6,"4db9h32opv7imszh","JavaSE 笔记（一）面向过程编程",{"bookId":120,"id":124,"indexOrder":13,"name":125},"c93u3v37br7hgn1q","JavaSE 笔记（二）面向对象基础篇",{"bookId":120,"id":127,"indexOrder":13,"name":128},"yglsjde9gi1jxkcb","JavaSE 笔记（三）泛型与集合类",{"bookId":120,"id":130,"indexOrder":13,"name":131},"ilhi987n986rmvo3","JavaSE 笔记（四）异常机制",{"bookId":120,"id":133,"indexOrder":13,"name":134},"pqv38vexmenglk4k","JavaSE 笔记（五）IO",{"bookId":120,"id":136,"indexOrder":13,"name":137},"jiq41n87i9ia7ilw","JavaSE 笔记（六）多线程",{"bookId":120,"id":139,"indexOrder":13,"name":140},"wn7x2mge9ws79zps","JavaSE 笔记（七）反射","此版本为早期录制的旧版本",2021,"JavaSE 21年旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Gv411T7pi\u002F","包含JavaSE基础路线全部教程笔记，打下坚实的基础","https:\u002F\u002Fpic2.zhimg.com\u002F80\u002Fv2-bf1a927f037a79f4d57d9ae543430a0d_1440w.webp","JavaSE 系列笔记 ☕️",{"books":149,"desc":199,"id":66,"image":200,"title":201},[150,166,178],{"cateId":66,"chapters":151,"desc":162,"id":153,"time":163,"title":164,"video":165},[152,156,159],{"bookId":153,"id":154,"indexOrder":13,"name":155},21,"iqbc2haub31bwqtz","Lombok 极速上手",{"bookId":153,"id":157,"indexOrder":13,"name":158},"ijay2hay19kn1k031","Mybatis 快速上手",{"bookId":153,"id":160,"indexOrder":13,"name":161},"ru4ogh2waocpn4jo","Maven 快速上手","JavaWeb阶段必须扩展知识点",2024,"常用知识讲解","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1gb421J7ok\u002F",{"cateId":66,"chapters":167,"desc":175,"id":169,"time":163,"title":176,"video":177},[168,172],{"bookId":169,"id":170,"indexOrder":13,"name":171},22,"ek20yvb6huhxizx7","JavaWeb 笔记（一）计算机网络基础",{"bookId":169,"id":173,"indexOrder":13,"name":174},"pgevws6w2krkffa4","JavaWeb笔记（二）Java与数据库","全面升级的JavaWeb课程","JavaWeb 网站开发","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1kS421X7rq\u002F",{"cateId":66,"chapters":179,"desc":196,"id":181,"time":142,"title":197,"video":198},[180,184,187,190,193],{"bookId":181,"id":182,"indexOrder":13,"name":183},7,"ggwwj09j2vkfftvd","JavaWeb 笔记（一）Java网络编程",{"bookId":181,"id":185,"indexOrder":13,"name":186},"sauvq105istskjaz","JavaWeb 笔记（二）数据库基础",{"bookId":181,"id":188,"indexOrder":13,"name":189},"xgbeasmvrhxx9tn4","JavaWeb 笔记（三）Java与数据库",{"bookId":181,"id":191,"indexOrder":13,"name":192},"k7dfwua3bsezvw9q","JavaWeb 笔记（四）前端基础",{"bookId":181,"id":194,"indexOrder":13,"name":195},"ycpagby2v7j4p728","JavaWeb 笔记（五）后端开发","搭建属于自己的Web网站","JavaWeb 旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1CL4y1i7qR\u002F","包含JavaWeb路线全套笔记，从零开始搭建自己的网站！","https:\u002F\u002Fpic3.zhimg.com\u002F80\u002Fv2-df3b38e3012258ed70c23b586309e3f6_1440w.webp","JavaWeb 系列笔记 🚛",{"books":203,"desc":288,"id":81,"image":289,"title":290},[204,220,235,255,273],{"cateId":81,"chapters":205,"desc":216,"id":207,"time":217,"title":218,"video":219},[206,210,213],{"bookId":207,"id":208,"indexOrder":13,"name":209},8,"h7sjo5oy0l03607e","SSM笔记（一）Spring基础",{"bookId":207,"id":211,"indexOrder":13,"name":212},"eve8gq72qmdb46sg","SSM笔记（二）SpringMvc基础",{"bookId":207,"id":214,"indexOrder":13,"name":215},"63v73g0zh1qlr6fk","SSM笔记（三）SpringSecurity基础","Spring的探索之路从这里开始",2023,"JavaSSM 基础部分","[\"https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Kv4y1x7is\u002F\", \"https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Lh4y1M7kx\u002F\", \"https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1fV411M7aS\u002F\"]",{"cateId":81,"chapters":221,"desc":232,"id":223,"time":217,"title":233,"video":234},[222,226,229],{"bookId":223,"id":224,"indexOrder":13,"name":225},16,"0k66v5r6slsfuog4","SpringBoot笔记（一）核心内容",{"bookId":223,"id":227,"indexOrder":13,"name":228},"bqlrnc2yvkaxo8s1","SpringBoot笔记（二）数据交互",{"bookId":223,"id":230,"indexOrder":13,"name":231},"wci9lb9tgea866jt","SpringBoot笔记（三）前后端分离","SpringBoot全新重制版","SpringBoot 新版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1xu4y1m7UP\u002F",{"cateId":81,"chapters":236,"desc":252,"id":238,"time":60,"title":253,"video":254},[237,240,243,246,249],{"bookId":238,"id":239,"indexOrder":13,"name":225},9,"e43gl1ilygps032v",{"bookId":238,"id":241,"indexOrder":13,"name":242},"emnmd8nzfdb3hr50","SpringBoot笔记（二）Git版本控制",{"bookId":238,"id":244,"indexOrder":13,"name":245},"jjlolj5igvttvyhv","SpringBoot笔记（三）Redis数据库",{"bookId":238,"id":247,"indexOrder":13,"name":248},"skgr4ivb5curdoux","SpringBoot笔记（四）其他框架介绍",{"bookId":238,"id":250,"indexOrder":13,"name":251},"le91fqhu4dqui1k4","SpringBoot笔记（五）Linux系统","逐步走向企业级开发","SpringBoot 旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1UL411V7f3\u002F",{"cateId":81,"chapters":256,"desc":270,"id":258,"time":60,"title":271,"video":272},[257,261,264,267],{"bookId":258,"id":259,"indexOrder":13,"name":260},10,"oejzo0l77zeb6a7e","SpringCloud笔记（一）微服务基础",{"bookId":258,"id":262,"indexOrder":13,"name":263},"f6eya9taaelsl35p","SpringCloud笔记（二）微服务进阶",{"bookId":258,"id":265,"indexOrder":13,"name":266},"35v1hbsfcdgagdnw","SpringCloud笔记（三）微服务应用",{"bookId":258,"id":268,"indexOrder":13,"name":269},"a782u84512tyuo1m","SpringCloud笔记（四）消息队列","体验微服务架构带来的魅力","SpringCloud 进阶","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1AL4y1j7RY\u002F",{"cateId":81,"chapters":274,"desc":285,"id":276,"time":142,"title":286,"video":287},[275,278,280,282],{"bookId":276,"id":277,"indexOrder":13,"name":209},11,"efjw75u8a251qxk5",{"bookId":276,"id":279,"indexOrder":13,"name":212},"guc134xb7sl78vju",{"bookId":276,"id":281,"indexOrder":13,"name":215},"u8ekxxucowr2b1tm",{"bookId":276,"id":283,"indexOrder":13,"name":284},"vkpmw9wbej21nei6","SSM笔记（四）MySQL进阶","此教程为2021年旧版教程","JavaSSM 旧版","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1xL4y1H7Tq\u002F","包含Spring全套框架笔记，从开始到Spring Boot，以及众多运维小知识。","https:\u002F\u002Fpic4.zhimg.com\u002F80\u002Fv2-28c3144421220d7c048703281bc34f63_1440w.webp","Spring 系列笔记 🍏",{"books":292,"desc":329,"id":96,"image":330,"title":331},[293,308],{"cateId":96,"chapters":294,"desc":305,"id":296,"time":60,"title":306,"video":307},[295,299,302],{"bookId":296,"id":297,"indexOrder":13,"name":298},12,"jd3e8u5cmvx5gco6","C语言（一）计算机思维导论",{"bookId":296,"id":300,"indexOrder":13,"name":301},"lqv77apvx82nkkio","C语言（二）基础语法",{"bookId":296,"id":303,"indexOrder":13,"name":304},"xb0b9t37gyv96xns","C语言（三）高级特性","包含高等院校需要教授的全部内容","C语言程序设计","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Cr4y137os\u002F",{"cateId":96,"chapters":309,"desc":326,"id":311,"time":60,"title":327,"video":328},[310,314,317,320,323],{"bookId":311,"id":312,"indexOrder":13,"name":313},13,"8a046ps2e4w6k4py","数据结构与算法（一）线性结构篇",{"bookId":311,"id":315,"indexOrder":13,"name":316},"3ma8db91f9zrnkja","数据结构与算法（二）树形结构篇",{"bookId":311,"id":318,"indexOrder":13,"name":319},"0lsjm59k7cgu4tpr","数据结构与算法（三）散列表篇",{"bookId":311,"id":321,"indexOrder":13,"name":322},"0qzy7bogo0g2pusa","数据结构与算法（四）图结构篇",{"bookId":311,"id":324,"indexOrder":13,"name":325},"6gmcxcikcilyxblj","数据结构与算法（五）排序算法篇","虽然很难，但是它是考研必学科目","数据结构与算法","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV13W4y127Ey\u002F","你的内心一直有一个坚定的声音在告诉你，一定要考上一名研究生，向着未来前进吧！","https:\u002F\u002Fpic2.zhimg.com\u002F80\u002Fv2-ac128404efb29ce1c9d1ccc61024f1d1_1440w.webp","C语言 系列笔记 🥬",{"books":333,"desc":367,"id":108,"image":368,"title":369},[334,349,358],{"cateId":108,"chapters":335,"desc":346,"id":337,"time":163,"title":347,"video":348},[336,340,343],{"bookId":337,"id":338,"indexOrder":13,"name":339},17,"urw2e6gg1lprv65w","Kotlin（一）基础语法",{"bookId":337,"id":341,"indexOrder":13,"name":342},"t7lnl87f74f3v1ju","Kotlin（二）类与对象",{"bookId":337,"id":344,"indexOrder":13,"name":345},"v1zzvki0knb1xvml","Kotlin（三）高级特性","包含Kotlin语言完整基础部分","Kotlin程序设计基础","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1P94y1c7tV\u002F",{"cateId":108,"chapters":350,"desc":355,"id":352,"time":163,"title":356,"video":357},[351],{"bookId":352,"id":353,"indexOrder":13,"name":354},18,"ovbzpe7065bye1st","Kotlin扩展（一）","包含Kotlin额外扩展知识","Kotlin扩展篇","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Hg4y1m7Ca\u002F",{"cateId":108,"chapters":359,"desc":364,"id":361,"time":163,"title":365,"video":366},[360],{"bookId":361,"id":362,"indexOrder":13,"name":363},19,"3at7ybv04dmjc0wp","Gradle基础教程","Gradle配置教程（Kotlin）","Gradle教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1Fc411x7xF\u002F","Kotlin让JVM平台焕发新的生机，让语言的表达更加优美","https:\u002F\u002Fpic2.zhimg.com\u002F80\u002Fv2-be815568f7c79c64cdaa171b0409786d_1440w.webp","Kotlin 系列笔记 ☘️",{"books":371,"desc":418,"id":120,"title":419},[372,391,403],{"cateId":120,"chapters":373,"desc":387,"id":375,"time":388,"title":389,"video":390},[374,378,381,384],{"bookId":375,"id":376,"indexOrder":13,"name":377},26,"zjf5qapwqtqiohcn","JavaScript笔记（一）基础语法",{"bookId":375,"id":379,"indexOrder":13,"name":380},"95jc6sjyjwcp9pvp","JavaScript笔记（二）核心知识",{"bookId":375,"id":382,"indexOrder":13,"name":383},"j35cdc1qz8dzq7pn","JavaScript笔记（三）进阶知识",{"bookId":375,"id":385,"indexOrder":13,"name":386},"sdhodlihphnpcg37","JavaScript笔记（四）前端基础","包含JavaScript最新语法规范讲解",2026,"JavaScript教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1xq6gBgESU",{"cateId":120,"chapters":392,"desc":400,"id":394,"time":37,"title":401,"video":402},[393,397],{"bookId":394,"id":395,"indexOrder":13,"name":396},23,"bsisgazdftiz3o9c","HTML5笔记（一）基础内容",{"bookId":394,"id":398,"indexOrder":13,"name":399},"njol93fs34gfwuzf","HTML5笔记（二）高级内容","包含HTML基础内容和相关知识点","HTML5核心教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1BrBiYNEWg",{"cateId":120,"chapters":404,"desc":415,"id":406,"time":37,"title":416,"video":417},[405,409,412],{"bookId":406,"id":407,"indexOrder":13,"name":408},25,"jo74ciirtg8wh90y","CSS笔记（一）基础入门",{"bookId":406,"id":410,"indexOrder":13,"name":411},"ap5ixyomoejuw4ue","CSS笔记（二）盒模型和布局",{"bookId":406,"id":413,"indexOrder":13,"name":414},"4djgk5xy1lzpiuf2","CSS笔记（三）变换和过渡","包含CSS3基础内容和相关知识点","CSS3核心教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1sQeEzFEKi","包含Web前端学习路径全部教程笔记，打下坚实的基础","Web前端 系列笔记",{"books":421,"desc":458,"id":423,"image":368,"title":459},[422,432,450],{"cateId":423,"chapters":424,"desc":429,"id":426,"time":163,"title":430,"video":431},100,[425],{"bookId":426,"id":427,"indexOrder":13,"name":428},20,"o0ab271mkdsas87","Markdown基础语法","编写简洁而又优美的文档","Markdown教程","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1eJ4m157kC",{"cateId":423,"chapters":433,"desc":447,"id":435,"time":60,"title":448,"video":449},[434,438,441,444],{"bookId":435,"id":436,"indexOrder":13,"name":437},14,"6386mh7anqt4tzyv","设计模式（一）面向对象设计原则",{"bookId":435,"id":439,"indexOrder":13,"name":440},"8ftkb38wfn6ox0ug","设计模式（二）创建型",{"bookId":435,"id":442,"indexOrder":13,"name":443},"i1msql1k8y70etey","设计模式（三）结构型",{"bookId":435,"id":445,"indexOrder":13,"name":446},"5434a3cyyjvwhs8s","设计模式（四）行为型","使你的编码水平得到质的飞跃","设计模式系列","https:\u002F\u002Fwww.bilibili.com\u002Fvideo\u002FBV1u3411P7Na\u002F",{"cateId":423,"chapters":451,"desc":456,"id":453,"time":60,"title":457},[452],{"bookId":453,"id":454,"indexOrder":13,"name":455},15,"zj9uvg0sp3b0sok8","Docker 容器技术 笔记","这里包含其他中间件课程笔记","其他中间件笔记","我们对知识的探索从未停止，只有不断地学习，才能走向美好的未来！","其他笔记分类 🌽",200,true,{"data":463,"status":460,"success":461},{"bookId":169,"content":464,"id":173,"indexOrder":66,"name":174},"![QQ_1722353274297](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F30\u002FmM4bTFPGDdYzSv9.png)\n\n# Java与数据库\n\n在开始本章节之前，需要先完成以下前置课程学习：\n\n* 《MySQL数据库技术》了解如何使用MySQL存储数据以及对数据的增删改查操作。\n\n**注意：** 请务必完成前置课程学习，本章节默认各位已经完成前置课程学习，不会再对已讲解内容重复介绍。\n\n前面我们学习了MySQL数据库以及编写各种SQL语句对数据库进行操作，那么如何通过Java如何去使用数据库来帮助我们存储数据呢，这将是本章节讨论的重点。\n\n## JDBC操作数据库\n\nJDBC是什么？JDBC英文名为：Java Data Base Connectivity（Java数据库连接）官方解释它是Java编程语言和广泛的数据库之间独立于数据库的连接标准的Java API，根本上说JDBC是一种规范，它提供的接口，一套完整的，允许便捷式访问底层数据库。可以用JAVA来写不同类型的可执行文件：JAVA应用程序、JAVA Applets、Java Servlet、JSP等，不同的可执行文件都能通过JDBC访问数据库，又兼备存储的优势。简单说它就是Java与数据库的连接的桥梁或者插件，用Java代码就能操作数据库的增删改查、存储过程、事务等。\n\n我们可以发现，JDK自带了一个`java.sql`包，而这里面就定义了大量的接口，不同类型的数据库，都可以通过实现此接口，编写适用于自己数据库的实现类。而不同的数据库厂商实现的这套标准，我们称为`数据库驱动`。\n\n![QQ_1721898279981](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F25\u002FTdj91k3n8izFEyR.png)\n\n### 准备工作\n\n那么我们首先来进行一些准备工作，以便开始JDBC的学习：\n\n* 配置IDEA或Navicat连接到我们的数据库，以便以后调试\n* 创建一个新的用于学习的数据库\n* 将mysql驱动jar依赖导入到项目中（推荐6.0版本以上，这里用到是8.0）\n\n一个Java程序并不是一个人的战斗，我们可以在别人开发的基础上继续向上开发，其他的开发者可以将自己编写的Java代码打包为`jar`，我们只需要导入这个`jar`作为依赖，即可直接使用别人提供的代码，这样就不需要我们自己从头开始手撕了，就像我们直接去使用JDK提供的类一样。\n\n首先选择项目结构：\n\n![QQ_1721899101166](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F25\u002F4p6QUCnv3rKAyN9.png)\n\n接着添加一个新的库：\n\n![QQ_1721899193206](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F25\u002FqXFDB8xoSQsZjmK.png)\n\n这样我们就成功将库引入了，可以在外部库栏目中看到我们引入的jar包：\n\n![QQ_1721899313082](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F25\u002F58JvVen9FykLlQq.png)\n\n### 使用JDBC连接数据库\n\n现在我们已经成功引入了MySQL数据库驱动，接着就可以正式开始使用JDBC了。在开始之前可以先打印一下看看自己的依赖是否成功引入了：\n\n```java\npublic static void main(String[] args) {\n  \t\u002F\u002FDriverManager是管理数据库驱动的工具类，我们可以通过它来查看当前已经引入的驱动列表\n    DriverManager.drivers().forEach(System.out::println);\n}\n```\n\n正常情况下这里应该会打印MySQL的驱动类：\n\n![QQ_1721900134689](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F25\u002F7yATgfIEeHmY6MG.png)\n\n要访问一个数据库，第一步肯定是创建一个新的的连接，我们可以通过*DriverManager*来创建一个新的数据库连接：\n\n```java\n\u002F\u002F使用getConnection方法来创建一个新的连接\ntry (Connection connection = DriverManager.getConnection(\"连接URL\",\"用户名\",\"密码\")){\n\n} catch (SQLException e) {\n    e.printStackTrace();\n}\n```\n\n这里最主要的就是连接数据库的URL，还记得我们在上一章介绍的网站URL吗，格式为 \u003C协议>:\u002F\u002F\u003C主机>:\u003C端口>\u002F\u003C路径>，互联网上所有的资源，都有一个唯一确定的URL，而MySQL本身也是以一个服务端的形式运行的，我们要连接也需要对应的URL才可以：\n\n```\njdbc:mysql:\u002F\u002Flocalhost:3306\u002Fstudy\n```\n\n接着我们需要创建一个用于执行SQL的Statement对象，然后就可以像在命令行中那样直接使用SQL命令了：\n\n```java\ntry (Connection connection = DriverManager.getConnection(\"jdbc:mysql:\u002F\u002Flocalhost:3306\u002Fstudy\",\"root\",\"123456\");\n     Statement statement = connection.createStatement()){\n    \u002F\u002F使用executeQuery来执行一个查询SQL语句\n    ResultSet set = statement.executeQuery(\"select * from user\");  \u002F\u002F选择user表全部内容\n} catch (SQLException e) {\n    e.printStackTrace();\n}\n```\n\n这里我们会得到一个ResultSet对象，我们使用它来获取查询结果，它类似于迭代器，需要使用next来向下迭代，但是使用上会有一些小小的区别：\n\n```java\nResultSet set = statement.executeQuery(\"select * from user\");\nwhile (set.next()) {   \u002F\u002F使用next开始读取查询结果的下一行\n    System.out.print(set.getInt(\"id\") + \" \");   \u002F\u002F直接获取本行指定字段下的数据\n    System.out.print(set.getString(\"name\") + \" \");\n    System.out.println(set.getInt(\"age\"));\n}\n```\n\n是不是感觉非常简单？接下来我们会为各位小伙伴详细介绍这里用到的几个类。\n\n### 了解DriverManager\n\n我们首先来了解一下DriverManager是什么东西，它其实就是管理我们的数据库驱动的。\n\n我们知道，要操作数据库，需要通过`DriverManager.getConnection`创建连接，而开始连接之前，JDBC会自动扫描我们所有引入的数据库驱动并进行加载，此时会调用`registerDriver`方法完成对Driver接口实现类的加载：\n\n```java\npublic static void registerDriver(java.sql.Driver driver, DriverAction da)\n    throws SQLException {\n    \u002F* Register the driver if it has not already been added to our list *\u002F\n    if (driver != null) {\n        registeredDrivers.addIfAbsent(new DriverInfo(driver, da));\n    } else {\n        \u002F\u002F This is for compatibility with the original DriverManager\n        throw new NullPointerException();\n    }\n    println(\"registerDriver: \" + driver);\n}\n```\n\n这里我们提到，系统会自动发现Driver接口的实现类并加载，这实际上是一种SPI机制：\n\n> Java的SPI（Service Provider Interface）机制是一种服务发现机制，它允许通过接口来加载服务实现。SPI通常用于在框架或库中提供可插拔的功能，使得不同的实现可以被动态加载和使用。这个机制在Java的标准库中以及许多开源项目中得到了广泛应用。\n>\n\n我们之前多多少少接触到过API的概念，实际上我们使用的很多类都是一种API，比如集合类，我们使用的往往是其接口，我们不需要关心其具体实现内容，我们只需要知道接口定义某个方法的用途，按照接口的定义去使用即可。\n\n而SPI则来到了另一侧，也就是接口定义了这些操作，我们需要去实现接口定义的这些操作，这样才可以使得这个功能可以实实在在地使用。\n\n![QQ_1722268678295](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F29\u002FGMunCqD56TR1rP3.png)\n\n因此，我们这里使用的MySQL数据库驱动，其实就是对JDBC官方定义的接口的一种实现，利用SPI机制就可以完成加载（实际上在META-INF\u002Fservices目录下有记录），只不过这个加载过程比较复杂，我们就不深究了。\n\n当数据库驱动加载完成之后，才可以继续完成数据库连接的建立，因为我们上一节的实例代码中直接就使用了`getConnection`创建连接，我们可以来观察一下`getConnection`做了什么：\n\n```java\n@CallerSensitive\npublic static Connection getConnection(String url,\n    String user, String password) throws SQLException {\n    java.util.Properties info = new java.util.Properties();\n\t\t\u002F\u002F这里判断的是数据库用户名和密码是否填写\n    if (user != null) { \n        info.put(\"user\", user);\n    }\n   \t...\n\t\t\u002F\u002F接着会调用内部的私有getConnection方法\n    return (getConnection(url, info, Reflection.getCallerClass()));\n}\n```\n\n```java\nprivate static Connection getConnection(\n    String url, java.util.Properties info, Class\u003C?> caller) throws SQLException {\n    \u002F\u002F这里需要通过调用此方法的类来获取其类加载器，便于后续加载数据库驱动\n    ClassLoader callerCL = caller != null ? caller.getClassLoader() : null;\n    if (callerCL == null || callerCL == ClassLoader.getPlatformClassLoader()) {\n        callerCL = Thread.currentThread().getContextClassLoader();\n    }\n\t\t\u002F\u002F检查数据库连接的url是否为空，如果是直接报错\n    if (url == null) {\n        throw new SQLException(\"The url cannot be null\", \"08001\");\n    }\n  \t...\n\t\t\u002F\u002F这一步用于确保数据库驱动已经全部加载，包括上面在一开始说的数据库驱动加载部分\n    ensureDriversInitialized();\n    \u002F\u002F 遍历所有已经加载的驱动，然后有合适的就可以直接建立连接了\n    SQLException reason = null;\n    for (DriverInfo aDriver : registeredDrivers) {\n        \u002F\u002F 如果调用此方法的Class有权限加载并使用当前驱动，就可以开始创建连接了\n        if (isDriverAllowed(aDriver.driver, callerCL)) {\n            try {\n                ...\n                Connection con = aDriver.driver.connect(url, info);\n                if (con != null) {\n                    \u002F\u002F 如果连接不为空，则创建成功，这里直接就返回了\n                    ...\n                    return (con);\n                }\n            } catch (SQLException ex) ...\n        } else ...\n    }\n    \u002F\u002F 如果代码走到这里那肯定是出问题了，要是一个能连接的驱动都没找到，直接报错\n    if (reason != null)    {\n        println(\"getConnection failed: \" + reason);\n        throw reason;\n    }\n\t\t...\n}\n```\n\n我们看到实际上在源代码中，有很多地方都会打印日志，为什么到使用的时候就没有了呢？\n\n```java\npublic static void println(String message) {\n    synchronized (logSync) {\n        if (logWriter != null) {  \u002F\u002F注意这里需要logWriter不为null才能打印\n            logWriter.println(message);\n          \t...\n        }\n    }\n}\n```\n\n所以，如果需要打印日志的话，我们给它设置一个logWriter即可：\n\n```java\nDriverManager.setLogWriter(new PrintWriter(System.out));\n```\n\n这样我们就可以在控制台看到日志打印信息了：\n\n![QQ_1722271131107](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F30\u002FWq3PJAHTip1XCys.png)\n\n下一部分我们接着来介绍Connection和Statement。\n\n### 了解Connection和Statement\n\nConnection是数据库的连接对象，也可以称作是一次会话，它可以执行 SQL 语句并在连接上下文中返回结果，这在我们一开始的时候就已经尝试过了。\n\nConnection对象中也包含数据库相关的一些信息，当我们连接成功后，可以通过`getMetaData`来获取数据库信息对象：\n\n```java\nDatabaseMetaData meta = connection.getMetaData();\nSystem.out.println(\"数据库名称: \" + meta.getDatabaseProductName());\nSystem.out.println(\"数据库版本: \" + meta.getDatabaseMajorVersion() + \".\" + meta.getDatabaseMinorVersion());\nSystem.out.println(\"当前用户: \" + meta.getUserName());\nSystem.out.println(\"数据库驱动: \" + meta.getDriverName());\nSystem.out.println(\"数据库驱动版本: \" + meta.getDriverMajorVersion() + \".\" + meta.getDriverMinorVersion());\nSystem.out.println(\"数据库驱动: \" + meta.getCatalogTerm());\n```\n\n我们也可以使用Connection对象获取一些连接上的信息，比如：\n\n```java\nSystem.out.println(\"当前选择的数据库: \" + connection.getCatalog());\n```\n\n```java\nSystem.out.println(\"数据库超时时间: \" + connection.getNetworkTimeout() + \"ms\");\n```\n\n```java\n\u002F\u002F 不支持 = 0\n\u002F\u002F 读未提交 = 1\n\u002F\u002F 读已提交 = 2\n\u002F\u002F 可重复读 = 4\n\u002F\u002F 序列化 = 8\nSystem.out.println(\"事务隔离级别: \" + connection.getTransactionIsolation());\n```\n\n我们接着来看Statement对象，它是我们执行SQL语句的关键，它用于执行静态 SQL 语句并返回其生成结果的对象。\n\n```java\nResultSet set = statement.executeQuery(\"select * from user\");\n```\n\n可以看到这里我们使用`executeQuery`方法来查询了表中所有数据，除了这个方法之外，Statement还提供了用于各种DML和DQL以及批处理等操作的方法，我们会在下节中为大家逐步介绍。\n\n### 执行DQL和DML操作\n\n还记得我们在MySQL中学习的DQL和DML操作吗？它们是用于向数据库中查询、插入、删除和更新数据的操作，包括SELECT、UPDATE、INSERT、DELETE，我们首先从最简单的SELECT语句开始。\n\n前面我们介绍了Statement对象，它为我们提供了很多方法用于执行SQL语句，其中`executeQuery`就是用于执行SELECT查询语句的，我们来看看它的使用方式，它在接口中是这样定义的：\n\n```java\n\u002F\u002F执行给定的 SQL 语句，该语句返回单个 ResultSet 对象。\nResultSet executeQuery(String sql) throws SQLException;\n```\n\n这里在执行完SQL语句后，会返回一个ResultSet对象作为结果，它表示一个数据库结果集的数据表，通常通过执行查询数据库的语句来生成。\n\n使用起来也很简单，类似于迭代器：\n\n```java\n\u002F\u002F一开始的位置为第0行，读取第一行数据要先调用一次next\nset.next();\n```\n\n![QQ_1722335538058](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F30\u002Fn387eiPyG5w9QqF.png)\n\n从0开始，每次需要查询下一行数据前都要执行一次`next`方法，接着我们可以使用各种`get`方法来获取当前行的数据，比如我们要获取当前这一行的name字段值：\n\n```java\nset.next();\nSystem.out.println(set.getString(\"name\"));\n```\n\n查询结果为：\n\n![QQ_1722333294783](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F30\u002FvXFJ251eMn4pbNj.png)\n\n**注意：** 如果使用列序号读取数据，列的下标是从1开始的。\n\n但是，如果我们使用Statement在执行完一次executeQuery得到ResultSet后，再次执行一次查询，那么之前得到的ResultSet在读取时会出现错误：\n\n> 默认情况下，每个Statement对象只能同时打开一个ResultSet对象。因此，如果一个ResultSet对象的读取与另一个对象的读取交错，则每个对象都必须由不同的Statement对象生成。如果存在打开的语句的当前对象，则Statement接口中的所有执行方法都会隐式关闭语句的当前ResultSet对象。\n\n```java\nResultSet set = statement.executeQuery(\"select * from user order by id desc\");\nResultSet set2 = statement.executeQuery(\"select * from user order by id desc\");\nset.next();\nSystem.out.println(set.getString(\"name\"));\n```\n\n得到错误：\n\n![QQ_1722334285876](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F30\u002FH6iY7jUywJt3mWd.png)\n\n我们接着来看DML操作，Statement为我们提供`executeUpdate`方法：\n\n```java\n\u002F\u002F执行给定的 SQL 语句，该语句可以是 、 UPDATE或DELETE语句，也可以是INSERT不返回任何内容的 SQL 语句，例如 SQL DDL 语句\nint executeUpdate(String sql) throws SQLException;\n\n\u002F\u002F当返回的行计数可能超过 Integer. MAX_VALUE时，应使用此方法。\nlong executeLargeUpdate(String sql) throws SQLException\n```\n\n这里返回的结果是一个int类型值，它代表：\n\n1. SQL 数据操作语言 （DML） 语句执行生效的行计数。\n2. 不返回任何内容的 SQL 语句，并且会直接得到 0 作为结果。\n\n我们从insert语句开始，看看是如何使用的：\n\n```java\nint i = statement.executeUpdate(\"insert into user (name, age) values ('小明', 18)\");\nSystem.out.println(\"生效行数: \" + i);\n```\n\n![QQ_1722335370807](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F30\u002FjN571UbaGpmzoBR.png)\n\n实际上用起来感觉和我们在命令行直接执行SQL语句差不多。 \n\n除了我们上面提到的`executeUpdate`和`executeQuery`方法外，还有一个普通的`execute`方法：\n\n```java\n\u002F\u002F 执行给定的 SQL 语句，该语句可能会返回多个结果。\nboolean execute(String sql) throws SQLException;\n```\n\n注意这个方法返回的结果是一个布尔类型结果，这个结果：\n\n*  如果第一个结果是 ResultSet 对象，返回true\n* 如果它是更新计数或没有结果，返回false\n\n也就是说，如果我们执行完SQL语句返回的是一个ResultSet结果集对象，那么就是真，也就是说只有选择语句才可以得到结果集，其他的DML语句是不可能得到这个结果的，所以肯定是假。因此，这个方法一般用于我们不确定传入的SQL语句到底是DQL还是DML语句的时候使用。\n\n```java\nboolean result = statement.execute(\"select * from user\");\nSystem.out.println(result ? \"存在结果集\" : \"不存在结果集\");\n```\n\n如果我们执行的是一个DQL语句，那么这里会得到结果集，结果集可以通过`getResultSet`方法获取：\n\n```java\nstatement.execute(\"select * from user\");\nResultSet set = statement.getResultSet();   \u002F\u002F主动获取ResultSet\nwhile (set.next()) {\n    System.out.println(set.getString(\"name\"));\n}\n```\n\n如果我们执行的时一个DML语句，那么这里会返回false，更新生效的行数可以通过`getUpdateCount`方法获取：\n\n```java\nstatement.execute(\"update user set name = '小明' where id = 1\");\nSystem.out.println(statement.getUpdateCount());\n```\n\n这样，关于数据库的基本操作就介绍完毕了。\n\n### 批处理操作\n\n我们接着来看批处理操作，当我们要执行很多条语句时，大家可能会一个一个地提交：\n\n```java\n\u002F\u002F现在要求把下面所有用户都插入到数据库中\nList\u003CString> users = List.of(\"小刚\", \"小强\", \"小王\", \"小美\", \"小黑子\");\n\u002F\u002F使用for循环来一个一个执行insert语句\nfor (String user : users) {\n    statement.executeUpdate(\"insert into user (name, age) values ('\" + user + \"', 18)\");\n}\n```\n\n虽然这样看似非常完美，也符合逻辑，但是实际上我们每次执行SQL语句，都像是去厨房端菜到客人桌上一样，我们每次上菜的时候只从厨房端一个菜，效率非常低，但是如果我们每次上菜推一个小推车装满N个菜一起上，效率就会提升很多，而数据库也是这样，我们每一次执行SQL语句，都需要一定的时间开销，但是如果我把这些任务合在一起告诉数据库，效率会截然不同：\n\n![QQ_1722352388506](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F30\u002F7lxfue9jU8nMYLQ.png)\n\n那么如何才能一口气全部交给数据库处理呢，最简单的方式是直接对我们的SQL语句进行优化，我们可以直接拼接出这样的一个字符串出来，只需要执行一次SQL即可：\n\n```sql\nINSERT INTO user (name, age) VALUES \n('小刚', 18),\n('小强', 18),\n('小王', 18),\n('小美', 18),\n('小黑子', 18);\n```\n\n但是有些时候很难实现对SQL语句的优化，我们也可以使用批处理来完成：\n\n```java\nList\u003CString> users = List.of(\"小刚\", \"小强\", \"小王\", \"小美\", \"小黑子\");\nfor (String user : users) {\n  \t\u002F\u002F使用addBatch将一个SQL语句添加到批处理列表中\n    statement.addBatch(\"insert into user (name, age) values ('\" + user + \"', 18)\");\n}\nint[] results = statement.executeBatch();  \u002F\u002F统一执行批处理\nSystem.out.println(Arrays.toString(results));\n```\n\n我们可以使用`addBatch`方法来将任务添加到批处理队列中，所有任务添加完成后，再使用`executeBatch`一次性执行批处理操作，这样同样可以防止多次提交SQL命令。\n\n并且这里会返回一个int数组代表每一个操作受影响的行数。\n\n### 将查询结果映射为对象\n\n现在我们从数据库中查询每条记录了，但是我们现在查询得到的数据依然是零零散散的，有没有更好的办法可以集中管理数据呢？实际上各位小伙伴不难发现，数据库中某一张表的一条记录，正好就是我们Java中某一个对象实体所包含的信息，而我们之前在设计数据库表的时候，也是这样去参考的：\n\n```\n1 小明 18\n```\n\n```java\npublic class User {\n    private int id;\n    private String name;\n    private int age;\n}\n```\n\n因此，为了方便，我们在查询到数据之后，一般会将每条记录都创建为一个对应的实体类对象。\n\n我们接着来完善一下它：\n\n```java\npublic class User {\n    ...\n\n    public User(int id, String name, int age) {\n        this.id = id;\n        this.name = name;\n        this.age = age;\n    }\n\n    public void say(){\n        System.out.println(\"我叫：\" + name + \"，编号为：\" + id + \"，我的年龄是：\" + age);\n    }\n}\n```\n\n好了，我们现在就可以在查询数据的时候直接转换为我们的实体类对象了：\n\n```java\nwhile (set.next()) {\n    User user = new User(set.getInt(\"id\"),\n            set.getString(\"name\"),\n            set.getInt(\"age\"));\n    user.say();\n}\n```\n\n![QQ_1722411444101](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F31\u002FuzDIRM9CkX1bOAs.png)\n\n只不过普通的类型对于我们操作不太方便，比如我们想要获取对象的一些数据，还需要额外编写对应的getter方法，为了方便，我们可以使用Java17新增的记录类型来编写，这是专用于数据表示的特殊类型，它在编译时自带了我们实体类封装所需要的getter方法，以及对应的比较方法重写，包括toString等，这样就不需要我们自己去编写了：\n\n```java\npublic record User(int id, String name, int age) {\n    public void say(){\n        System.out.println(\"我叫：\" + name + \"，编号为：\" + id + \"，我的年龄是：\" + age);\n    }\n}\n```\n\n效果和之前完全一样，并且我们可以直接使用它自动生成的方法：\n\n```java\nUser user = new User(set.getInt(\"id\"),\n        set.getString(\"name\"),\n        set.getInt(\"age\"));\nSystem.out.println(user);\n```\n\n![QQ_1722411685367](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F31\u002FnQXGEJThAONbviK.png)\n\n是不是感觉很方便？当然，除了使用Java17提供的记录类型之外，后面我们还会学习Lombok，它相比记录类型更加灵活和强大，利用注解同样可以在编译期完成对应方法的生成。\n\n（选学）利用反射和泛型，直接得到对应的实体类，无需硬写类型：\n\n```java\nprivate static \u003CT> T convert(ResultSet set, Class\u003CT> clazz){\n    try {\n        Constructor\u003CT> constructor = clazz.getConstructor(clazz.getConstructors()[0].getParameterTypes());   \u002F\u002F默认获取第一个构造方法\n        Class\u003C?>[] param = constructor.getParameterTypes();  \u002F\u002F获取参数列表\n        Object[] object = new Object[param.length];  \u002F\u002F存放参数\n        for (int i = 0; i \u003C param.length; i++) {   \u002F\u002F是从1开始的\n            object[i] = set.getObject(i+1);\n            if(object[i].getClass() != param[i])\n                throw new SQLException(\"错误的类型转换：\"+object[i].getClass()+\" -> \"+param[i]);\n        }\n        return constructor.newInstance(object);\n    } catch (ReflectiveOperationException | SQLException e) {\n        e.printStackTrace();\n        return null;\n    }\n}\n```\n\n实际上，在后面我们会学习Mybatis框架，它对JDBC进行了深层次的封装，而它就进行类似上面反射的操作来便于我们对数据库数据与实体类的转换。\n\n### 实现登陆与SQL注入攻击\n\n在生活中，很多网站都支持用户名和密码登录，这也是一种安全机制，防止别人非法访问我们自己的账户。尤其是一些银行相关的网站，如果不设置密码就可以直接访问我们的账户，那随便来个人都可以转走我们的钱了。\n\n通过一个密码验证，就可以很好地解决这个问题，在进行各种操作之前，需要使用账号密码登录，验证是你之后，才可以开始，这也是现在最主流的方式。\n\n实际上想要设计一个这样的系统也很简单，同样是设计一张用户表，只不过这次我们需要带上用户自己设定的密码，并且用户的名字或是ID必须是唯一的，否则无法区分：\n\n![QQ_1722413094272](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F31\u002FIbmOj4HJUsytQfa.png)\n\n接着我们设计一个对应的实体类：\n\n```java\npublic record Account (int id, String name, String password){\n    public boolean verifyPassword(String password) {\n        return this.password.equals(password);\n    }\n}\n```\n\n接着我们就可以从控制台接受数据并校验了：\n\n```java\ntry (Connection connection = DriverManager.getConnection(\"jdbc:mysql:\u002F\u002Flocalhost:3306\u002Fweb_study\", \"test\", \"123456\");\n     Statement statement = connection.createStatement();\n     Scanner scanner = new Scanner(System.in)){\n    System.out.print(\"请输入用户名: \");\n    String username = scanner.nextLine();\n    System.out.print(\"请输入密码: \");\n    String password = scanner.nextLine();\n    ResultSet set = statement.executeQuery(\"select * from account where name = '\" + username + \"'and password = '\" + password + \"'\");\n    if(set.next()) {\n        Account account = new Account(set.getInt(1),\n                set.getString(2), set.getString(3));\n        System.out.println(account + \" 登录成功\");\n    } else {\n        System.out.println(\"用户名或密码错误\");\n    }\n} catch (SQLException e) {\n    e.printStackTrace();\n}\n```\n\n![QQ_1722415319102](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F31\u002F18tfCopcKGgL5In.png)\n\n这样我们就实现了一个最基本的用户名密码登录系统了，用户可以通过自己输入用户名和密码来登陆。\n\n但是，实际上这样的系统存在一些问题，如果我们输入一些不太正常的内容：\n\n```sh\n请输入用户名: test\n请输入密码: 1111' or 1=1; -- \n```\n\n此时，我们的密码输入的是一个不太常规的内容：`1111' or 1=1; -- `，居然登录成功了，各位小伙伴可以想象一下如果这样的字符串拼接到我们的SQL语句中，会变成什么样呢？\n\n```sql\nselect * from account where name = 'test' and password = '1111' or 1=1; --;'\n```\n\n由于在SQL中`--`为注释，所以最终执行的SQL就变成了这样：\n\n```sql\nselect * from account where name = 'test' and password = '1111' or 1=1;\n```\n\n我们发现此时的SQL语句已经完全脱离我们想要的意思了，最后多出来了一个`or 1=1`，由于1=1一定为真，此时or后面的条件直接被判定为真了，导致后面的密码判定条件失效。\n\n因此，如果允许这样的数据插入，那么我们原有的SQL语句结构就遭到了破坏，使得用户能够随意登陆别人的账号，所以我们可能需要限制用户的输入来防止用户输入一些SQL语句关键字，但是SQL关键字非常多，这并不是解决问题的最好办法，那该怎么办呢。\n\n### PreparedStatement\n\n我们发现，如果单纯地使用Statement来执行SQL命令，会存在严重的SQL注入攻击漏洞！而这种问题，我们可以使用PreparedStatement来解决。\n\n>  `PreparedStatement` 在执行之前会被预编译，因此如果同样的 SQL 查询多次执行，性能会更好。数据库只需编译一次，而不是每次都编译，并且`PreparedStatement` 使用参数化查询，自动对输入进行转义，从而有效防止 SQL 注入攻击，通过使用参数，SQL 语句更加清晰，将值与查询逻辑分开，使得代码更易于维护和理解。\n\n它相比我们之前使用的Statement来说，不仅性能更好，还更加安全。\n\n我们还是以之前的用户登录为例，这次我们换成更加安全的PreparedStatement来操作：\n\n```java\nPreparedStatement statement = connection.prepareStatement(\"select * from account where name = ? and password = ?\")\n```\n\n我们需要提前给到PreparedStatement一个SQL语句，并且使用`?`作为占位符，它会预编译一个SQL语句，通过直接将我们的内容进行替换的方式来填写数据。\n\n现在我们修改一下之前的写法：\n\n```java\ntry (Connection connection = DriverManager.getConnection(\"jdbc:mysql:\u002F\u002Flocalhost:3306\u002Fweb_study\", \"test\", \"123456\");\n     PreparedStatement statement = connection.prepareStatement(\"select * from account where name = ? and password = ?\");\n     Scanner scanner = new Scanner(System.in)){\n    System.out.print(\"请输入用户名: \");\n    statement.setString(1, scanner.nextLine());  \u002F\u002F我们需要手动调用set方法填入参数到对应位置上去\n    System.out.print(\"请输入密码: \");\n    statement.setString(2, scanner.nextLine());\n    ResultSet set = statement.executeQuery();\n    if(set.next()) {\n        Account account = new Account(set.getInt(1),\n                set.getString(2), set.getString(3));\n        System.out.println(account + \" 登录成功\");\n    } else {\n        System.out.println(\"用户名或密码错误\");\n    }\n} catch (SQLException e) {\n    e.printStackTrace();\n}\n```\n\n执行之前的SQL注入攻击案例：\n\n![QQ_1722416161535](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F31\u002FkCcFwoa18Si2ysD.png)\n\n我们发现，此时无法再被破解了。我们来看看实际执行的SQL语句是什么，这里直接打印Statement对象：\n\n```java\nSystem.out.println(statement);\n```\n\n实际执行的SQL语句如下：\n\n```sql\nselect * from account where name = 'test' and password = '1111\\' or 1=1; --'\n```\n\n此时我们发现，我们输入的恶意内容中单引号被转义了，也就是说它现在仅仅代表的是单引号这个字符，而真正用作囊括内容的是后面的那个单引号，也就是PreparedStatement自动将可能会产生歧义的地方进行了处理，保证了我们输入的内容是什么，那么这里填入的一定是什么。这样就可以防止语义被改变，从根源上解决SQL注入攻击。\n\n### 事务操作\n\n还记得我们在MySQL中讲解的事务概念吗？\n\n> 事务是数据库管理系统中一个重要的概念，涉及到一系列操作的执行，这些操作要么全部成功，要么全部失败。\n\nJDBC默认的事务处理行为是自动提交，所以前面我们执行一个SQL语句就会被直接提交（相当于没有启动事务）因此需要使JDBC进行事务管理时，首先要通过Connection对象调用`setAutoCommit(false)`方法, 将SQL语句的提交（commit）由驱动程序转交给应用程序负责。\n\n```java\nconnection.setAutoCommit(false);\n```\n\n一旦关闭自动提交，那么现在执行所有的操作如果在最后不进行`commit()`来提交事务的话，那么所有的操作都会丢失，只有提交之后，所有的操作才会被保存：\n\n```java\nconnection.setAutoCommit(false);\nstatement.executeUpdate(\"insert into user (name, age) values ('小强', 18)\");\n```\n\n![QQ_1722421744878](https:\u002F\u002Fs2.loli.net\u002F2024\u002F07\u002F31\u002FQOxKItqvPiM5C7L.png)\n\n我们尝试在最后进行一次`commit()`操作即可正常保存：\n\n```java\nconnection.setAutoCommit(false);\nstatement.executeUpdate(\"insert into user (name, age) values ('小强', 18)\");\nconnection.commit();\n```\n\n和命令行一样，在同一个连接下，如果我们不提交的情况下使用查询，确实可以查到我们之前插入的数据，但是实际上它并不在数据库中：\n\n```java\nconnection.setAutoCommit(false);\nstatement.executeUpdate(\"insert into user (name, age) values ('小强', 18)\");\nResultSet rs = statement.executeQuery(\"select * from user\");\nwhile (rs.next()) System.out.println(rs.getString(\"name\"));\n```\n\n我们也可以使用`rollback()`来手动回滚之前的全部操作：\n\n```java\nconnection.rollback();\n```\n\n比如：\n\n```java\nstatement.executeUpdate(\"insert into user (name, age) values ('小赵', 18)\");\nstatement.executeUpdate(\"insert into user (name, age) values ('小刘', 18)\");\nconnection.rollback();\n\u002F\u002F只有这里的插入最终生效了，前面的被回滚了\nstatement.executeUpdate(\"insert into user (name, age) values ('小强', 18)\");\nconnection.commit();\n```\n\n有时候为了操作更方便，我们还可以自由创建回滚点，快速回滚到我们想要回滚的位置上：\n\n```java\nconnection.setAutoCommit(false);\nstatement.executeUpdate(\"insert into user (name, age) values ('小赵', 18)\");\nSavepoint savepoint = connection.setSavepoint();\nstatement.executeUpdate(\"insert into user (name, age) values ('小刘', 18)\");\nconnection.rollback(savepoint);  \u002F\u002F回滚到回滚点，撤销前面到回滚点部分的全部操作\nstatement.executeUpdate(\"insert into user (name, age) values ('小强', 18)\");\nconnection.commit();\n```\n\n通过开启事务，我们就可以更加谨慎地进行一些操作了，如果我们想从事务模式切换为原有的自动提交模式，我们可以直接将其设置回去：\n\n```java\nconnection.setAutoCommit(false);\nstatement.executeUpdate(\"insert into user (name, age) values ('小赵', 18)\");\nconnection.setAutoCommit(true);  \u002F\u002F重新开启自动提交，开启时把之前的事务模式下的未提交的内容给提交了\nstatement.executeUpdate(\"insert into user (name, age) values ('小刘', 18)\"); \n\u002F\u002F没有commit也成功了\n```\n\n有关事务相关操作，我们就暂时先介绍到这里。\n\n### 结果集高级用法\n\n前面我们介绍了Statement，它用于执行SQL语句，我们这一节接着来讲解它的高级用法。\n\n我们在创建Statement时，实际上可以单独配置Statement的结果集类型和并发性质，它们会决定得到的ResultSet具备的功能。默认情况下，使用不带任何参数的`createStatement()`返回 Statement 对象创建的结果集类型为TYPE_FORWARD_ONLY ，并发级别为 CONCUR_READ_ONLY。\n\n我们先来看结果集类型：\n\n* ResultSet.TYPE_FORWARD_ONLY - 顾名思义，结果集只能不断向前检索结果，不能倒回去看之前的。\n* ResultSet.TYPE_SCROLL_INSENSITIVE - 支持任意滚动，但是对修改不敏感\n* ResultSet.TYPE_SCROLL_SENSITIVE - 支持任意滚动，对修改敏感\n\n默认情况下，我们得到的ResultSet只能向前检索：\n\n```java\nResultSet rs = statement.executeQuery(\"select * from user\");\n\nrs.next();   \u002F\u002F来到第一行\nrs.next();   \u002F\u002F来到第二行\nrs.previous();  \u002F\u002F此时会出现错误，因为默认情况下不允许向后\nSystem.out.println(rs.getString(2));\n```\n\n也就是说，如果结果集类型为TYPE_FORWARD_ONLY，那么除了next之外的所有移动行相关的操作，都是不支持的。我们可以在创建Statement的时候手动设置为滚动类型，我们使用任意一种双向滚动类型即可：\n\n```java\nStatement statement = connection.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY)\n```\n\n此时随意跳转行数均可：\n\n```java\nrs.last();   \u002F\u002F使用last直接跳转到最后一行\nSystem.out.println(rs.getString(2));\n\nrs.absolute(2);   \u002F\u002F直接移动到第二行\nSystem.out.println(rs.getString(2));\n\nrs.relative(-1);   \u002F\u002F相对于当前位置向上移动一行\nSystem.out.println(rs.getString(2));\n```\n\n**注意：** 虽然双向滚动类型的ResultSet非常方便，但是由于随时都可能读取之前的数据，因此可能会导致内存持续占用不会释放，当数据量过大时，会导致内存溢出错误。\n\n我们接着来看不同的并发级别：\n\n* CONCUR_READ_ONLY - 只读模式，只能读取数据，不能修改\n* CONCUR_UPDATABLE - 可更新模式，读取数据时也可以修改数据\n\n我们可以在遍历ResultSet的时候直接对当前行数据进行更新，而不需要手动输入SQL命令：\n\n```java\nResultSet rs = statement.executeQuery(\"select * from user\");\nwhile (rs.next()) {\n    rs.updateInt(\"age\", rs.getInt(\"age\") + 1);  \u002F\u002F直接对age字段进行修改\n    rs.updateRow();   \u002F\u002F更新完后需要统一执行一次updateRow才能生效\n}\n```\n\n运行之后，可能会出现问题：\n\n![QQ_1722530107567](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F02\u002FPyavEUFeDu5pxTY.png)\n\n这是因为默认情况下我们的ResultSet为只读模式，我们需要在创建时手动将其修改为可更改模式：\n\n```java\nStatement statement = connection.createStatement(ResultSet.TYPE_FORWARD_ONLY, ResultSet.CONCUR_UPDATABLE)\n```\n\n这样，就可以顺利执行上面的操作了。\n\n我们接着来看ResultSet的可保持性，它通常在事务模式下使用，它可以决定事务提交后是否保留ResultSet中的数据，默认情况下可保持性为HOLD_CURSORS_OVER_COMMIT，它有以下选项：\n\n* HOLD_CURSORS_OVER_COMMIT - 在事务提交或回滚后，保留ResultSet的数据。 \n* CLOSE_CURSORS_AT_COMMIT - 在事务提交或回滚后，关闭ResultSet。\n\n至此，有关ResultSet的高级用法就介绍到这里。\n\n## JUL日志系统\n\n**注意：** 开启本板块前，请先完成Mybatis视频教程和Lombok视频教程，务必穿插学习后再继续向后学习JavaWeb剩余内容，实战部分也会用到哦。\n\n**首先一问：** 为什么需要日志系统？\n\n我们之前一直都在使用`System.out.println`来打印信息，但是，如果项目中存在大量的控制台输出语句，会显得很凌乱，而且日志的粒度是不够细的，假如我们现在希望，项目只在debug的情况下打印某些日志，而在实际运行时不打印日志，采用直接输出的方式就很难实现了，因此我们需要使用日志框架来规范化日志输出。\n\n而JDK为我们提供了一个自带的日志框架，位于`java.util.logging`包下，我们可以使用此框架来实现日志的规范化打印，使用起来非常简单：\n\n```java\npublic class Main {\n    public static void main(String[] args) {\n      \t\u002F\u002F 首先获取日志打印器，名称随意\n        Logger logger = Logger.getLogger(\"test\");\n      \t\u002F\u002F 调用info来输出一个普通的信息，直接填写字符串即可\n        logger.info(\"我是普通的日志\");\n    }\n}\n```\n\n我们可以在主类中使用日志打印，得到日志的打印结果：\n\n```\n十一月 15, 2021 12:55:37 下午 com.test.Main main\n信息: 我是普通的日志\n```\n\n我们发现，通过日志输出的结果会更加规范，在后续的学习中，日志将时刻伴随我们左右。\n\n### JUL基本使用\n\n日志的打印并不是简单的输出，有些时候我们可以会打印一些比较重要的日志信息，或是一些非常紧急的日志信息，根据不同类型的信息进行划分，日志一般分为7个级别，详细信息我们可以在Level类中查看：\n\n```java\npublic class Level implements java.io.Serializable {\n\t...\n  \n    \u002F\u002F出现严重故障的消息级别，值为1000，也是可用的日志级别中最大的\n    public static final Level SEVERE = new Level(\"SEVERE\",1000, defaultBundle);\n    \u002F\u002F存在潜在问题的消息级别，比如边充电边打电话就是个危险操作，虽然手机爆炸的概率很小，但是还是会有人警告你最好别这样做，这是日志级别中倒数第二大的\n    public static final Level WARNING = new Level(\"WARNING\", 900, defaultBundle);\n    \u002F\u002F所有常规提示日志信息都以INFO级别进行打印\n    public static final Level INFO = new Level(\"INFO\", 800, defaultBundle);\n  \t\u002F\u002F以下日志级别依次降低，不太常用\n    public static final Level CONFIG = new Level(\"CONFIG\", 700, defaultBundle);\n    public static final Level FINE = new Level(\"FINE\", 500, defaultBundle);\n    public static final Level FINER = new Level(\"FINER\", 400, defaultBundle);\n    public static final Level FINEST = new Level(\"FINEST\", 300, defaultBundle);\n \n  ...\n}\n```\n\n之前通过`info`方法直接输出的结果就是使用的默认级别的日志，实际上每个级别都有一个对应的方法用于打印：\n\n```java\npublic static void main(String[] args) {\n    Logger logger = Logger.getLogger(Main.class.getName());\n    logger.severe(\"severe\");  \u002F\u002F最高日志级别\n    logger.warning(\"warning\");\n    logger.info(\"info\"); \u002F\u002F默认日志级别\n    logger.config(\"config\");\n    logger.fine(\"fine\");\n    logger.finer(\"finer\");\n    logger.finest(\"finest\");   \u002F\u002F最低日志级别\n}\n```\n\n当然，如果需要更加灵活地控制日志级别，我们也可以通过`log`方法来主动设定该条日志的输出级别：\n\n```java\nLogger logger = Logger.getLogger(Main.class.getName());\nlogger.log(Level.SEVERE, \"严重的错误\", new NullPointerException(\"祝你明天就遇到我\"));\nlogger.log(Level.WARNING, \"警告的内容\");\nlogger.log(Level.INFO, \"普通的信息\");\nlogger.log(Level.CONFIG, \"级别低于普通信息\");\n```\n\n不过，可能会有小伙伴发现，某些级别的日志，并没有在控制台打印。这其实因为Logger默认情况下只会打印INFO级别以上的日志，而以下的日志则会直接省略，我们可以通过配置来进行调整，只不过调整日志打印级别比较麻烦，需要经过后续的学习我们再来讨论这个问题。\n\n### 日志核心内容\n\n前面我们介绍了日志的基本使用，我们接着来看日志打印的核心部分：Handler，它用于处理我们的日志内容打印，JDK为我们提供了很多种类的Handler用于多种不同类型的日志打印，比较常见的就是打印到控制台，当然我们也可以打印到一个日志文件中，名字一般为`xxx.log`这种格式。常用的Handler实现有：\n\n* ConsoleHandler  -  将日志通过System. err打印到控制台，现在默认就是使用的这个。\n* FileHandler  -  将日志直接写入到指定的文件中。\n* SocketHandler   -  将日志利用Socket通过网络发送到另一个主机。\n\n当然，一个Logger中可以包含多个Handler用于同时向不同的地方打印日志，我们可以通过`getHandlers`方法来获取Logger对象中已经配置的Handler对象：\n\n```java\nLogger logger = Logger.getLogger(Main.class.getName());\nSystem.out.println(Arrays.toString(logger.getHandlers()));\n```\n\n此时打印的列表中不存在任何Handler对象，可见，我们创建的Logger默认是不带任何Handler对象的，那么我们之前的日志是怎么打印出来的呢？这实际上是Logger的父级提供的，这里我们先暂时不介绍继承关系。我们使用`setUseParentHandlers`方法来屏蔽所有父级提供的日志处理器：\n\n```java\nlogger.setUseParentHandlers(false);\n```\n\n现在由于Logger没有配置任何Handler处理器，因此我们打印日志就不会有任何效果了。\n\n我们可以来尝试自己配置一个用于控制台打印的Handler处理器，这里直接创建一个新的ConsoleHandler对象：\n\n```java\nConsoleHandler handler = new ConsoleHandler();\nlogger.addHandler(handler);\nlogger.info(\"Hello World\");\n```\n\n现在我们打印日志就可以出现想要的结果了：\n\n```\n8月 28, 2024 12:12:37 上午 com.test.Main main\n信息: Hello World\n```\n\n是不是很简单？我们接着来尝试将日志记录到我们本地的文件中，这里使用FileHandler类型：\n\n```java\nFileHandler handler = new FileHandler(\"test.log\", true);   \u002F\u002F第二个参数开启后会续写已有的日志，如果不开启会直接覆盖重写\nlogger.addHandler(handler);\n```\n\n最后我们就可以得到一个日志文件了，默认是以XML格式进行写入的：\n\n![QQ_1724775618370](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002F4mZsxMOGDA8jTBn.png)\n\n这种格式有助于程序的日志读取，但是对于我们人来说，非常难以阅读，那有没有什么办法将文件的日志打印变成控制台那种格式呢？实际上每一个Handler都有一个Formatter对象，它用于控制日志的格式，默认情况下，ConsoleHandler会配置一个SimpleFormatter对象，格式为：\n\n```\n时间 类名 方法\n日志级别: 日志内容\n```\n\n我们刚刚在FileHandler中见到的是默认生成的XMLFormatter，它会将日志以XML的形式进行打印，现在我们也可以手动修改它为SimpleFormatter类型：\n\n```java\nHandler handler = new FileHandler(\"test.log\");\nhandler.setFormatter(new SimpleFormatter());\n```\n\n此时日志文件中写入的内容就是简单的日志格式了：\n\n![QQ_1724776088333](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002FVJKTIlU51cBO6aS.png)\n\n我们最后来看看上一节中遗留的日志级别问题，我们知道日志的默认打印级别为INFO，此时低于INFO的所有日志都是被屏蔽的，而要修改日志的默认打印级别，我们需要同时调整Handler和Logger的level属性：\n\n```java\nhandler.setLevel(Level.FINEST);   \u002F\u002F注意，填写的日志打印级别是什么，高于等于此级别的所有日志都会被打印\nlogger.setLevel(Level.FINEST);\nlogger.fine(\"Hello World\");\n```\n\n现在我们再次打印低于INFO级别的日志就可以正确得到结果了。Logger类还为我们提供了两个比较特殊的日志级别，它们专门用于配置特殊情况：\n\n```java\n\u002F\u002F表示直接关闭所有日志信息，值为int的最大值\npublic static final Level OFF = new Level(\"OFF\",Integer.MAX_VALUE, defaultBundle);\n\u002F\u002F表示开启所有日志信息，无论是什么级别都进行打印\npublic static final Level ALL = new Level(\"ALL\", Integer.MIN_VALUE, defaultBundle);\n```\n\n因为这这里OFF的值为int的最大值，也就是说没有任何日志级别的值大于它，因此，如果将打印等级配置为OFF，那么所有类型的日志信息都不会被打印了，而ALL则相反。\n\n### 日志继承关系\n\nJUL中Logger之间存在父子关系，这种父子关系类似于继承，我们可以通过Logger的`getParent`方法来获取其父Logger对象：\n\n```java\nLogger logger = Logger.getLogger(Main.class.getName());\nSystem.out.println(logger.getParent());\n```\n\n这里我们会得到一个：\n\n```\njava.util.logging.LogManager$RootLogger@24d46ca6\n```\n\n这个RootLogger对象为所有日志记录器的最顶层父级对象，它包含一个默认的ConsoleHandler处理器用于进行控制台打印，而日志在打印时，子Logger会继承父Logger提供的所有Handler进行日志处理，因此我们在默认情况下才能正常使用日志打印：\n\n```java\nLogger logger = Logger.getLogger(\"test\");\nLogger parent = logger.getParent();\nSystem.out.println(Arrays.toString(parent.getHandlers()));\n```\n\n根据我们上节课学习的知识，在默认情况下如果我们需要修改日志打印等级，那么同时也需要将父级的Handler也进行日志等级配置：\n\n```java\nparent.getHandlers()[0].setLevel(Level.ALL);\nlogger.setLevel(Level.ALL);\nlogger.finest(\"别吃我家鸽鸽下的蛋\");\n```\n\n当然，如果我们在不屏蔽父级Handler的情况下为子级配置一个Handler，那么此时两个Handler都会生效：\n\n```java\nlogger.addHandler(new ConsoleHandler());\nlogger.info(\"你干嘛\");\n```\n\n日志中出现了两次：\n\n```\n8月 28, 2024 12:57:39 上午 com.test.Main main\n信息: 你干嘛\n8月 28, 2024 12:57:39 上午 com.test.Main main\n信息: 你干嘛\n```\n\n不过需要注意一下顺序，当父级和子级都配置时，那么子级的Handler优先进行处理，接着才是父级。\n\n除了默认的RootLogger作为父类，实际上Logger还会通过名称进行分级，自动构建一个继承关系，比如下面：\n\n```java\nLogger logger1 = Logger.getLogger(\"com\");\nLogger logger2 = Logger.getLogger(\"com.test\");\nLogger logger3 = Logger.getLogger(\"com.test.inner1\");\nLogger logger4 = Logger.getLogger(\"com.test.inner2\");\n\nSystem.out.println(logger4.getParent() == logger2);   \u002F\u002F全true\nSystem.out.println(logger3.getParent() == logger2);\nSystem.out.println(logger2.getParent() == logger1);\n```\n\n就像包名一样，日志的名称会按照包的分级，进行自动继承，就像下面这个图一样：\n\n![QQ_1724778477259](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002F4X8PqesSAC7ijZW.png)\n\n至于为什么要这样进行划分，学习完下一节的内容就会明白了。\n\n### 日志默认配置\n\n在学习日志配置之前，我们先来学习一下Properties格式，Properties格式的文件是Java的一种配置文件，我们之前在学习Mybatis的时候学习了XML，但是我们发现XML配置文件读取实在是太麻烦，那么能否有一种简单一点的配置文件呢？此时就可以使用Properties文件，它的格式如下：\n\n```properties\nname=Test\ndesc=Description\n```\n\n该文件配置很简单，格式类似于我们Java中的Map键值对，中间使用等号进行连接。当然，键的名称我们也可以分为多级进行配置，每一级使用`.`进行划分，比如我们现在要配置数据库的连接信息，就可以编写为这种形式：\n\n```properties\njdbc.datasource.driver=com.cj.mysql.Driver\njdbc.datasource.url=jdbc:mysql:\u002F\u002Flocalhost:3306\u002Ftest\njdbc.datasource.username=test\njdbc.datasource.password=123456\n```\n\n那么配置文件编写好了，Java该如何进行读取呢？JDK为我们提供了一个叫做`Properties`的类型，它继承自Hashtable类（是HashMap的同步加锁版）使用起来和HashMap是差不多的：\n\n```java\npublic class Properties extends Hashtable\u003CObject,Object> {\n```\n\n我们现在就来创建一个试试看吧：\n\n```java\nProperties properties = new Properties();\nproperties.load(new FileReader(\"test.properties\"));   \u002F\u002F使用load方法读取本地文件中的所有配置到Map中\nSystem.out.println(properties);\n```\n\n实际上，我们也可以通过这种方式来获取我们的一些系统属性，System类中有一个`getProperties`方法用于存储所有系统相关的属性值，这里我们打印一下系统名称和版本：\n\n```java\nProperties properties = System.getProperties();\nSystem.out.println(properties.get(\"os.name\"));\nSystem.out.println(properties.get(\"os.version\"));\n```\n\n当然，程序中的Properties对象也可以快速保存为一个对应的`.properties`文件：\n\n```java\nProperties properties = System.getProperties();\nproperties.store(new FileWriter(\"system.properties\"), \"系统属性\");\n```\n\n了解完Properties之后，我们接着回到JUL中，实际上JUL也可以通过进行配置文件来规定日志打印器的一些默认值，比如我们现在想配置默认的日志打印级别：\n\n```properties\n# RootLogger 的默认处理器为\nhandlers=java.util.logging.ConsoleHandler\n# RootLogger 的默认的日志级别\n.level=ALL\n\n# 配置ConsoleHandler的默认level\njava.util.logging.ConsoleHandler.level=ALL\n```\n\n接着我们需要在程序开始之前加载这里的配置：\n\n```java\nLogManager manager = LogManager.getLogManager();   \u002F\u002F获取LogManager读取配置文件\nmanager.readConfiguration(new FileInputStream(\"test.properties\"));\nLogger logger = Logger.getLogger(\"test\");\nlogger.config(\"Hello World\");\n```\n\n这样就可以通过配置文件的形式修改一些功能的默认属性了，而不需要我们再使用代码进行配置。实际上在JUL的这类内部也有着对应的配置处理操作，如果发现有默认配置优先使用配置里面的，比如Handler的构造方法：\n\n```java\nHandler(Level defaultLevel, Formatter defaultFormatter,\n        Formatter specifiedFormatter) {\n\n    LogManager manager = LogManager.getLogManager();\n    String cname = getClass().getName();\n\n    final Level level = manager.getLevelProperty(cname + \".level\", defaultLevel);\n    final Filter filter = manager.getFilterProperty(cname + \".filter\", null);\n    final Formatter formatter = specifiedFormatter == null\n                                ? manager.getFormatterProperty(cname + \".formatter\", defaultFormatter)\n                                : specifiedFormatter;\n    final String encoding = manager.getStringProperty(cname + \".encoding\", null);\n\t\t...\n}\n```\n\n关于使用配置文件的形式修改JUL部分内容的默认值就先讲解到这里。\n\n### 自定义日志格式\n\n前面我们提到了每一个Handler都可以配置一个对应的Formatter来决定日志打印的格式，除了官方为我们提供的两种默认格式外，我们也可以自定义我们想要的日志打印格式。\n\n我们只需要继承Formatter类，就可以创建一个自定义的日志格式处理逻辑了：\n\n```java\npublic class MyFormatter extends Formatter {\n    @Override\n    public String format(LogRecord record) {\n        return \"我是自定义日志格式\";\n    }\n}\n```\n\n接着我们通过上节课的方式，直接把ConsoleHandler的默认Formatter配置为我们自己的类：\n\n```properties\njava.util.logging.ConsoleHandler.formatter=com.test.MyFormatter\n```\n\n现在随便打印一个日志看看控制台会输出什么吧：\n\n![QQ_1724780782795](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002F7wld2r6WfbH9VZE.png)\n\n其中参数为LogRecord，它提供了当前日志记录的相关信息，比如：\n\n```java\n@Override\npublic String format(LogRecord record) {\n    System.out.println(\"所在类: \" + record.getSourceClassName());\n    System.out.println(\"方法名称: \" + record.getSourceMethodName());\n    System.out.println(\"日志级别: \" + record.getLevel().getLocalizedName());\n    return \"我是自定义日志格式\";\n}\n```\n\n因此，我们也可以利用这些属性来编写一个类似于的SimpleFormatter的日志格式，比如这里包含类名、时间等，类似于下面图中的日志格式：\n\n![QQ_1724781560808](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002FFWTolmRGuX2J1cU.png)\n\n我们来尝试编写一下：\n\n```java\npublic String format(LogRecord record) {\n    StringBuilder builder = new StringBuilder();\n    \u002F\u002F日期\n    Date date = new Date(record.getMillis());\n    SimpleDateFormat dateFormat = new SimpleDateFormat(\"yyyy-MM-dd HH:mm:ss.SSS\");\n    builder.append(dateFormat.format(date));\n    \u002F\u002F级别\n    builder.append(\"  \").append(record.getLevel());\n    builder.append(\" --- \");\n    \u002F\u002F线程名称\n    builder.append('[').append(Thread.currentThread().getName()).append(']');\n    \u002F\u002F类名称\n    builder.append(\" \").append(String.format(\"%-15s\", record.getSourceClassName()));\n    \u002F\u002F消息内容\n    builder.append(\" : \").append(record.getMessage());\n\n    return builder.toString();\n}\n```\n\n至此，有关JUL相关的基础内容我们就介绍到这里，下一节我们接着来介绍其他框架对于其的兼容性。\n\n### 第三方框架兼容性\n\n我们发现，如果我们现在需要全面使用日志系统，而不是传统的直接打印，那么就需要在每个类都去编写获取Logger的代码，这样显然是很冗余的，能否简化一下这个流程呢？\n\n前面我们学习了Lombok，我们也体会到Lombok给我们带来的便捷，我们可以通过一个注解快速生成构造方法、Getter和Setter，同样的，Logger也是可以使用Lombok快速生成的。\n\n```java\n@Log\npublic class Main {\n    public static void main(String[] args) {\n        System.out.println(\"自动生成的Logger名称：\"+log.getName());\n        log.info(\"我是日志信息\");\n    }\n}\n```\n\n只需要添加一个`@Log`注解即可，添加后，我们可以直接使用一个静态变量log，而它就是自动生成的Logger。我们也可以手动指定名称：\n\n```java\n@Log(topic = \"打工是不可能打工的\")\npublic class Main {\n    public static void main(String[] args) {\n        System.out.println(\"自动生成的Logger名称：\"+log.getName());\n        log.info(\"我是日志信息\");\n    }\n}\n```\n\n我们接着来看Mybatis，经过前面的学习，我们知道，Mybatis也有日志系统，它详细记录了所有的数据库操作等，要开启日志系统，我们需要进行配置：\n\n```xml\n\u003Csetting name=\"logImpl\" value=\"STDOUT_LOGGING\" \u002F>\n```\n\n`logImpl`包括很多种配置项，包括 SLF4J | LOG4J | LOG4J2 | JDK_LOGGING | COMMONS_LOGGING | STDOUT_LOGGING | NO_LOGGING，而默认情况下是未配置，也就是说不打印。将其设定为STDOUT_LOGGING表示直接使用标准输出将日志信息打印到控制台，现在我们也可以将其设置为JDK提供的日志框架：\n\n```xml\n\u003Csetting name=\"logImpl\" value=\"JDK_LOGGING\" \u002F>\n```\n\n将其配置为JDK_LOGGING表示使用JUL进行日志打印，因为Mybatis的日志级别都比较低，因此我们需要设置一下`logging.properties`默认的日志级别：\n\n```properties\n# RootLogger 的默认处理器为\nhandlers=java.util.logging.ConsoleHandler\n# RootLogger 的默认的日志级别\n.level=ALL\n\n# 配置ConsoleHandler的默认level\njava.util.logging.ConsoleHandler.level=ALL\n```\n\n这样，Mybatis就可以正确使用JDK的日志框架进行日志打印了，只不过格式稍微有点炸裂，可能还是得我们自己编写一个自定义的Formatter才行。\n\n## Junit单元测试\n\n**首先一问：** 我们为什么需要单元测试？\n\n随着我们的项目逐渐变大，比如我们之前编写的图书管理系统，我们都是边在写边在测试，而我们当时使用的测试方法，就是直接在主方法中运行测试，但是，在很多情况下，我们的项目可能会很庞大，不可能每次都去完整地启动一个项目来测试某一个功能，这样显然会降低我们的开发效率，因此，我们需要使用单元测试来帮助我们针对于某个功能或是某个模块单独运行代码进行测试，而不是启动整个项目，比如：\n\n```java\npublic class Main {\n    public static void main(String[] args) {\n        System.out.println(\"Hello World\");\n        func1();\n        func2();\n        func3();\n    }\n    \n    private static void func1() {\n        System.out.println(\"我是第一个功能\");\n    }\n\n    private static void func2() {\n        System.out.println(\"我是第二个功能\");\n    }\n\n    private static void func3() {\n        System.out.println(\"我是第三个功能\");\n    }\n}\n```\n\n如果现在我们想单独测试某一个功能的对应方法，而不是让整个项目完全跑起来，这就非常麻烦了。而单元测试则可以针对某一个方法直接进行测试执行，无需完整启动项目。\n\n![QQ_1724827763008](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002Fsk8mJcetoxlYNBI.png)\n\n同时，在我们项目的维护过程中，难免会涉及到一些原有代码的修改，很有可能出现改了代码导致之前的功能出现问题（牵一发而动全身），而我们又不一定能立即察觉到，因此，我们可以提前保存一些测试用例，每次完成代码后都可以跑一遍测试用例，来确保之前的功能没有因为后续的修改而出现问题。我们还可以利用单元测试来评估某个模块或是功能的耗时和性能，快速排查导致程序运行缓慢的问题，这些都可以通过单元测试来完成，可见单元测试对于开发的重要性。\n\n### 初次使用JUnit\n\n首先需要导入JUnit依赖，Jar包已经放在官网资源库网盘中，直接去下载即可。同时IDEA需要安装JUnit插件（终极版默认是已经捆绑安装的，因此无需多余配置）之后我们会在Maven课程中继续介绍JUnit在Maven项目中如何进行使用。\n\n安装好之后，我们就可以直接上手使用了，使用方式很简单，只需添加一个`@Test`注解即可快速创建新的测试用例，这里我们尝试新建一个类用于单元测试：\n\n```java\npublic class MainTest {\n    \n}\n```\n\n接着就可以编写我们的测试用例了，现在我们需要创建一个`public`的无参无返回值方法（不能是静态方法）并在方法内编写我们的需要进行测试的代码：\n\n```java\npublic void test1(){\n    Main.func1();\n}\n```\n\n最后在方法上添加`@Test`注解，此时IDEA会提示我们可以运行，旁边出现运行按钮：\n\n![QQ_1724828793159](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002FQ6uP3yz1NTVIqDF.png)\n\n接着点击运行，就可以直接执行我们的测试方法了，然后可以在控制台看到当前的测试用例耗时以及状态：\n\n![QQ_1724834749820](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002FWQoIXkt1iBjmO89.png)\n\n一个测试类中可以同时有多个测试案例：\n\n```java\npublic class MainTest {\n\n    @Test\n    public void test1(){\n        Main.func1();\n    }\n\n    @Test\n    public void test2(){\n        Main.func2();\n    }\n\n    @Test\n    public void test3(){\n        Main.func3();\n    }\n}\n```\n\n我们只需要点击类旁边的运行按钮，就可以直接执行当前类中所有的测试案例：\n\n![QQ_1724839789746](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002FX1Anuvgohb6OyF2.png)\n\n有些时候，可能我们并不想开启其中某个测试用例，我们也可以使用`@Disable`来关闭某一个测试用例：\n\n```java\n@Test\n@Disabled\npublic void test2(){\n\n}\n```\n\n此时再次全部运行，将忽略二号测试案例进行测试：\n\n![QQ_1725176910092](https:\u002F\u002Fs2.loli.net\u002F2024\u002F09\u002F01\u002FoYsG2yTtpEXknA1.png)\n\n我们还可以为测试案例添加一个自定义的名称，不然测试案例一多我们就分不清楚到底哪个案例是干嘛的，我们需要使用`@DisplayName`注解来为其命名：\n\n```java\n@Test\n@DisplayName(\"这个是一个快乐的测试案例\")\npublic void test1(){\n```\n\n这样我们的控制台也可以看到对应的名称：\n\n![QQ_1724839198905](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002FGKXxyd9AQSMeHOE.png)\n\n除此之外，Junit还提供了一些预设的名称生成器，按照一定规则进行名称处理，可以通过`@DisplayNameGeneration`注解来配置使用，列表如下：\n\n| 显示名称生成器        | 行为                                       |\n| :-------------------- | :----------------------------------------- |\n| `Standard`            | 方法名称作为测试名称。                     |\n| `Simple`              | 同上，但是会删除无参数方法的尾随括号。     |\n| `ReplaceUnderscores`  | 同上，但是会用空格替换方法名称中的下划线。 |\n| `IndicativeSentences` | 包含类名和方法名称连接之后的名称。         |\n\n当然，对于一个测试案例来说，我们肯定希望测试的结果是我们所期望的一个值，因此，如果测试的结果并不是我们所期望的结果，那么这个测试就应该没有成功通过，我们可以通过断言工具类`Assertions`来对结果进行判定：\n\n```java\n@Test\npublic void test1(){\n    Random random = new Random();\n    int value = random.nextInt() % 2;   \u002F\u002F生成一个随机数，进行对2取余操作\n    Assertions.assertEquals(1, value);   \u002F\u002F如果是单数则匹配成功，如果不是则匹配失败\n}\n```\n\n当测试案例失败时，控制台会出现应该AssertionFailedError错误，同时IDEA也会提示我们测试失败：\n\n![QQ_1724838573199](https:\u002F\u002Fs2.loli.net\u002F2024\u002F08\u002F28\u002F8usXgxHIePUZVAk.png)\n\n有关断言相关工具，我们会在下一节进行详细介绍。\n\n### 断言工具\n\nJUnit提供了非常多的断言操作，相比JUnit 4，它们都被封装在一个新的`Assertions`类中，这些断言操作基本上都是用于判断某个测试结果是否符合我们的预期情况，其中最简单的就是判断结果是否等于某个值，这与我们上一节演示的操作是一样的：\n\n```java\n@Test\npublic void test1(){\n    int a = 10, b = 5;\n    int c = a + b;\n  \t\u002F\u002F判断结果是否相等，前面的是预期结果，后面的就是实际结果\n    Assertions.assertEquals(15, c);\n}\n```\n\n当断言操作发现实际结果与预期不符时，会直接抛出异常告诉我们这个测试案例没有通过，并最终以失败状态结束。我们也可以为本次断言添加一个`message`来助于我们快速了解是什么类型的测试结果出现问题：\n\n```java\nAssertions.assertEquals(14, c, \"计算结果验证\");\n```\n\n此时控制台就会得到：\n\n![QQ_1725178634235](https:\u002F\u002Fs2.loli.net\u002F2024\u002F09\u002F01\u002FMlftC67rANvuGao.png)\n\n除了使用值进行比较外，我们也可以直接对某个`boolean`类型的结果快速进行判断，使用`assertTrue`方法：\n\n```java\nAssertions.assertTrue(14 == c, \"计算结果验证\");\n```\n\n与其相似的还有两个相同对象的判断：\n\n```java\nAssertions.assertSame(999, 999);  \u002F\u002F判断两个值是否为同一个对象\n```\n\n如果判断流程比较复杂，我们也可以使用Java8的Lambda来编写结果判断逻辑，提供一个BooleanSupplier对象：\n\n```java\nAssertions.assertTrue(() -> {\n    if(c \u003C 10) return true;\n    if(c > 20) return false;\n    return c == 15;\n}, \"计算结果验证\");\n```\n\n对于更加复杂的组合结果判断，我们还可以使用`assertAll`来包含多个判断操作：\n\n```java\nAssertions.assertAll(\"整体测试\",\n        () -> Assertions.assertTrue(c == 14),\n        () -> Assertions.assertTrue(c > 10),\n        () -> Assertions.assertTrue(c \u003C 20)\n);\n```\n\n进行整体测试时，所有的测试结果将合并到一起输出。\n\n除了我们上面提到的真假判断外，还有很多不同类型的结果判断，比如异常判断，我们希望这个案例抛出指定的异常：\n\n```java\nAssertions.assertThrows(IOException.class, () -> {\n    System.out.println(1\u002F0);\n}, \"此测试案例并未抛出指定异常\");\n```\n\n由于此时抛出的是一个ArithmeticException并不是我们需要的IOException或是其子类，所以说断言失败：\n\n![image-20241115174848264](https:\u002F\u002Fs2.loli.net\u002F2024\u002F11\u002F15\u002FVwkAoeGIzChBtq2.png)\n\n除了上述例子中出现的断言方法之外，JUnit还提供了上百种断言方法供大家使用，这里就不挨个介绍了。\n\n除了断言工具外，对于一些不影响结果的测试，我们可以使用“假设”工具来实现对结果的判断但不作为测试结果的判断依据，它通常在执行给定测试没有意义时使用。\n\n```java\npublic void test1(){\n    Assumptions.assumeTrue(1 == 3);\n}\n```\n\n测试结果中会将其显示为已忽略，而不是失败：\n\n![image-20241115222449539](https:\u002F\u002Fs2.loli.net\u002F2024\u002F11\u002F15\u002FurBZYCMxmNGKP8v.png)\n\n### 条件测试和执行\n\n有些时候我们可能需要配置各种条件来执行某些测试案例，比如某些测试案例必须在指定JDK版本执行，或是某些案例只需要在某个特定操作系统执行，Junit支持我们就为测试案例设置条件来实现这些功能。\n\n比如，我们要限制某个测试案例只在指定操作系统下进行，那么就可以使用`@EnabledOnOs`来指定：\n\n```java\n@Test\n@EnabledOnOs(OS.MAC)\npublic void test1(){\n    System.out.println(\"我是只在Mac下执行的测试案例\");\n}\n\n@Test\n@EnabledOnOs(OS.WINDOWS)\n\u002F\u002F@DisabledOnOs(OS.MAC)  或是使用相反注解来为指定操作系统关闭此用例\npublic void test2(){\n    System.out.println(\"我是只在Windows下执行的测试案例\");\n}\n```\n\n这样，当我们在指定操作系统下执行时，此测试案例才会启动，否则会直接忽略：\n\n![image-20241115224429055](https:\u002F\u002Fs2.loli.net\u002F2024\u002F11\u002F15\u002FMh6eIo28tYkCc5A.png)\n\n同样的，如果我们要指定在某个JDK版本执行测试案例，也可以使用`@EnabledOnJre`来进行指定：\n\n```java\n@Test\n@EnabledOnJre(JRE.JAVA_8)\n\u002F\u002F@DisabledOnJre(JRE.JAVA_8) 或是使用相反的注解来为指定JDK关闭\npublic void test1(){\n    System.out.println(\"我是只在Java8下执行的测试案例\");\n}\n\n@Test\n@EnabledOnJre(JRE.JAVA_17)\npublic void test2(){\n    System.out.println(\"我是只在Java17下执行的测试案例\");\n}\n```\n\n或是一个指定的JDK版本范围：\n\n```java\n@Test\n@EnabledForJreRange(min = JRE.JAVA_8, max = JRE.JAVA_17)\npublic void test1(){\n    System.out.println(\"我是只在Java8-17下执行的测试案例\");\n}\n```\n\n除了这种简单判断外，我们还可以直接从系统属性中获取我们需要的参数来进行判断。\n\n> 使用`System.getProperties()`来获取所有的系统属性，包括系统的架构、版本、名称等信息。\n\n使用`@EnabledIfSystemProperty`来对系统属性进行判断：\n\n```java\n@Test\n@EnabledIfSystemProperty(named = \"os.arch\", matches = \"aarch64\")\n\u002F\u002F其中matches参数支持正则表达式\npublic void test1(){\n    System.out.println(\"我是只在arm64架构下做的测试\");\n}\n```\n\n当然，有时候为了方便，我们也可以直接读取环境变量：\n\n```java\n@Test\n@EnabledIfEnvironmentVariable(named = \"TEST_STR\", matches = \"666\")\npublic void test1(){\n    System.out.println(\"我是只在环境变量: TEST_STR = 666\");\n}\n```\n\n如果你认为这还不够灵活，你还可以直接声明一个自定义方法来进行判断：\n\n```java\n@Test\n@EnabledIf(\"testCondition\")\npublic void test1(){\n    System.out.println(\"我是自定义的测试条件\");\n}\n\npublic boolean testCondition() {\n    return 1 > 0;\n}\n```\n\n> 条件方法可以位于测试类之外。在这种情况下，它必须用其*完全限定的名称*来引用\n>\n> ```java\n> @EnabledIf(\"example.ExternalCondition#customCondition\")\n> ```\n>\n> ```java\n> class ExternalCondition {\n>   \t\u002F**\n>      * 在几种情况下，条件方法需要static：\n>      * 当@EnabledIf或@DisabledIf在类上使用时\n>      * 当@EnabledIf或@DisabledIf用于@ParameterizedTest或@TestTemplate方法时\n>      * 当条件方法位于外部类中时\n>      *\u002F\n>     static boolean customCondition() {\n>         return true;\n>     }\n> }\n> ```\n\n### 生命周期、顺序控制和嵌套测试\n\n我们可以自由设定某些操作在测试开始之前或之后执行，比如测试前的准备工作或是测试后的收尾工作：\n\n```java\n@Test\npublic void test1() {\n    System.out.println(\"我是测试方法1\");\n}\n\n@BeforeAll  \u002F\u002F使用BeforeAll必须为static方法\npublic static void start() {\n    System.out.println(\"我是测试前必须要执行的准备工作\");\n}\n```\n\n其中，`@BeforeAll`表示此准备工作在所有测试用例执行之前执行，这样，当测试开始前，会优先进行指定的准备工作，防止准备不足导致的测试失败。相反的，`@AfterAll`则会在所有测试用例完成之后执行。\n\n除了在所有方法执行前后插入准备工作，我们也可以为所有的方法单个插入准备工作：\n\n```java\n@BeforeEach  \u002F\u002F使用BeforeEach不能为static方法\npublic void start() {\n    System.out.println(\"我是测试前必须要执行的准备工作\");\n}\n```\n\n这样，在每个测试用例执行之前，都会执行一次这里的准备工作：\n\n![image-20241116004216666](https:\u002F\u002Fs2.loli.net\u002F2024\u002F11\u002F16\u002FKw2S1lxy3AbOI5n.png)\n\n我们接着来了解一下测试类的生命周期。默认情况下，执行测试实际上也会对类进行实例化，并通过实例化对象来调用其中的测试方法，并且，每一个测试用例执行之前，都会创建一个新的对象，而不是直接执行：\n\n```java\npublic class MainTest {\n\n    public MainTest() {\n        System.out.println(\"AAA\");\n    }\n  \n  ...\n```\n\n像这样，我们可以得到这样的输出结果：\n\n![image-20241116004839643](https:\u002F\u002Fs2.loli.net\u002F2024\u002F11\u002F16\u002FyGfK7EvemLFwpCt.png)\n\n每次执行测试用例都会创建一个新的对象来执行，这在某些场景下可能会显得不太方便，比如初始化类需要花费大量时间或是执行非常费时的IO操作时，这会导致我们要花费大量时间来等待每次测试用例的初始化操作。我们也可以手动修改测试类的初始化行为，默认情况下为`PER_METHOD`模式：\n\n```java\n@TestInstance(TestInstance.Lifecycle.PER_CLASS)\npublic class MainTest {\n```\n\n将其修改为`PER_CLASS`模式后，初始化操作只会执行一次，因为现在是以类为单位：\n\n![image-20241116005150384](https:\u002F\u002Fs2.loli.net\u002F2024\u002F11\u002F16\u002F8SyYQUWC2lLeEnB.png)\n\n当然，如果您现在依然对测试用例执行前后有其他准备工作需求，也可以使用之前的`@BeforeEach`和`@AfterEach`来实现灵活控制。\n\n有些时候我们可能需要控制某些测试案例的顺序，默认情况下，所有的测试案例都是按照方法的名称顺序来进行的，比如：\n\n```java\n@Test\npublic void test3() {  \u002F\u002F按照名称顺序，虽然这里是第一个定义的，但是它是第三个\n    System.out.println(\"我是测试用例3\");\n}\n\n@Test\npublic void test1() {\n    System.out.println(\"我是测试用例1\");\n}\n\n@Test\npublic void test2() {\n    System.out.println(\"我是测试用例2\");\n}\n```\n\n除了默认的名称顺序之外，JUnit提供了以下顺序：\n\n- `MethodOrderer.DisplayName`：根据显示名称对测试方法进行*字母数字*排序（请参阅[显示名称生成优先级规则](https:\u002F\u002Fjunit.org\u002Fjunit5\u002Fdocs\u002Fcurrent\u002Fuser-guide\u002F#writing-tests-display-name-generator-precedence-rules)）\n- `MethodOrderer.MethodName`：根据测试方法的名称和形式参数列表，*以字母数字*排序\n- `MethodOrderer.OrderAnnotation`：根据通过`@Order`注释指定的值对测试方法*进行数值*排序\n- `MethodOrderer.Random`：*伪随机*排序测试方法，并支持自定义*种子*的配置\n\n其中，注解顺序可以由我们自己通过注解来手动定义执行顺序：\n\n```java\n@Test\n@Order(1)\nvoid nullValues() {\n    \u002F\u002F perform assertions against null values\n}\n```\n\n有些时候我们可能需要对测试用例进行进一步的分层，比如用户相关的测试全部归为一个组，而管理相关的测试全部归为一个组，此时我们可以使用嵌套测试，通过在类中定义多个内部类来完成：\n\n```java\npublic class MainTest {\n\n    @Test\n    public void test() {\n        System.out.println(\"我是外部测试类型\");\n    }\n\n    @Nested\n    class Test1 {\n\n        @Test\n        public void test1_1() {\n            System.out.println(\"我是内部测试类型1-1\");\n        }\n\n        @Test\n        public void test1_2() {\n            System.out.println(\"我是内部测试类型1-2\");\n        }\n    }\n\n    @Nested\n    class Test2 {\n        @Test\n        public void test2_1() {\n            System.out.println(\"我是内部测试类型2-1\");\n        }\n\n        @Test\n        public void test2_2() {\n            System.out.println(\"我是内部测试类型2-2\");\n        }\n    }\n}\n```\n\n此时测试的结果展示也是嵌套的样式：\n\n![image-20241116001214709](https:\u002F\u002Fs2.loli.net\u002F2024\u002F11\u002F16\u002F35vkn4qQzlbVrif.png)\n\n注意，当我们在嵌套测试中使用诸如`@BeforeEach`、`@BeforeAll`这种注解时，它仅会作用于所属内部类中的所有测试用例，而不是包含外部类中和其他内部类中的全部测试用例。\n\n嵌套类的执行同样可以通过`@TestClassOrder`来控制嵌套类的执行顺序。\n\n### 重复和参数化测试\n\n对于某些存在随机性的测试案例，我们可能需要多次执行才能确定其是否存在某些问题，比如某个案例存在一个BUG，导致其10次里面会有1次出现错误，现在我们想要保证其10次都不会出现问题才算通过，此时我们就可以使用重复测试案例来使其多次执行：\n\n```java\n@RepeatedTest(10)\npublic void test1() {\n    Random random = new Random();\n    if (random.nextInt(10) == 0) {\n        throw new IllegalStateException();\n    }\n}\n```\n\n此时会重复执行10次此案例，并且当每一次执行都没有出现问题时，才会正常通过：\n\n![image-20241116010355502](https:\u002F\u002Fs2.loli.net\u002F2024\u002F11\u002F16\u002FiLSas1TzdHtOgUI.png)\n\n某些测试可能并不是固定单个输入参数，有时我们可能也需要对多个输入参数进行测试，来做到全方面的问题排查。它与重复测试比较类似，但是参数可以由我们自己决定：\n\n```java\n@ParameterizedTest  \u002F\u002F使用此注解来表示此测试是一个参数化测试\n@ValueSource(strings = { \"aa\", \"bb\", \"ccc\" })   \u002F\u002F指定参数列表\npublic void test1(String str) {  \u002F\u002F需要添加一个参数\n    if (str.length() == 3) {\n        throw new IllegalStateException();\n    }\n}\n```\n\n这里我们使用`@ValueSource`来进行参数来源设定，也就是需要进行测试的参数列表，接着下面会根据参数挨个执行此测试用例，保证每一种情况都正常执行：\n\n![image-20241116021604397](https:\u002F\u002Fs2.loli.net\u002F2024\u002F11\u002F16\u002FiT3UvfHc2Ns4koS.png)\n\n这里的`@ValueSource`是最简单的一种参数设定，我们可以直接设置一系列值，支持以下类型：`short`、`byte`、`int`、`long`、`float`、`double`、`char`、`boolean`、`java.lang.String`、`java.lang.Class`\n\n除了直接设置指定类型常量值，我们也可以传入空值或是一些为空的字符串、数组等：\n\n```java\n@ParameterizedTest\n@NullSource  \u002F\u002F将值设置为null进行测试\npublic void test1(String str) {\n```\n\n```java\n@ParameterizedTest\n@EmptySource  \u002F\u002F将值设置为空进行测试，如空字符串、空数组、空集合等\npublic void test1(int[] arr) {\n```\n\n> `@NullAndEmptySource`：结合了`@NullSource`和`@EmptySource`两个注解的功能。\n\n我们也可以使用枚举值来进行测试，比如我们希望测试某个枚举类型下所有的枚举作为参数进行测试：\n\n```java\nenum Type {\n    SMALL, MEDIUM, LARGE\n}\n\n@ParameterizedTest\n@EnumSource(Type.class)  \u002F\u002F这将依次测试枚举类中的所有枚举\npublic void test1(Type type) {\n    System.out.println(type);\n}\n```\n\n或是指定某些枚举常量：\n\n```java\n@ParameterizedTest\n\u002F\u002F模式默认为INCLUDE，即使用指定的枚举常量进行测试\n@EnumSource(mode = EnumSource.Mode.INCLUDE, names = { \"SMALL\", \"LARGE\" })\npublic void test1(Type type) {\n    System.out.println(type);\n}\n```\n\n除了以上方式获取参数，我们也可以使用特定的方法来生成我们需要的测试参数，只需要添加`@MethodSource`注解即可指定方法：\n\n```java\n@ParameterizedTest\n@MethodSource(\"stringProvider\")\npublic void test1(String str) {\n    System.out.println(str);\n}\n\nstatic List\u003CString> stringProvider() {\n    return List.of(\"apple\", \"banana\");\n}\n```\n\n方法的返回值可以是任何可迭代（Iterable）内容，如数组、集合类、Stream等。同样的，对于其他类中的方法，需要和之前一样使用*完全限定的方法名称*来引用。\n\n和方法一样，字段同样可以作为参数的来源，但它必须是静态的：\n\n```java\nstatic List\u003CString> list = List.of(\"AAA\", \"BBB\");\n\n@ParameterizedTest\n@FieldSource(\"list\")\npublic void test1(String str) {\n    System.out.println(str);\n}\n```\n\n不仅仅是一个普通的集合或是数组可以作为字段参数来源，如Supplier这种懒加载的数据，也可以作为参数来源：\n\n```java\nstatic Supplier\u003CList\u003CString>> list = () -> List.of(\"AAA\", \"BBB\");\n\n@ParameterizedTest\n@FieldSource(\"list\")\npublic void test1(String str) {\n    System.out.println(str);\n}\n```\n\n当然，JUnit还支持从CSV表格中导入或自定义参数提供器等，这里就不做详细介绍了，官方文档：https:\u002F\u002Fjunit.org\u002Fjunit5\u002Fdocs\u002Fcurrent\u002Fuser-guide\u002F#writing-tests-parameterized-tests-sources-ArgumentsSource\n\n## 实战：图书管理系统\n\n**注意：** 在开始之前，请先完成Maven视频课程，本次实战以及后续内容统一采用Maven进行依赖管理，不再使用之前的Jar包导入方式。\n\n项目需求：\n\n* 在线录入学生信息和书籍信息\n* 查询书籍信息列表\n* 查询学生信息列表\n* 查询借阅信息列表\n* 完整的日志系统\n\nMySQL数据库JDBC驱动依赖：\n\n```xml\n\u003Cdependency>\n    \u003CgroupId>mysql\u003C\u002FgroupId>\n    \u003CartifactId>mysql-connector-java\u003C\u002FartifactId>\n    \u003Cversion>8.0.33\u003C\u002Fversion>\n\u003C\u002Fdependency>\n```\n\nMybatis框架依赖：\n\n```xml\n\u003Cdependency>\n    \u003CgroupId>org.mybatis\u003C\u002FgroupId>\n    \u003CartifactId>mybatis\u003C\u002FartifactId>\n    \u003Cversion>3.5.16\u003C\u002Fversion>\n\u003C\u002Fdependency>\n```\n\nJUnit5依赖：\n\n```xml\n\u003Cdependency>\n    \u003CgroupId>org.junit.jupiter\u003C\u002FgroupId>\n    \u003CartifactId>junit-jupiter\u003C\u002FartifactId>\n    \u003Cversion>5.8.1\u003C\u002Fversion>\n    \u003Cscope>test\u003C\u002Fscope>\n\u003C\u002Fdependency>\n```\n\nLombok依赖：\n\n```xml\n\u003Cdependency>\n    \u003CgroupId>org.projectlombok\u003C\u002FgroupId>\n    \u003CartifactId>lombok\u003C\u002FartifactId>\n    \u003Cversion>1.18.36\u003C\u002Fversion>\n\u003C\u002Fdependency>\n```",{"data":466,"status":460,"success":461},[467,472],{"id":8,"image":468,"link":469,"name":470,"type":471},"\u002Fimage\u002Fadv\u002Frainyun-2025-06.webp","https:\u002F\u002Fwww.rainyun.com\u002Fitbaima_","雨云优惠购","cloud",{"id":66,"image":473,"link":474,"name":475,"type":476},"\u002Fimage\u002Fadv\u002Fsimcard-2025-11.webp","https:\u002F\u002Fmall.itbaima.cn","号卡优惠","simcard"]