日志规范: 一些接口,提供给日志的实现框架设计的标准。日志框架: 牛人或者第三方公司已经做好的日志记录实现代码,后来者直接可以拿去使用。因为对 Commons Logging 的接口不满意,有人就搞了 SLF4J 。因为对 Log4j 的性能不满意,有人就搞了 Logback 。
日志规范大多是一些接口,提供给实现框架去设计的。
常见的规范是:
Commons Logging 简称:JCL
Simple Logging Facade for Java 简称:slf4j
常见实现框架:
Log4J
Logback
1、Logback日志框架
Logback 是由 log4j 创始人设计的另一个开源日志组件,性能比 log4j 要好官方网站:https://logback.qos.ch/index.htmlLogback是基于 slf4j 的日志规范实现的框架。
Logback主要分为三个技术模块:
logback -core : logback -core 模块为其他两个模块奠定了基础,必须有 。logback -classic :它是 log4j 的一个改良版本,同时它完整实现了 slf4j API 。logback -access 模块与 Tomcat 和 Jetty 等 Servlet 容器集成,以提供 HTTP 访问日志功能
①:在项目下新建文件夹lib,导入Logback的相关jar包到该文件夹下,并添加到项目依赖库中去。
idea全选jar包--右键---Add is library---ok
②:将Logback的核心配置文件logback.xml直接拷贝到src目录下(必须是src下)。
③:在代码中获取日志的对象
public static final Logger LOGGER = LoggerFactory.getLogger("类对象");
④:使用日志对象LOGGER调用其方法输出不能的日志信息
Logback日志系统的特性都是通过核心配置文件logback.xml控制的。
Logback日志输出位置、格式设置:
通过 logback.xml 中的 <append> 标签可以设置输出位置和日志信息的详细格式。通常可以设置 2 个日志输出位置: 一个是控制台、一个是系统文件中输出到控制台的配置标志
<appender name="CONSOLE" class="ch.qos.logback.core.ConsoleAppender">
输出到系统文件的配置标志
<appender name="FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
import org.slf4j.Logger; import org.slf4j.LoggerFactory; /* * 目标:快速搭建LogBack日志框架 记录程序的执行情况到控制台 到文件中。 * */ public class Test { // 创建Logback的日志对象,代表了日志技术 public static final Logger LOGGER = LoggerFactory.getLogger("Test.class"); public static void main(String[] args) { try { LOGGER.debug("main方法执行"); LOGGER.info("开始记录第二行"); int a = 10; int b = 0; LOGGER.trace("a=" +a); LOGGER.trace("b=" +b); System.out.println(a / b); } catch (Exception e) { e.printStackTrace(); LOGGER.error("功能出现异常," + e); } } }
日志级别设置:
日志级别
级别程度依次是 :TRACE < DEBUG < INFO < WARN < ERROR ; 默认 级别是 debug (忽略大小写),对应其方法。作用:用于控制系统中哪些日志级别是可以输出的,只输出级别不低于设定级别的日志信息。ALL 和 OFF 分别是打开全部日志信息,及关闭全部日志信息。具体在 <root level=“ INFO ”> 标签的 level 属性中 设置 日志 级别 。
电影购票系统:
电影类:
import com.itheima.run.MovieSystem; import java.math.BigDecimal; import java.math.RoundingMode; import java.util.Date; import java.util.List; public class Movie { private String name; private String actor; private double time; private double price; private int number; // 余票 private Date startTime; // 放映时间 public Movie() { } public Movie(String name, String actor, double time, double price, int number, Date startTime) { this.name = name; this.actor = actor; this.time = time; this.price = price; this.number = number; this.startTime = startTime; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getActor() { return actor; } public void setActor(String actor) { this.actor = actor; } public double getScore() { List<Double> scores = MovieSystem.MOVIES_SCORE.get(name); if(scores!=null && scores.size() > 0){ double sum = 0; for (Double score : scores) { sum += score; } return BigDecimal.valueOf(sum).divide(BigDecimal.valueOf(scores.size()), 2 , RoundingMode.UP).doubleValue(); }else { return 0; } } public double getTime() { return time; } public void setTime(double time) { this.time = time; } public double getPrice() { return price; } public void setPrice(double price) { this.price = price; } public int getNumber() { return number; } public void setNumber(int number) { this.number = number; } public Date getStartTime() { return startTime; } public void setStartTime(Date startTime) { this.startTime = startTime; } }
用户类:
/** 用户类(客户和商家的父类 ) */ public class User { private String loginName; // 假名 不能重复 private String userName; // 真名 private String passWord; private char sex; private String phone; private double money; public User(){ } public User(String loginName, String userName, String passWord, char sex, String phone, double money) { this.loginName = loginName; this.userName = userName; this.passWord = passWord; this.sex = sex; this.phone = phone; this.money = money; } public String getLoginName() { return loginName; } public void setLoginName(String loginName) { this.loginName = loginName; } public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; } public String getPassWord() { return passWord; } public void setPassWord(String passWord) { this.passWord = passWord; } public char getSex() { return sex; } public void setSex(char sex) { this.sex = sex; } public String getPhone() { return phone; } public void setPhone(String phone) { this.phone = phone; } public double getMoney() { return money; } public void setMoney(double money) { this.money = money; } }
商家角色:
public class Business extends User{ // 店铺名称 private String shopName; // 店铺地址 private String address; public String getShopName() { return shopName; } public void setShopName(String shopName) { this.shopName = shopName; } public String getAddress() { return address; } public void setAddress(String address) { this.address = address; } }
客户角色:
import java.util.HashMap; import java.util.Map; /** 客户角色 */ public class Customer extends User{ // 定义一个属性存储购买记录。 private Map<String, Boolean> buyMovies = new HashMap<>(); public Map<String, Boolean> getBuyMovies() { return buyMovies; } public void setBuyMovies(Map<String, Boolean> buyMovies) { this.buyMovies = buyMovies; } }
运行类:
import com.itheima.bean.Business; import com.itheima.bean.Customer; import com.itheima.bean.Movie; import com.itheima.bean.User; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.math.BigDecimal; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.*; public class MovieSystem { /** 定义系统的数据容器用户存储数据 1、存储很多用户(客户对象,商家对象) */ public static final List<User> ALL_USERS = new ArrayList<>(); /** 2、存储系统全部商家和其排片信息 。 商家1 = [p1,p2,p3,...] 商家2 = [p2,p3,...] ... */ public static final Map<Business, List<Movie>> ALL_MOVIES = new HashMap<>(); public static final Scanner SYS_SC = new Scanner(System.in); // 定义一个静态的User类型的变量记住当前登录成功的用户对象 public static User loginUser; public static SimpleDateFormat sdf = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss"); public static final Logger LOGGER = LoggerFactory.getLogger("MovieSystem.class"); /** 3、准备一些测试数据 */ static { Customer c = new Customer(); c.setLoginName("zyf888"); c.setPassWord("123456"); c.setUserName("黑马刘德华"); c.setSex('男'); c.setMoney(10000); c.setPhone("110110"); ALL_USERS.add(c); Customer c1 = new Customer(); c1.setLoginName("gzl888"); c1.setPassWord("123456"); c1.setUserName("黑马关之琳"); c1.setSex('女'); c1.setMoney(2000); c1.setPhone("111111"); ALL_USERS.add(c1); Business b = new Business(); b.setLoginName("baozugong888"); b.setPassWord("123456"); b.setUserName("黑马包租公"); b.setMoney(0); b.setSex('男'); b.setPhone("110110"); b.setAddress("火星6号2B二层"); b.setShopName("甜甜圈国际影城"); ALL_USERS.add(b); // 注意,商家一定需要加入到店铺排片信息中去 List<Movie> movies = new ArrayList<>(); ALL_MOVIES.put(b , movies); // b = [] Business b2 = new Business(); b2.setLoginName("baozupo888"); b2.setPassWord("123456"); b2.setUserName("黑马包租婆"); b2.setMoney(0); b2.setSex('女'); b2.setPhone("110110"); b2.setAddress("火星8号8B八层"); b2.setShopName("巧克力国际影城"); ALL_USERS.add(b2); // 注意,商家一定需要加入到店铺排片信息中去 List<Movie> movies3 = new ArrayList<>(); ALL_MOVIES.put(b2 , movies3); // b2 = [] } public static void main(String[] args) { showMain(); } /** 首页展示 */ private static void showMain() { while (true) { System.out.println("===============黑马电影首页================="); System.out.println("1、登录"); System.out.println("2、用户注册"); System.out.println("3、商家注册"); System.out.println("请输入操作命令:"); String command = SYS_SC.nextLine(); switch (command) { case "1": // 登录了 login(); break; case "2": // break; case "3": break; default: System.out.println("命令有误,请确认!"); } } } /** 登录功能 */ private static void login() { while (true) { System.out.println("请您输入登录名称:"); String loginName = SYS_SC.nextLine(); System.out.println("请您输入登录密码:"); String passWord = SYS_SC.nextLine(); // 1、根据登录名称查询用户对象。 User u = getUserByLoginName(loginName); // 2、判断用户对象是否存在,存在说明登录名称正确了 if(u != null){ // 3、比对密码是否正确 if(u.getPassWord().equals(passWord)){ // 登录成功了:... loginUser = u; // 记住登录成功的用户 LOGGER.info(u.getUserName() +"登录了系统~~~"); // 判断是用户登录的,还是商家登录的。 if(u instanceof Customer) { // 当前登录的是普通用户 showCustomerMain(); }else { // 当前登录的肯定是商家用户 showBusinessMain(); } return; }else { System.out.println("密码有毛病~~"); } }else { System.out.println("登录名称错误,请确认"); } } } /** 商家的后台操作界面 */ private static void showBusinessMain() { while (true) { System.out.println("============黑马电影商家界面==================="); System.out.println(loginUser.getUserName() + (loginUser.getSex()=='男'? "先生":"女士" + "欢迎您进入系统")); System.out.println("1、展示详情:"); System.out.println("2、上架电影:"); System.out.println("3、下架电影:"); System.out.println("4、修改电影:"); System.out.println("5、退出:"); System.out.println("请输入您要操作的命令:"); String command = SYS_SC.nextLine(); switch (command){ case "1": // 展示全部排片信息 showBusinessInfos(); break; case "2": // 上架电影信息 addMovie(); break; case "3": // 下架电影信息 deleteMovie(); break; case "4": // 修改电影信息 updateMovie(); break; case "5": System.out.println(loginUser.getUserName() +"请您下次再来啊~~~"); return; // 干掉方法 default: System.out.println("不存在该命令!!"); break; } } } /** 影片修改功能 */ private static void updateMovie() { System.out.println("================修改电影===================="); Business business = (Business) loginUser; List<Movie> movies = ALL_MOVIES.get(business); if(movies.size() == 0) { System.out.println("当期无片可以修改~~"); return; } // 2、让用户选择需要下架的电影名称 while (true) { System.out.println("请您输入需要修改的电影名称:"); String movieName = SYS_SC.nextLine(); // 3、去查询有没有这个影片对象。 Movie movie = getMovieByName(movieName); if(movie != null){ // 修改它 System.out.println("请您输入修改后的片名:"); String name = SYS_SC.nextLine(); System.out.println("请您输入修改后主演:"); String actor = SYS_SC.nextLine(); System.out.println("请您输入修改后时长:"); String time = SYS_SC.nextLine(); System.out.println("请您输入修改后票价:"); String price = SYS_SC.nextLine(); System.out.println("请您输入修改后票数:"); String totalNumber = SYS_SC.nextLine(); // 200\n while (true) { try { System.out.println("请您输入修改后的影片放映时间:"); String stime = SYS_SC.nextLine(); movie.setName(name); movie.setActor(actor); movie.setPrice(Double.valueOf(price)); movie.setTime(Double.valueOf(time)); movie.setNumber(Integer.valueOf(totalNumber)); movie.setStartTime(sdf.parse(stime)); System.out.println("恭喜您,您成功修改了该影片了!!!"); showBusinessInfos(); return; // 直接退出去 } catch (Exception e) { e.printStackTrace(); LOGGER.error("时间解析出了毛病"); } } }else { System.out.println("您的店铺没有上架该影片!"); System.out.println("请问继续修改吗?y/n"); String command = SYS_SC.nextLine(); switch (command) { case "y": break; default: System.out.println("好的!"); return; } } } } /** 影片下架功能 */ private static void deleteMovie() { System.out.println("================下架电影===================="); Business business = (Business) loginUser; List<Movie> movies = ALL_MOVIES.get(business); if(movies.size() == 0) { System.out.println("当期无片可以下架~~"); return; } // 2、让用户选择需要下架的电影名称 while (true) { System.out.println("请您输入需要下架的电影名称:"); String movieName = SYS_SC.nextLine(); // 3、去查询有没有这个影片对象。 Movie movie = getMovieByName(movieName); if(movie != null){ // 下架它 movies.remove(movie); System.out.println("您当前店铺已经成功下架了:" + movie.getName()); showBusinessInfos(); return; }else { System.out.println("您的店铺没有上架该影片!"); System.out.println("请问继续下架吗?y/n"); String command = SYS_SC.nextLine(); switch (command) { case "y": break; default: System.out.println("好的!"); return; } } } } /** 去查询当前商家下的排片 */ public static Movie getMovieByName(String movieName){ Business business = (Business) loginUser; List<Movie> movies = ALL_MOVIES.get(business); for (Movie movie : movies) { if(movie.getName().contains(movieName)) { return movie; } } return null; } /** 商家进行电影上架 Map<Business , List<Movie>> ALL_MOVIES u1 = [p1,p2,p3] u2 = [p1,p2,p3] */ private static void addMovie() { System.out.println("================上架电影===================="); // 根据商家对象(就是登录的用户loginUser),作为Map集合的键 提取对应的值就是其排片信息 :Map<Business , List<Movie>> ALL_MOVIES Business business = (Business) loginUser; List<Movie> movies = ALL_MOVIES.get(business); System.out.println("请您输入新片名:"); String name = SYS_SC.nextLine(); System.out.println("请您输入主演:"); String actor = SYS_SC.nextLine(); System.out.println("请您输入时长:"); String time = SYS_SC.nextLine(); System.out.println("请您输入票价:"); String price = SYS_SC.nextLine(); System.out.println("请您输入票数:"); String totalNumber = SYS_SC.nextLine(); // 200\n while (true) { try { System.out.println("请您输入影片放映时间:"); String stime = SYS_SC.nextLine(); // public Movie(String name, String actor, double time, double price, int number, Date startTime) // 封装成电影对象 ,加入集合movices中去 Movie movie = new Movie(name, actor ,Double.valueOf(time) , Double.valueOf(price) , Integer.valueOf(totalNumber) , sdf.parse(stime)); movies.add(movie); System.out.println("您已经成功上架了:《" + movie.getName() + "》"); return; // 直接退出去 } catch (ParseException e) { e.printStackTrace(); LOGGER.error("时间解析出了毛病"); } } } /** 定义一个静态的Map集合存储电影的评分 */ public static final Map<String , List<Double>> MOVIES_SCORE = new HashMap<>(); /** 展示商家的详细:展示当前商家的信息。 */ private static void showBusinessInfos() { System.out.println("================商家详情界面================="); LOGGER.info(loginUser.getUserName() +"商家,正在看自己的详情~~~"); // 根据商家对象(就是登录的用户loginUser),作为Map集合的键 提取对应的值就是其排片信息 :Map<Business , List<Movie>> ALL_MOVIES Business business = (Business) loginUser; System.out.println(business.getShopName() + "\t\t电话:" + business.getPhone() + "\t\t地址:" + business.getAddress() + "\t\t余额:" + business.getMoney()); List<Movie> movies = ALL_MOVIES.get(business); if(movies.size() > 0) { System.out.println("片名\t\t\t主演\t\t时长\t\t评分\t\t票价\t\t余票数量\t\t放映时间"); for (Movie movie : movies) { System.out.println(movie.getName()+"\t\t\t" + movie.getActor()+ "\t\t" + movie.getTime() + "\t\t" + movie.getScore() + "\t\t" + movie.getPrice() + "\t\t" + movie.getNumber() + "\t\t" + sdf.format(movie.getStartTime())); } }else { System.out.println("您的店铺当前无片在放映~~~~"); } } /** 客户操作界面 */ private static void showCustomerMain() { while (true) { System.out.println("============黑马电影客户界面==================="); System.out.println(loginUser.getUserName() + (loginUser.getSex()=='男'? "先生":"女士" + "欢迎您进入系统" + "\t余额:" + loginUser.getMoney())); System.out.println("请您选择要操作的功能:"); System.out.println("1、展示全部影片信息功能:"); System.out.println("2、根据电影名称查询电影信息:"); System.out.println("3、评分功能:"); System.out.println("4、购票功能:"); System.out.println("5、退出系统:"); System.out.println("请输入您要操作的命令:"); String command = SYS_SC.nextLine(); switch (command){ case "1": // 展示全部排片信息 showAllMovies(); break; case "2": break; case "3": // 评分功能 scoreMovie(); showAllMovies(); break; case "4": // 购票功能 buyMovie(); break; case "5": return; // 干掉方法 default: System.out.println("不存在该命令!!"); break; } } } private static void scoreMovie() { // 1、查询当前登录成功的用户历史购买记录,看哪些电影是它可以评分的。 Customer c = (Customer) loginUser; Map<String, Boolean> movies = c.getBuyMovies(); if(movies.size() == 0 ){ System.out.println("当前您没有看过电影,不能评价!"); return; } // 买过了 ,看哪些电影是它可以评分的。 movies.forEach((name, flag) -> { if(flag){ System.out.println(name +"此电影已评价"); }else { System.out.println("请您对:" + name +"进行打分(0-10):"); double score = Double.valueOf(SYS_SC.nextLine()); // 先根据电影名称拿到评分数据 List<Double> scores = MOVIES_SCORE.get(name); // MOVIES_SCORE = [ 名称=[10] , ... ] if(scores == null){ // 说明此电影是第一次评价 scores = new ArrayList<>(); scores.add(score); MOVIES_SCORE.put(name , scores); }else { scores.add(score); } movies.put(name, true); } }); } /** 用户购票功能 ALL_MOVIES = {b1=[p1,p2,p3,..] , b2=[p2,p3,...]} */ private static void buyMovie() { showAllMovies(); System.out.println("=============用户购票功能================="); while (true) { System.out.println("请您输入需要买票的门店:"); String shopName = SYS_SC.nextLine(); // 1、查询是否存在该商家。 Business business = getBusinessByShopName(shopName); if(business == null){ System.out.println("对不起,没有该店铺!请确认"); }else { // 2、此商家全部的排片 List<Movie> movies = ALL_MOVIES.get(business); // 3、判断是否存在上映的电影 if(movies.size() > 0) { // 4、开始进行选片购买 while (true) { System.out.println("请您输入需要购买电影名称:"); String movieName = SYS_SC.nextLine(); // 去当前商家下,查询该电影对象。 Movie movie = getMovieByShopAndName(business, movieName); if(movie != null){ // 开始购买 while (true) { System.out.println("请您输入要购买的电影票数:"); String number = SYS_SC.nextLine(); int buyNumber = Integer.valueOf(number); // 判断电影是否购票 if(movie.getNumber() >= buyNumber){ // 可以购买了 // 当前需要花费的金额 double money = BigDecimal.valueOf(movie.getPrice()).multiply(BigDecimal.valueOf(buyNumber)) .doubleValue(); if(loginUser.getMoney() >= money){ // 终于可以买票了 System.out.println("您成功购买了"+ movie.getName() + buyNumber + "张票!总金额是:" + money); // 更新自己的金额 更新商家的金额 loginUser.setMoney(loginUser.getMoney() - money); business.setMoney(business.getMoney() + money); movie.setNumber(movie.getNumber() - buyNumber); Customer c = (Customer) loginUser; // 记录购买电影的信息 // 第一个参数是购买的电影,第二个参数是没有评价的标记! c.getBuyMovies().put(movie.getName(), false); return;// 结束方法 }else { // 钱不够! System.out.println("是否继续~~"); System.out.println("是否继续买票?y/n"); String command = SYS_SC.nextLine(); switch (command) { case "y": break; default: System.out.println("好的!"); return; } } }else { // 票数不够 System.out.println("您当前最多可以购买:" + movie.getNumber()); System.out.println("是否继续买票?y/n"); String command = SYS_SC.nextLine(); switch (command) { case "y": break; default: System.out.println("好的!"); return; } } } }else { System.out.println("电影名称有毛病~~"); } } }else { System.out.println("该电影院关门了~~~"); System.out.println("是否继续买票?y/n"); String command = SYS_SC.nextLine(); switch (command) { case "y": break; default: System.out.println("好的!"); return; } } } } } public static Movie getMovieByShopAndName(Business business , String name){ List<Movie> movies = ALL_MOVIES.get(business); for (Movie movie : movies) { if(movie.getName().contains(name)){ return movie; } } return null; } /** 根据商家店铺名称查询商家对象 * @return */ public static Business getBusinessByShopName(String shopName){ Set<Business> businesses = ALL_MOVIES.keySet(); for (Business business : businesses) { if(business.getShopName().equals(shopName)){ return business; } } return null; } /** 用户功能:展示全部商家和其排片信息 */ private static void showAllMovies() { System.out.println("=============展示全部商家排片信息================="); ALL_MOVIES.forEach((business, movies) -> { System.out.println(business.getShopName() + "\t\t电话:" + business.getPhone() + "\t\t地址:" + business.getAddress()); System.out.println("\t\t\t片名\t\t\t主演\t\t时长\t\t评分\t\t票价\t\t余票数量\t\t放映时间"); for (Movie movie : movies) { System.out.println("\t\t\t" + movie.getName()+"\t\t\t" + movie.getActor()+ "\t\t" + movie.getTime() + "\t\t" + movie.getScore() + "\t\t" + movie.getPrice() + "\t\t" + movie.getNumber() + "\t\t" + sdf.format(movie.getStartTime())); } }); } public static User getUserByLoginName(String loginName){ for (User user : ALL_USERS) { // 判断这个用户的登录名称是否是我们想要的 if(user.getLoginName().equals(loginName)){ return user; } } return null; // 查询此用户登录名称 } }