【Spring Boot论坛项目实战】3、开发社区核心功能

文章列表:
1、初识Spring Boot,开发社区首页
2、开发社区登录模块
3、开发社区核心功能

开发社区核心功能

1 过滤敏感词

1111
前缀树:根节点为空,除了根节点外的节点只包含一个字母
检测敏感词需要三个指针:第一个指针指向树,第二、三个指针指向字符串,遍历时分别指向敏感词的开头与结尾
检测到的结果存到StringBuilder里

1.1 定义前缀树

//前缀树
    private class TrieNode{
        //关键词结束的标识
        private boolean isKeywordEnd = false;
        //子节点(key是子节点字符,value是子节点)
        private Map<Character, TrieNode> subNodes = new HashMap<>();

        public boolean isKeywordEnd() {
            return isKeywordEnd;
        }

        public void setKeywordEnd(boolean keywordEnd) {
            isKeywordEnd = keywordEnd;
        }

        //添加子节点
        public void addSubNode(Character c,TrieNode node){
            subNodes.put(c, node);
        }
        //获取子节点
        public TrieNode getSubNode(Character c){
            return subNodes.get(c);
        }
    }

1.2 根据敏感词,初始化前缀树

    //根节点
    private TrieNode rootNode = new TrieNode();
    //初始化
    @PostConstruct//当容器实例化这个Bean之后,在调用构造方法之后这个方法被自动调用
    public void init(){
        try(
                InputStream is = this.getClass().getClassLoader().getResourceAsStream("sensitive-words.txt");
                BufferedReader reader = new BufferedReader(new InputStreamReader(is));
                ){
            String keyword;
            while((keyword=reader.readLine()) != null){
                //添加到前缀树
                this.addKeyword(keyword);
            }
        }catch (IOException e){
            logger.error("加载敏感词文件失败"+e.getMessage());
        }

    }
    //将一个敏感词添加到前缀树中去
    private void addKeyword(String keyword){
        TrieNode tempNode = rootNode;
        for(int i=0;i<keyword.length();i++){
            char c = keyword.charAt(i);
            TrieNode subNode = tempNode.getSubNode(c);

            if(subNode==null){
                //初始化子节点
                subNode = new TrieNode();
                tempNode.addSubNode(c,subNode);
            }

            //指向子节点,进入下一轮循环
            tempNode=subNode;
            //设置结束的标识
            if(i==keyword.length()-1){
                tempNode.setKeywordEnd(true);
            }
        }
    }

1.3 编写过滤敏感词的方法

//替换符
    private static final String REPLACEMENT = "***";
/**
     * 过滤敏感词
     * @param text 待过滤的文本
     * @return 过滤后的文本
     */
    public String filter(String text){
        if(StringUtils.isBlank(text)){
            return null;
        }
        //指针1:指向树的节点
        TrieNode tempNode = rootNode;
        //指针2:遍历字符串指针,指向敏感词首位
        int begin = 0;
        //指针3:指向敏感词末尾
        int position=0;
        //结果
        StringBuilder sb = new StringBuilder();

        while (position < text.length()) {
            char c = text.charAt(position);

            // 跳过符号
            if (isSymbol(c)) {
                // 若指针1处于根节点,将此符号计入结果,让指针2向下走一步
                if (tempNode == rootNode) {
                    sb.append(c);
                    begin++;
                }
                // 无论符号在开头或中间,指针3都向下走一步
                position++;
                continue;
            }

            // 检查下级节点
            tempNode = tempNode.getSubNode(c);
            if (tempNode == null) {
                // 以begin开头的字符串不是敏感词
                sb.append(text.charAt(begin));
                // 进入下一个位置
                position = ++begin;
                // 重新指向根节点
                tempNode = rootNode;
            } else if (tempNode.isKeywordEnd()) {
                // 发现敏感词,将begin~position字符串替换掉
                sb.append(REPLACEMENT);
                // 进入下一个位置
                begin = ++position;
                // 重新指向根节点
                tempNode = rootNode;
            } else {
                // 检查下一个字符
                position++;
            }
        }

        // 将最后一批字符计入结果
        sb.append(text.substring(begin));

        return sb.toString();
    }

    //判断是否为符号
    private boolean isSymbol(Character c){
        return !CharUtils.isAsciiAlphanumeric(c) && (c<0x2E80 || c>0x9FFF);//东亚文字范围
    }

1.4 测试

@RunWith(SpringRunner.class)
@SpringBootTest
@ContextConfiguration(classes = CommunityApplication.class)
public class SensitiveTests {
    @Autowired
    private SensitiveFilter sensitiveFilter;
    @Test
    public void testSeneitiveFilter(){
        String text="这里可以赌博,可以嫖娼,可以吸毒,可以开票";
        text = sensitiveFilter.filter(text);
        System.out.println(text);

        text="这里可以%赌%博%,可以%嫖%娼%,可以%吸%毒%,可以%开%票%";
        text = sensitiveFilter.filter(text);
        System.out.println(text);
    }
}

2 发布帖子

在这里插入图片描述
发布帖子需要用到异步请求:当前网页不刷新,访问服务器,服务器会返回一些结果,这个结果不是网页,通过结果对当前网页做局部刷新

2.1 AJAX

AJAX:异步的js和xml

2.2 示例:使用jQuery发送AJAX请求

    public static String getJSONString(int code, String msg, Map<String,Object> map){
        JSONObject json = new JSONObject();
        json.put("code",code);
        json.put("msg", msg);
        if(map!=null){
            for(String key : map.keySet()){
                json.put(key, map.get(key));
            }
        }
        return json.toJSONString();
    }

    public static String getJSONString(int code, String msg){
        return getJSONString(code, msg, null);
    }

    public static String getJSONString(int code){
        return getJSONString(code, null, null);
    }

    //main方法用于测试JSON方法
    public static void main(String[] args){
        Map<String, Object> map = new HashMap<>();
        map.put("name", "zhangsan");
        map.put("age", 25);
        System.out.println(getJSONString(0, "ok", map));
    }
@Controller
@RequestMapping("/alpha")
public class AlphaController {

    //AJAX示例
    @RequestMapping(path = "/ajax",method = RequestMethod.POST)
    @ResponseBody//异步请求,不返回网页,只返回字符串
    public String testAjax(String name,int age){
        System.out.println(name);
        System.out.println(age);
        return CommunityUtil.getJSONString(0,"操作成功");
    }
}
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>AJAX</title>
</head>
<body>
<p>
    <input type="button" value="发送" onclick="send();">
</p>

<script src="https://code.jquery.com/jquery-3.3.1.min.js" crossorigin="anonymous"></script>
<script>
    function send() {
        $.post(
            "/community/alpha/ajax",
            {"name":"张三","age":23},
            function(data) {
                console.log(typeof(data));
                console.log(data);

                data = $.parseJSON(data);
                console.log(typeof(data));
                console.log(data.code);
                console.log(data.msg);
            }
        );
    }
</script>
</body>
</html>

2.3 实践:采用AJAX请求,实现发布帖子的功能

1.数据访问层:增加一个插入帖子的方法
2.业务层:增加一个发帖方法(调用数据访问层),转义HTML字符,敏感词过滤
3.视图层:异步方式,需要写js代码
注意返回的是字符串,要加上@ResponseBody
方法的参数:页面上传入的内容
4.修改前端网页为动态引擎

3 帖子详情

开发过程:很详细了!
在这里插入图片描述
Controller层知识点:
1.@PathVariable:在路径中获取参数

@RequestMapping(path="/detail/{discussPostId}",method = RequestMethod.GET)
public String getDiscussPost(@PathVariable("discussPostId") int discussPostId, Model model){...}

2.通过model将参数发送给模板引擎

 model.addAttribute("post", post);
<span th:utext="${post.title}">

3.返回模板路径

return "site/discuss-detail";

4 事务管理

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

4.1 声明式事务

@Service
public class AlphaService {

    @Autowired
    private UserMapper userMapper;
    @Autowired
    private DiscussPostMapper discussPostMapper;
    //propagation传播机制 a调用b
    //REQUIRED 支持当前事务,如果不存在,就创建新事物
    //REQUIRES_NEW 创建新事物且暂停当前事务
    //NESTED 如果当前存在事务,则嵌套在该事务中执行;否则和REQUIRED一样
    @Transactional(isolation = Isolation.READ_COMMITTED,propagation = Propagation.REQUIRED)
    public Object save1(){
        //新增用户
        User user = new User();
        user.setUsername("alpha");
        user.setSalt(CommunityUtil.generateUUID().substring(0,5));
        user.setPassword(CommunityUtil.md5("123"+user.getSalt()));
        user.setEmail("alpha@qq.com");
        user.setHeaderUrl("http://image.nowcoder.com/head/99t.png");
        user.setCreateTime(new Date());

        userMapper.insertUser(user);

        //新增帖子
        DiscussPost post = new DiscussPost();
        post.setUserId(user.getId());
        post.setTitle("hello");
        post.setContent("hi");
        post.setCreateTime(new Date());
        discussPostMapper.insertDiscussPost(post);

        Integer.valueOf("abc");

        return "ok";
    }
}
@RunWith(SpringRunner.class)
@SpringBootTest
@ContextConfiguration(classes = CommunityApplication.class)
public class TransactionTests {
    @Autowired
    private AlphaService alphaService;
    @Test
    public void testSave1(){
        Object obj = alphaService.save1();
        System.out.println(obj);
    }
}

4.2 编程式事务

@Autowired
    private TransactionTemplate transactionTemplate;
    public Object save2(){
        //设置隔离级别
        transactionTemplate.setIsolationLevel(TransactionDefinition.ISOLATION_READ_COMMITTED);
        //设置传播机制
        transactionTemplate.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
        
        //执行SQL访问事务
        return transactionTemplate.execute(new TransactionCallback<Object>() {
            @Override
            public Object doInTransaction(TransactionStatus transactionStatus) {
                //新增用户
                User user = new User();
                user.setUsername("beta");
                user.setSalt(CommunityUtil.generateUUID().substring(0,5));
                user.setPassword(CommunityUtil.md5("123"+user.getSalt()));
                user.setEmail("beta@qq.com");
                user.setHeaderUrl("http://image.nowcoder.com/head/999.png");
                user.setCreateTime(new Date());

                userMapper.insertUser(user);

                //新增帖子
                DiscussPost post = new DiscussPost();
                post.setUserId(user.getId());
                post.setTitle("nihao");
                post.setContent("11");
                post.setCreateTime(new Date());
                discussPostMapper.insertDiscussPost(post);

                Integer.valueOf("abc");

                return "ok";
            }
        });
    }
@Test
    public void testSave2(){
        Object obj = alphaService.save2();
        System.out.println(obj);
    }

5 显示评论

在这里插入图片描述

5.1 数据层

0.先对数据库有个了解:
在这里插入图片描述
entity_type:评论的对象可以是帖子、帖子里的评论、课程、题,等等
target_id:记录评论指向的人(不太懂)
status:0表示正常,1表示被删除

1.新建实体类,属性与表一致、get/set方法、toString

2.新建Mapper接口,注解@Mapper,声明查询方法

3.写mapper对应的xml,实现查询方法
知识点:
(1)namespace=对应的mapper地址
在这里插入图片描述
(2)id对应mapper里的方法名,resultType对应返回类型
在这里插入图片描述

5.2 业务层

1.注解@Service
2.注入Mapper

5.3 表现层

page分页有点问题,不能直接点数字,只能点其他按钮

6 添加评论

在这里插入图片描述
要用到事务了

这里的分页也有点问题:评论后在最后页,刷新后重定向在第一页。在最后页回复评论,刷新后重定向又在第一页

错误:不知道哪里代码错了,评论后”评论数“并没有增加,把老师代码全部粘过来,刷新也不会增加,去数据库改了之后,再评论就会增加”评论数“。所以有关数据库的错误,保证代码没错之后要在数据库把相应的数据改过来。

7 私信列表

在这里插入图片描述
在这里插入图片描述
conversation_id:会话id,小id_大id
status:0-未读;1-已读;2-删除

表现层实现两个功能:私信列表、点进去之后的私信详情
私信列表:用户名、头像、私信条数、时间等(这些信息用Map包装),循环
分页可以复用首页的逻辑

aaa用户的密码是aaa

8 发送列表

在这里插入图片描述

9 统一异常处理

在这里插入图片描述
无论是哪个层次的异常,都会汇到表现层,要对表现层统一处理异常

Spring Boot自动给我们的处理:将error文件夹拖拽到templates目录下(文件夹名一定要叫error,error下的文件名一定要是错误状态:404,500等)

@ControllerAdvice(annotations = Controller.class)//该注解只扫描带有controller注解的Bean
public class ExceptionAdvice {
    private static final Logger logger = LoggerFactory.getLogger(ExceptionAdvice.class);
    
    @ExceptionHandler({Exception.class})//所有异常都用该方法处理
    public void handleException(Exception e, HttpServletRequest request, HttpServletResponse response) throws IOException {
        logger.error("服务器发生异常:" + e.getMessage());
        for(StackTraceElement element : e.getStackTrace()){
            logger.error(element.toString());
        }

        String xRequestedWith = request.getHeader("x-requested-with");//请求方式
        if("XMLHttpRequest".equals(xRequestedWith)){//异步请求
            response.setContentType("application/plain;charset=utf-8");
            PrintWriter writer = response.getWriter();
            writer.write(CommunityUtil.getJSONString(1,"服务器异常"));
        }else{//普通请求
            response.sendRedirect(request.getContextPath()+"/error");
        }
        
    }
}

10 统一记录日志

在这里插入图片描述
将日志写在Service里的话,万一要更改就很麻烦,如果将记录日志单独实现就会方便很多,AOP可以做到这一点
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
步骤:
1.加注解@Component
2.加注解@Aspect
3.声明切点

@Pointcut("execution(* com.nowcoder.community.service.*.*(..))")
    public void pointcut(){ }

4.写具体要执行的操作

    //在连接点之前做一些事情
    @Before("pointcut()")
    public void before(){
        System.out.println("before");
    }

示例:

@Component//声明为Bean
@Aspect//声明为Aspect组件
public class AlphaAspect {
    @Pointcut("execution(* com.nowcoder.community.service.*.*(..))")
    //切点,第一个*表示方法的返回值,后面两个*表示包名下的所有类的所有方法,(..)表示方法里所有的参数
    public void pointcut(){

    }
    //在连接点之前做一些事情
    @Before("pointcut()")
    public void before(){
        System.out.println("before");
    }

    //在连接点后面做一些事情
    @After("pointcut()")
    public void after(){
        System.out.println("after");
    }

    //在返回值以后做一些事情
    @AfterReturning("pointcut()")
    public void afterReturning(){
        System.out.println("afterReturning");
    }
    //在抛出异常以后做一些事情
    @AfterThrowing("pointcut()")
    public void afterThrowing(){
        System.out.println("afterThrowing");
    }
    //在前后
    @Around("pointcut()")
    public Object around(ProceedingJoinPoint joinPoint) throws Throwable{
        System.out.println("around-before");
        Object obj = joinPoint.proceed();
        System.out.println("around-after");
        return obj;
    }
}

清空控制台:
在这里插入图片描述

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
【资源说明】 1、基于Spring Boot社区论坛项目源码+数据库+项目说明.zip 2、该资源包括项目的全部源码,下载可以直接使用! 3、本项目适合作为计算机、数学、电子信息等专业的课程设计、期末大作业和毕设项目,作为参考资料学习借鉴。 4、本资源作为“参考资料”如果需要实现其他功能,需要能看懂代码,并且热爱钻研,自行调试。 基于Spring Boot社区论坛项目源码+数据库+项目说明.zip # WeCommunity > 微博、微信、微X......各种微,我们也跟上潮流叫微社区,英文名是 WeCommunity!是不是和微信(WeChat)很像? 微社区是一个用户交流平台,类似论坛,提供有:帖子管理、评论、关注、点赞、搜索、私信、数据统计等功能项目目前部署在阿里云1核2G云服务器上,所有的组件都是单机模式运行。 ### 依葫芦画瓢的项目架构图 ![项目架构设计图](https://i.loli.net/2020/07/01/umXATrRW2PCLhEI.png "项目架构设计图") ### 功能描述 1. 用户管理   用户可以注册、登陆、退出,修改头像,查看某个用户的主页,其包括某个用户的关注、粉丝、获得的点赞数、发布的帖子,用户自己能查看自己发表的评论。 2. 帖子管理   普通用户可以发布帖子、修改帖子,管理员可以删除帖子、恢复删除的帖子,版主可以将帖子置顶/取消置顶、加精华/取消加精华。 3. 关注   用户可以关注/取消关注某个用户,假如A关注了B,那么A的关注列表中就有了B,B的粉丝列表中就有了A。 4. 评论   用户可以对帖子进行评论,还可以对评论进行回复。 5. 点赞   用户可以对帖子、评论进行点赞。 6. 系统通知   某个用户评论、点赞了帖子,或者关注了某个用户,那么被评论、点赞、关注的用户会收到一条通知。目前系统中有3种通知:评论通知、点赞通知、被关注通知。 7. 数据统计   管理员可以查看网站指定日期范围的UV(独立访客)、DAU(日活跃用户)数据。 8. 私信   用户可以对网站内的其他用户发送私信,双方互发消息,只有他们自己能看到自己的私信。 9. 定时调度   每个帖子都有个权重分,影响帖子的展示排名,定时调度主要是定时更新帖子的权重。 10. 搜索   搜索系统使用Elasticsearch实现,支持对帖子标题、帖子内容的搜索。 ### 技术选型 - Spring Boot - SpringMVC - Spring - MyBatis3、通用mapper - Spring Security:安全框架 - Redis:缓存及数据存储 - Kafka:消息队列 - Elasticsearch-6.3.0:分布式搜索引擎 - Quartz:定时调度框架 - Nginx - Thymeleaf:模板引擎 - Caffeine:Java本地缓存库 - MySQL - 七牛云:第三方文件存储服务 ### 各个功能模块所对应的技术点 ![功能模块对应的技术点](https://i.loli.net/2020/07/01/u3DRnvrxfUNKhtc.jpg) ### 界面设计 ![主页](https://i.loli.net/2020/07/01/VbQYPd9wvWzxjy8.jpg "主页") ![帖子发布](https://i.loli.net/2020/07/01/ANeDU75GaMB36ZT.jpg "帖子发布") ![私信列表_私信详情](https://i.loli.net/2020/07/01/p6HQtoPlJXNGdwz.jpg "私信列表_私信详情") ![系统通知_](https://i.loli.net/2020/07/01/mbvtnlCgZyYWqLS.jpg "系统通知_") ### 文件说明 - wecommunity.sql:数据库文件 - wecommunity文件夹:Maven 项目源码 - static文件夹:前端静态资源,需独立部署 ### 本地开发运行部署 - 下载zip直接解压或安装git后执行克隆命令 `git clone https://github.com/AatroxC/WeCommunity.git` - 安装各组件并启动:Redis、ZooKee
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值