zoukankan      html  css  js  c++  java
  • 带条件的分页【重点】

               

    二、前台

    1、不带条件的查询

        [1 ]实现Dao方法:getPageList ()、getTotalItemNum ()
        ◆在BookDao中写接口方法
        /**
         * 获取在指定条件下的分页数据
         * @param pageNo 不能使用pageCondition返回的pageNoStr。要使用经过Page对象纠正的
         * @param pageSize
         * @param pageCondition
         * @return
         */
        List<Book> getPageList (int pageNo, int pageSize, PageCondition pageCondition);
        
        /**
         * 带条件的查询
         * @param pageCondition
         * @return 返回在指定的条件下一共同拥有多少条记录
         */
        int getTotalItemNum (PageCondition pageCondition );
        
        
        ◇此时须要更改Dao :
             /**
             * 获取单一值的接口方法
             * @param connection
             * @param sql
             * @param params
             * @return
             */
             <V > V getSingleValue (Connection connection , String sql , Object ... params);
        
        ◇DaoImpi中实现该接口方法:
            @Override
             public <V > V getSingleValue (Connection connection , String sql,
                    Object ... params ) {
                V v = null;
                 try {
                    v = (V ) runner .query (connection , sql , new ScalarHandler(), params);
                 } catch (SQLException e ) {
                     // TODO Auto-generated catch block
                    e .printStackTrace ();
                 }
                 return v ;
             }
        
        [2 ]实现Page类
        package com.atguigu.bookstore.fun;
        import java .util .List ;

        /**
         * 封装分页信息的类
         * @author Phenix
         *
         * @param <T>相应详细的实体类
         */
        public class Page<T> {
            
             //代表当前页码
             private int pageNo ;
            
             //页面上的数据
             private List <T > list ;
            
             //每页显示的数据数量
             public static final int PAGE_SIZE = 3;
            
             //总记录数
             private int totalItemNum ;
            
             //总页数
             private int totalPageNum ;
            
            
             public Page (String pageNoStr , int totalItemNum ) {
                
                 //先获取总记录数,以计算总页数
                 this.totalItemNum = totalItemNum ;
                
                 //计算总页数
                 this.totalPageNum = (totalItemNum % PAGE_SIZE == 0) ? totalItemNum/ PAGE_SIZE : totalItemNum/PAGE_SIZE + 1 ;
                
                 //对pageNoStr进行类型转换
                pageNo = 1;
                 try {
                    pageNo = Integer .parseInt (pageNoStr );
                 } catch (NumberFormatException e ) {/*假设转换失败则保持默认值*/ }

                 //限定pageNo的有效范围
                 if(pageNo < 1) {
                    pageNo = 1;
                 }
                
                 if(pageNo > totalPageNum ) {
                    pageNo = totalPageNum ;
                 }
                
             }

             public List <T > getList () {
                 return list ;
             }

             public void setList (List <T > list ) {
                 this.list = list;
             }

             public int getPageNo () {
                 return pageNo ;
             }

             public int getTotalItemNum () {
                 return totalItemNum ;
             }

             public int getTotalPageNum () {
                 return totalPageNum ;
             }

             public boolean isHasPrev () {
                 return pageNo > 1;
             }
            
             public boolean isHasNext () {
                 return pageNo < totalPageNum ;
             }
            
             public int getPrevNo () {
                 return pageNo - 1;
             }
            
             public int getNextNo () {
                 return pageNo + 1;
             }
            
        }
        
        [3 ]实现与分页相关的Service层代码
        ◇BookService中加入例如以下接口方法:获取book类的分页信息
            Page <Book > getBookPage (String pageNoStr );
        ◇在BookServiceImpi中实现该方法
            @Override
             public Page <Book > getBookPage (PageCondition pageCondition) {
             int totalItemNum = bookDao.getTotalItemNum( pageCondition);
            
            Page <Book > page = new Page<>(pageCondition. getPageNoStr(), totalItemNum);
            
             //这里。pageNo必须使用Page类纠正过的,不能使用pageCondition中的pageNoStr
            List <Book > list = bookDao .getPageList (page .getPageNo (), Page.PAGE_SIZE, pageCondition);
            
            page .setList (list );
            
             return page ;//返回book类的分页信息
        }

        ◆在client .jsp请求
        <iframe src ="client/BookClientServlet?method=getPage" name="targetFrame"></iframe>
        ◆BookClientServlet中进行转发
             protected void getPage (HttpServletRequest request ,
                HttpServletResponse response ) throws ServletException, IOException {
            
            String pageNoStr = request .getParameter ("pageNo");  
            Page <Book > page = bookService .getBookPage (pageNoStr );   
            WebUtils .myForward (request , response , "/client/book/bookList.jsp" , "page", page);
            
        }
        ◆在bookList .jsp中接收数据
            $ {page .list }
        
        

        [4 ]页面 (文本框输入数字跳转页面 )
            ① 给文本框加一个Id 属性: <input id ="pageCode" class="inpType" type="text" name= "pageNo" />
            ②导入jquery: <script type ="text/javascript" src="script/jquery-1.7.2.js"></script>【已经加过base标签】
            ③书写JS代码:
            版本号一:能够容错,由于Page类内置了容错校验
             <script type ="text/javascript">
                $ (function (){
                    $ ("#pageCode").change(function(){
                         //去掉前后空格
                        var pageCode =$ .trim (this. value);
                         //推断用户输入的页码是否为数字
                         if(isNaN(pageCode)){
                             //假设不是将文本框置空。并停止函数的运行
                             this.value="" ;
                             return false;
                         }
                         //假设是数字。则跳转到指定的页码
                        var url ="${pageContext.request.contextPath }/"+
                         "client/BookClientServlet?method=getPage&pageNo=" +pageCode ; 
                        window .location .href =url ;
                     });
                 });
             </script >
        

        
    2、在首页显示真实的分类数据
        ①创建ClientServlet ,用于跳转到前台页面 ,跟实体类无关。和前台页面相关
        ②改动index .jsp中的代码
                 <jsp :forward page="client/ClientServlet?method=toClientUI"></jsp :forward >
        ③在toClientUI ()方法中获取数据库分类列表数据,跳转到client .jsp显示
             public class ClientServlet extends BaseServlet {
             private static final long serialVersionUID = 1L;
            
             private CateService cateService =new CateServiceImpl ();
            
             protected void toClientUI (HttpServletRequest request , HttpServletResponse response) throws ServletException, IOException {
                List <Category > cateList = cateService .getCateList ();
                
                WebUtils .myForward (request , response , "/client/client.jsp", "cateList", cateList);
             }

             }
        
        注意:改动web .xml将ClientServlet注冊一下。
        <servlet >
             <display -name >ClientServlet </display -name >
             <servlet -name >ClientServlet </servlet -name >
             <servlet -class> com.atguigu.bookstore.servlet.client.ClientServlet </servlet -class>
        </servlet >
        <servlet -mapping >
          <servlet -name >ClientServlet </servlet -name >
          <url-pattern>/client /ClientServlet </url -pattern >
        </servlet -mapping >
          
        ④改动client .jsp 中的分类列表
            <li>所有分类 </li >
             <c :forEach items ="${cateList }" var ="category" >
                 <li ><a href ="#"> ${category.cateName }</a></li >
             </c :forEach >
          
        ⑤后台中跳转到前台页面的超链接也要跟着改动
        <a href="client/ClientServlet?

    method=toClientUI" >进入前台 </a >

        

    3、★带条件的查询★重点★

        难点:查询条件时有时无,要做到适配不同的情况
        ①查询条件包含
            pageNo页码
            minPrice 价格区间最小值
            maxPrice 价格区间最大值
            cateId   分类Id
        ②将查询条件封装为PageCondition类
             (1) 为每一个查询条件设置默认值
                    pageNo 页码,默认值:1,终于还是交给Page类纠正
                    minPrice 默认值:0
                    maxPrice 默认值:Integer .MAX_VALUE
                    cateId 默认值:null为了避免有可能产生的歧义,所以应声明为Integer类型
             (2) 在构造器中。获取String类型的參数,进行类型转换,若不论什么一个条件数据转换失败则保持默认值
             (3) 提供getXxx()方法
            代码例如以下:
             public class PageCondition {
                
             //  (1)pageNo 页码 声明为String类型是为了直接交给Page类的构造器
                 private String pageNoStr = "1";
                
             //  (2)minPrice 价格区间的最小值 默认值的作用是在页面没有传入该參数时仍然能够查询
                 private int minPrice = 0;
                
             //  (3)maxPrice 价格区间的最大值
                 private int maxPrice = Integer .MAX_VALUE ;
                
             //  (4)cateId   分类ID
                 private Integer cateId = null;
                
                 public PageCondition (String pageNoStr , String minPriceStr, String maxPriceStr, String cateIdStr) {
                    
                     this.pageNoStr = pageNoStr ;
                    
                     //转换失败。使用默认值
                     try {
                         this.minPrice = Integer.parseInt( minPriceStr);
                     } catch (NumberFormatException e ) {/*保持默认值*/}
                    
                     try {
                         this.maxPrice = Integer.parseInt( maxPriceStr);
                     } catch (NumberFormatException e ) {/*保持默认值*/}
                    
                     try {
                         this.cateId = Integer.parseInt(cateIdStr);
                     } catch (NumberFormatException e ) {/*保持默认值*/}
                    
                 }

                 public String getPageNoStr () {
                     return pageNoStr ;
                 }

                 public int getMinPrice () {
                     return minPrice ;
                 }

                 public int getMaxPrice () {
                     return maxPrice ;
                 }

                 public Integer getCateId () {
                     return cateId ;
                 }

                @Override
                 public String toString () {
                     return "PageCondition [pageNoStr=" + pageNoStr + ", minPrice="
                             + minPrice + ", maxPrice=" + maxPrice + ", cateId=" + cateId
                             + "]";
                 }
                
             }
        ③在Dao中升级原有的分页方法
             (1) int getTotalItemNum (PageCondition pageCondition )。依据相关条件进行相关查询
            ·原本不带条件的SQL语句:SELECT COUNT (*) FROM book
            ·带所有条件的SQL语句
                SELECT
                  COUNT (*)
                FROM
                  book
                WHERE price <= 10000
                  AND price >= 0
                  AND cate_id = 1
            ·适配详细情况的SQL语句
                pageCondition .getCateId ()的返回值是否为null。决定是否附加“AND cate_id = 1 
                ○第一部分
                SELECT
                  `book_id` bookId ,
                  `book_name` bookName ,
                  `author` author ,
                  `price` price ,
                  `store_num` storeNum ,
                  `salse_amount` salseAmount ,
                  `imp_path` imgPath ,
                  `cate_id` cateId
                FROM
                  book
                WHERE price >= 0
                  AND price <= 10000
                ○第二部分
                  AND cate_id = 1
                  依据pageCondition .getCateId ()的返回值是否为null,决定是否附加
                ○第三部分
                  LIMIT ?

    , ?

                
                
            ◆详细更改例如以下:

            ★1、在BookDao .java中重载两个方法

             /**
             * 获取在指定条件下的分页数据
             * @param pageNo 表示当前页,不能使用pageCondition返回的pageNoStr。要使用经过Page对象纠正的
             * @param pageSize 页面显示的条目数
             * @param pageCondition
             * @return
             */
            List <Book > getPageList (int pageNo, int pageSize, PageCondition pageCondition);
            
             /**
             * 带条件的查询
             * @param pageCondition
             * @return 返回在指定的条件下一共同拥有多少条记录
             */
             int getTotalItemNum (PageCondition pageCondition );
        

            ★2、在BookDaoImpl .java中实现上述两个方法

            @Override
             public List <Book > getPageList (int pageNo, int pageSize,
                    PageCondition pageCondition ) {
                
                Connection connection = JDBCUtils .getConnection ();
                
                String sql = "SELECT `book_id` bookId,"
                         + "`book_name` bookName,"
                         + "`author` author,"
                         + "`price` price,"
                         + "`store_num` storeNum,"
                         + "`salse_amount` salseAmount,"
                         + "`imp_path` imgPath,`cate_id` cateId "
                         + "FROM book WHERE price>=?

    and price<=?

    ;

                
                Integer cateId = pageCondition .getCateId ();
                
                 //依据实际情况决定是否增加这个条件
                 if(cateId != null) sql = sql + " and `cate_id`=" + cateId;
                
                sql = sql + " LIMIT ?,?";
                
                List <Book > list = this.getBeanList (connection , sql ,
                        pageCondition .getMinPrice (),
                        pageCondition .getMaxPrice (),
                         (pageNo -1)* pageSize,
                        pageSize );
                
                JDBCUtils .releaseConnection (connection );
                
                 return list ;
                
             }
             //获取总记录数
            @Override
             public int getTotalItemNum (PageCondition pageCondition ) {
                
                Connection connection = JDBCUtils .getConnection ();
                
                String sql = "SELECT COUNT(*) FROM book WHERE price >= ? AND price <= ?

    ";

                
                Integer cateId = pageCondition .getCateId ();
                
                 if(cateId != null) sql = sql + " AND cate_id = " + cateId;
                
                 long itemNum = this.getSingleValue (connection , sql , pageCondition.getMinPrice(), pageCondition.getMaxPrice());
                
                JDBCUtils .releaseConnection (connection );
                
                 return (int) itemNum;
             }
        
            ☆对以上更改进行单元測试
             private BookDao bookDao = new BookDaoImpl ();
            @Test
             public void testGetPageListInCon () {

                PageCondition pageCondition = new PageCondition ("1", null, null, null);
                
                List <Book > list = bookDao .getPageList (1, 3 , pageCondition);
                
                 for (Iterator iterator = list .iterator (); iterator.hasNext();) {
                    Book book = (Book ) iterator .next ();
                    System .out .println (book );
                 }
                
             }
            @Test
             public void testGetItemNumInCon () {
                
                PageCondition pageCondition = new PageCondition ("1", "30", "50", "2" );
                
                 int totalItemNum = bookDao.getTotalItemNum( pageCondition);
                
                System .out .println (totalItemNum );
                
             }
        

            ★3、在BookService中对该方法重载

            Page <Book > getBookPage (PageCondition pageCondition );
            

            ★4、在BookServiceImpl实现该接口

            @Override
             public Page <Book > getBookPage (PageCondition pageCondition) {
             //总记录数从Dao中获取,先获取总记录数,以计算总页数
             int totalItemNum = bookDao.getTotalItemNum( pageCondition);
             //创建分页类对象,并在构造器中对传入的当前页和总记录数进行校验
            Page <Book > page = new Page<Book>( pageCondition.getPageNoStr(), totalItemNum );
            
             //这里pageNo必须使用Page类纠正过的。不能使用PageCondition中的pageNoStr,它不具备纠正能力
             //getPageList()获取在指定条件下的分页数据
            List <Book > list = bookDao .getPageList (page .getPageNo (), Page.PAGE_SIZE, pageCondition);
             //将page对象设置到list集合其中
            page .setList (list );
            
             return page ;
             }
            

            ☆对该接口实行单元測试:

            @Test
             public void testGetPageInCon () {
                 //查询第一页。价格在30-77之间。书的分类Id是5的有记录
                PageCondition pageCondition = new PageCondition ("1", "30", "77", "5" );   
                Page <Book > page = bookService.getBookPage( pageCondition);     
                 int totalItemNum = page .getTotalItemNum ();  
                 int totalPageNum = page .getTotalPageNum ();
                 int pageNo = page .getPageNo ();  
                Iterator <Book > iterator = page .getList ().iterator ();
                
                 while (iterator .hasNext ()) {            
                    Book book = (Book ) iterator .next ();         
                    System .out .println (book .getBookId ());
                     //5 7 18
                 }
                
                System .out .println ("总记录数:"+totalItemNum );   //总记录数:3
                System .out .println ("总页数:"+totalPageNum);     //总页数:1
                System .out .println ("当前页:"+pageNo);//    当前页:1   
             }
        

            ★5、在BookClientServlet中添加getPageInCondition ()方法

             protected void getPageInCondition (HttpServletRequest request,
                HttpServletResponse response ) throws ServletException, IOException {
            
             //获取请求參数
            String pageNoStr = request .getParameter ("pageNo");
            String minPriceStr = request .getParameter ("minPrice");
            String maxPriceStr = request .getParameter ("maxPrice");
            String cateIdStr = request .getParameter ("cateId");
            
             //封装条件对象
            PageCondition pageCondition = new PageCondition(pageNoStr, minPriceStr, maxPriceStr, cateIdStr );
            
            Page <Book > page = bookService.getBookPage( pageCondition);
            
            WebUtils .myForward (request , response , "/client/book/bookList.jsp" , "page", page);
            
             }
            

            ★6、在bookList .jsp中更改价格查询的提交地址

             <form action="client/BookClientServlet?method=getPageInCondition" method="post">
                价格
                 <input type ="text" class= "inpType" name ="minPrice" />-
                 <input type ="text" class= "inpType" name ="maxPrice" />
                 <input type ="submit" value ="查询" />
             </form >
            

            ★7、在client .jsp中做相同更改

             <li >所有分类 </li >
             <c :forEach items ="${cateList }" var ="category" >
             <li ><a href="client/BookClientServlet?

    method=getPageInCondition&cateId=${category.cateId }" >$ {category .cateName }</a ></li >

             </c :forEach >
            
            

            ★8、给分类信息增加target属性使其在mainContent中显示

             <li >所有分类 </li >
             <c :forEach items ="${cateList }" var ="category" >
             <li ><a target ="targetFrame" href="client/BookClientServlet?method=getPageInCondition&cateId=${category.cateId }" >$ {category .cateName }</a ></li >
             </c :forEach >
        
        
            

    4、在页面上保持查询条件

        [1 ]会导致“丢失”查询条件的行为

             (1) 翻页的超链接
                ·可以保持的条件:pageNo
                ○分类、价格会丢失
                →可将下面条件附着在超链接之后
                 &cateId =$ {param .cateId }&minPrice =$ {param .minPrice }&maxPrice= ${param.maxPrice }
                比如:
                 <a href="client/BookClientServlet?

    method=getPageInCondition&pageNo=${page.prevNo }

                 &cateId =$ {param .cateId }&minPrice =$ {param .minPrice }&maxPrice= ${param.maxPrice }">上一页</a>

                
        ·改进:对超链接进行批量操作
        <script type ="text/javascript">
        $(function(){
            var conditonStr ="&cateId=${param.cateId }&minPrice=${param.minPrice }&maxPrice=${param.maxPrice }";
            $ ("a"). each(function(){
                 this.href=this .href +conditonStr ;
             });
            
            $ ("#pageCode").change(function(){
                 //去掉前后空格
                var pageCode =$ .trim (this. value);
                 //推断用户输入的页码是否为数字
                 if(isNaN(pageCode)){
                     //假设不是将文本框置空。并停止函数的运行
                     this.value="" ;
                     return false;
                 }
                 //假设是数字。则跳转到指定的页码
                var url ="${pageContext.request.contextPath }/"+
                 "client/BookClientServlet?method=getPageInCondition&pageNo=" +pageCode +conditonStr ;  
                window .location .href =url ;
             });
        });
        </script >
        
        
             (2) 跳转页面的文本框
                ·可以保持的条件:pageNo
                ○分类、价格会丢失
                
            ◇在bookList .jsp中改动JS代码中的URL地址
             <script type ="text/javascript">
            $ (function (){
                $ ("#pageCode").change(function(){
                     //去掉前后空格
                    var pageCode =$ .trim (this. value);
                     //推断用户输入的页码是否为数字
                     if(isNaN(pageCode)){
                         //假设不是将文本框置空,并停止函数的运行
                         this.value="" ;
                         return false;
                     }
                     //假设是数字。则跳转到指定的页码
                    var url ="${pageContext.request.contextPath }/"+
                     "client/BookClientServlet?method=getPageInCondition&pageNo=" +pageCode +
                     "&cateId=${param.cateId }&minPrice=${param.minPrice }&maxPrice=${param.maxPrice }"; 
                    window .location .href =url ;
                 });
             });
            </script >
            注意:方法名要更改为:getPageInCondition    
            疑问?EL表达式能写在JS里面吗?
            解答:能够的。由于Jsp页面的翻译和执行先于JS ,浏览器取得该超链接地址是被翻译过的。
            
             (3) 查询价格的表单提交
                ·可以保持价格信息
                ○会丢失pageNo和分类
                因此要附着: &pageNo =$ {param .pageNo }&cateId= ${param.cateId }
                
            ◇将其附着在表单提交的地址后面
             <form action="client/BookClientServlet?method=getPageInCondition&pageNo=${param.pageNo }&cateId=${param.cateId }" method="post">
                价格
                 <input type ="text" class= "inpType" name ="minPrice" value="${param.minPrice }"/>-
                 <input type ="text" class= "inpType" name ="maxPrice" value="${param.maxPrice }" />
                 <input type ="submit" value ="查询" />
             </form >
                
                
             (4) 点击分类超链接【会话控制能够解决】
                分类信息client .jsp和bookList .jsp非同一个页面在bookList .jsp中即使用param对象,在client .jsp中也获取不到它的值
                因此临时先不动分类Client .jsp,之后能够将pageCondition放到session里面,那么这两个页面就能够协同工作了。
                ·可以保持分类
                ○丢失分类、价格、页码
                    
                

        [2 ]保持查询条件

             (1) 在跳转页面时将已存在的查询条件携带提交
             (2) ${param.xxx }
            翻页的超链接:自己发出的不存在保持查询条件的问题。能够将查询条件附着在超链接上面,
             <a href ="...let?method=getPageInCondition&pageNo=${num }&minPrice=${param.minPrice }"> ${num }</a>
             <a href ="...let?method=getPageInCondition&pageNo=${num }&minPrice=30"> ${num }</a>
             (3) 注意:Servlet方法的名字要改成getPageInCondition
        
        
        
        [3 ]在bookList .jsp中显示当前的分类名称
        ◇在client .jsp中将每个分类都附着cateName信息
        <li>所有分类 </li >
        <c:forEach items ="${cateList }" var ="category" >
             <li ><a target ="targetFrame" href="client/BookClientServlet?

    method=getPageInCondition&cateId=${category.cateId }&cateName=${category.cateName }">${ category.cateName }</a></li >

        </c:forEach>
        
        ◇在bookList .jsp中获取cateName
        //<td colspan="2">当前分类:${param.cateName }</td>
        <td colspan ="2"> 当前分类:${param.cateName==null ? '所有分类':param.cateName }</td>
        
        ◆当点击所有分类的时候显示空白,如今要给所有分类也加上超链接以便于用户能回到所有分类
        <li><a target ="targetFrame" href="client/BookClientServlet?

    method=getPageInCondition&cateName=所有分类" >所有分类 </a ></li >

        

        眼下的局限性:

        仅仅能在当前分类下保持查询(先点击“科学”再点击价格能够保持,可是再次点击别的分类就保持不了)
        
        

        

    5、点击图书名称显示单本书的信息  

        ①给书名加超链接
        <li>书名: <a href="client/BookClientServlet?method=getBook&bookId=${book.bookId }">${book.bookName }</a></li >
        
        ②在BookClientServlet中写getBook ()方法
        protected void getBook(HttpServletRequest request ,
                HttpServletResponse response ) throws ServletException, IOException {
            
            String bookId = request .getParameter ("bookId");
            
            Book book = bookService .getBookById (bookId );
            
            WebUtils .myForward (request , response , "/client/book/book.jsp" , "book", book);
            
        }
            
        ③创建book .jsp页面用于显示图书信息
        注意:编码为UTF -8
        ◇导入: <base href ="http://${pageContext.request.serverName }:${pageContext.request.serverPort }${pageContext.request.contextPath }/" />
        ◇<link rel ="stylesheet" type ="text/css" href="style/css.css" />
        ◇<script type ="text/javascript" src="script/jquery-1.7.2.js"></script>
        <script type ="text/javascript">
            $ (function (){
                $ ("button").click(function(){       
                     //借助window.history.go(-1)实现后退功能
                    window .history .go (-1);       
                 }); 
             });
        </script >
            
        ◇在body中书写
        <body>
             <img class="bookImg" src="${book.imgPath }" >
             <p >书名:$ {book .bookName }</p >
             <p >作者:$ {book .author }</p >
             <p >价格:$ {book .price }</p >
             <p >库存:$ {book .storeNum }</p >
             <p >销量:$ {book .salseAmount }</p >
             <p ><button >回到上一页 </button ></p >
        </body> 
            
  • 相关阅读:
    excel
    AWS学习之EC2
    约瑟夫问题
    centos7 系統vps安裝mysql5.6及設置本地遠程連接筆記
    搜索框的测试checklist
    产品把整个项目组拉走去创业,这是什么神操作
    python基础-python函数参数为print语句时的输出
    python基础学习笔记-切片难点
    session 、cookie、token的区别(转)
    robot framework python3环境下学习笔记(1)——安装robot framework
  • 原文地址:https://www.cnblogs.com/cynchanpin/p/6985817.html
Copyright © 2011-2022 走看看