日撸 Java 三百行day54-55

news2025/1/18 17:14:15

文章目录

  • 说明
  • day54 基于 M-distance 的推荐
    • 1. M-distance 理解
    • 2.代码理解
      • 1.代码中变量的解读
      • 2.leave-one-out测试
      • 3.计算MAE(平均绝对误差)
      • 4.计算RMSE(均方根误差)
  • day55 基于 M-distance 的推荐(续)
    • 1.基于用户和基于项目的推荐
    • 2.基于用户推荐代码思路
      • 2.1 抽象文本内容
      • 2.2 构造函数, 初始化MBR对象(借助jdk1.8新特性--Stream流)
      • 2.3 leave-one-out测试
      • 2.4 代码结果

说明

闵老师的文章链接: 日撸 Java 三百行(总述)_minfanphd的博客-CSDN博客
自己也把手敲的代码放在了github上维护:https://github.com/fulisha-ok/sampledata

day54 基于 M-distance 的推荐

1. M-distance 理解

例如day51-53中的KNN我们预测一个物品类别,我们是以测试样本和我们训练样本的距离的远近来找k个最相似似的邻居,对这k个邻居评分来预测测试样本的类别。而M-distance是根据平均分来计算两个用户 (或项目) 之间的距离。
如下图标这是以基于项目来预测得分的例子。例如预测用户u0对m2的评分,我们怎么来找邻居呢?我们是求出每个项目的平均分,找离m2平均分半径范围内的项目求平均分来预测得分。
在这里插入图片描述

2.代码理解

1.代码中变量的解读

文档中的内容有100000行记录,数据如下(部分):
在这里插入图片描述
一行代表的意思:0用户对项目0的评分,评分为5分;有943个用户,1682部电影,100000个评分;对部分变量说明

  • 总数统计(用户数量,项目数量,评分数)
private  int numItems; 
private int numUsers;
private int numRatings;

-在这里插入图片描述

  • compressedRatingMatrix(压缩的评分矩阵-实际上就是把文件内容读出来)
    就本次文本来看,共有100000条记录
 private int[][] compressedRatingMatrix;

在这里插入图片描述

  • userDegrees(每个用户评分的项目数量)
 private int[] userDegrees;

在这里插入图片描述

  • userStartingIndices(每个用户的起始索引,例如用户1的起始索引是272)
 private int[] userStartingIndices;

在这里插入图片描述- userAverageRatings(每个用户评价项目的一个平均分)

private double[] userAverageRatings;

在这里插入图片描述

  • itemDegrees (每个项目被评分的次数-也可以理解为有多少用户评分了)
private int[] itemDegrees;

在这里插入图片描述

  • itemAverageRatings (每个项目的平均分)
 private double[] itemAverageRatings;

在这里插入图片描述
MBR构造函数即是对上面的变量进行赋值,初始化。

2.leave-one-out测试

之前在knn中已经接触过了,即将数据集中的每一个样本都被单独作为测试集,而剩下的样本就作为训练集.
以一个测试集来举例。例如我们将第一行的数据作为测试集(0,0,5)我们知道这是用户0对项目0评分为5分,我们现在结合其他项目来预测项目0的评分。接下来的步骤为:

  • 先移除这个用户0对项目0的评分,重新计算对项目0的平均分
    在这里插入图片描述
  • 找邻居(这里是基于项目进行预测,去找用户评论过的项目的平均分与当前项目的平均分差值在一个半径范围内,则作为邻居,并累计他的评分)。如我们知道用户0评论了272部电影,排除项目0,我们要从271部电影中去找邻居来预测项目0的评分。
  • 若找到了邻居则求他们的平均分。如用户0找到94个邻居,总分387分,那我们预测用户0对项目0的预测分数为:4.117021276595745
    在这里插入图片描述
  • 完整代码:
    public void leaveOneOutPrediction() {
        double tempItemAverageRating;
        // Make each line of the code shorter.
        int tempUser, tempItem, tempRating;
        System.out.println("\r\nLeaveOneOutPrediction for radius " + radius);

        numNonNeighbors = 0;
        for (int i = 0; i < numRatings; i++) {
            tempUser = compressedRatingMatrix[i][0];
            tempItem = compressedRatingMatrix[i][1];
            tempRating = compressedRatingMatrix[i][2];

            // Step 1. Recompute average rating of the current item.
            tempItemAverageRating = (itemAverageRatings[tempItem] * itemDegrees[tempItem] - tempRating)
                    / (itemDegrees[tempItem] - 1);

            // Step 2. Recompute neighbors, at the same time obtain the ratings
            // Of neighbors.
            int tempNeighbors = 0;
            double tempTotal = 0;
            int tempComparedItem;
            for (int j = userStartingIndices[tempUser]; j < userStartingIndices[tempUser + 1]; j++) {
                tempComparedItem = compressedRatingMatrix[j][1];
                if (tempItem == tempComparedItem) {
                    continue;// Ignore itself.
                } // Of if

                if (Math.abs(tempItemAverageRating - itemAverageRatings[tempComparedItem]) < radius) {
                    tempTotal += compressedRatingMatrix[j][2];
                    tempNeighbors++;
                }
            }
            // Step 3. Predict as the average value of neighbors.
            if (tempNeighbors > 0) {
                predictions[i] = tempTotal / tempNeighbors;
            } else {
                predictions[i] = DEFAULT_RATING;
                numNonNeighbors++;
            }
        }
    }

3.计算MAE(平均绝对误差)

预测值与实际值之间的平均绝对偏差程度(MAE 的值越小,表示预测结果与实际值的偏差越小,预测模型的准确性越高)

    public double computeMAE() throws Exception {
        double tempTotalError = 0;
        for (int i = 0; i < predictions.length; i++) {
            tempTotalError += Math.abs(predictions[i] - compressedRatingMatrix[i][2]);
        } 

        return tempTotalError / predictions.length;
    }

4.计算RMSE(均方根误差)

预测值与实际值之间的平方值偏差程度。RMSE 的值越小,表示预测结果与实际值的均方差越小,预测模型的准确性越高

 public double computeRSME() throws Exception {
        double tempTotalError = 0;
        for (int i = 0; i < predictions.length; i++) {
            tempTotalError += (predictions[i] - compressedRatingMatrix[i][2])
                    * (predictions[i] - compressedRatingMatrix[i][2]);
        } 
        
        double tempAverage = tempTotalError / predictions.length;
        return Math.sqrt(tempAverage);
    }

在这里插入图片描述

day55 基于 M-distance 的推荐(续)

1.基于用户和基于项目的推荐

  • day 54基于项目的推荐
    我们预测m2的评分是:计算每个项目(m0~m5)的一个平均分,找到邻居(m1,m3)我们求他的品平均就是我们对m2的一个预测分数。(在这个过程中我们是根据项目邻居去评估的得分)
    在这里插入图片描述
  • 基于用户的推荐
    我们预测m2的评分:计算每个用户评分的项目(u0~u4),找到邻居(u1),求他们的平均分即我们对m2的预测得分。
    在这里插入图片描述

2.基于用户推荐代码思路

我最开始也想到的是对compressedRatingMatrix重新赋值,使数组进用户与项目关系互换。但最后我还是想采用列表的方式来编码。我大致思路如下:

2.1 抽象文本内容

将文本内容涉及的三个指标抽象为一个对象Text.其中userNum代表用户的编号,itemNum代表项目的编号,score代表用户对项目的评分。

    class Text{
        private Integer userNum;
        private Integer itemNum;
        private Integer score;

        public Integer getUserNum() {
            return userNum;
        }

        public void setUserNum(Integer userNum) {
            this.userNum = userNum;
        }

        public Integer getItemNum() {
            return itemNum;
        }

        public void setItemNum(Integer itemNum) {
            this.itemNum = itemNum;
        }

        public Integer getScore() {
            return score;
        }

        public void setScore(Integer score) {
            this.score = score;
        }

        public Text(Integer userNum, Integer itemNum, Integer score) {
            this.userNum = userNum;
            this.itemNum = itemNum;
            this.score = score;
        }
    }

2.2 构造函数, 初始化MBR对象(借助jdk1.8新特性–Stream流)

stream流的知识可以百度使用(结合Lambda表达式),他可以对集合进行非常复杂的查找、过滤、筛选等操作。

我大致思路是将文本内容放在一个列表中:List textList,我对这个textList采用stream流进行分组,按电影编号分组Map<Integer, List > textGroupByItem和按用户编号分组Map<Integer, List > textGroupByUser。并对相应的数组进行赋值。

stream流的使用:

// 按电影编号分组
textGroupByItem = textList.stream().collect(Collectors.groupingBy(Text::getItemNum));
//按用户编号分组
textGroupByUser = textList.stream().collect(Collectors.groupingBy(Text::getUserNum));
//对列表中某一属性求和
tempUserTotalScore[i] = textsByUser.stream().mapToDouble(Text::getScore).sum();

代码:

    public MBR(String paraFileName, int paraNumUsers, int paraNumItems, int paraNumRatings, boolean basedUser) throws Exception {
        if (basedUser){
            //基于用户的计算
            //step1. initialize these arrays
            numItems = paraNumItems;
            numUsers = paraNumUsers;
            numRatings = paraNumRatings;

            userDegrees = new int[numUsers];
            userAverageRatings = new double[numUsers];
            itemDegrees = new int[numItems];
            itemAverageRatings = new double[numItems];
            predictions = new double[numRatings];

            System.out.println("Reading " + paraFileName);

            //step2. Read the data file
            File tempFile = new File(paraFileName);
            if (!tempFile.exists()) {
                System.out.println("File " + paraFileName + " does not exists.");
                System.exit(0);
            }
            BufferedReader tempBufReader = new BufferedReader(new FileReader(tempFile));
            String tempString;
            String[] tempStrArray;
            while ((tempString = tempBufReader.readLine()) != null) {
                // Each line has three values
                tempStrArray = tempString.split(",");
                //把数据读入到textList列表中
                Text text = new Text(Integer.parseInt(tempStrArray[0]), Integer.parseInt(tempStrArray[1]), Integer.parseInt(tempStrArray[2]));
                textList.add(text);
            }
            tempBufReader.close();

            //按电影号分组
            textGroupByItem = textList.stream().collect(Collectors.groupingBy(Text::getItemNum));
            textGroupByUser = textList.stream().collect(Collectors.groupingBy(Text::getUserNum));
            double[] tempUserTotalScore = new double[numUsers];
            double[] tempItemTotalScore = new double[numItems];
            for (int i = 0; i < numUsers; i++) {
                // 用户的总分
                List<Text> textsByUser = textGroupByUser.get(i);
                tempUserTotalScore[i] = textsByUser.stream().mapToDouble(Text::getScore).sum();
                userDegrees[i] = textsByUser.size();
                userAverageRatings[i] = tempUserTotalScore[i] / userDegrees[i];
            }

            for (int i = 0; i < numItems; i++) {
                try {
                    // 电影的总分
                    List<Text> textsByItem = textGroupByItem.get(i);
                    tempItemTotalScore[i] = textsByItem.stream().mapToDouble(Text::getScore).sum();
                    itemDegrees[i] = textsByItem.size();
                    itemAverageRatings[i] = tempItemTotalScore[i] / itemDegrees[i];
                } catch (Exception e) {
                    System.out.println(e.getMessage());
                }

            }
        }
    }

2.3 leave-one-out测试

leave-one-out测试中,以文本中第一条记录为例子(0,0,5)我们要预测对项目0的评分
(1)第一步:排除用户0对项目0的评分(用户0评论了272个项目),重新计算用户0的平均分(271个项目)
(2)第二步:我们看对项目0评分的用户个数(452个),依次遍历用户的平均分与我们重新计算的平均分之差是否在半径范围内,从而累计邻居个数以及他们的总分。
(3)第三步:预测用户0对项目0的得分

stream流的使用:

// 对列表过滤数据
textsByUser = textsByUser.stream().filter(e -> !e.getItemNum().equals(outItem)).collect(Collectors.toList());

代码:

    public void leaveOneOutPredictionByUser() {
        double tempItemAverageRating;
        // Make each line of the code shorter.
        int tempUser, tempItem, tempRating;
        System.out.println("\r\nLeaveOneOutPredictionUser for radius " + radius);

        numNonNeighbors = 0;
        for (int i = 0; i < numRatings; i++) {
            Text text = textList.get(i);
            tempUser = text.getUserNum();
            tempItem = text.getItemNum();
            tempRating = text.getScore();

            // Step 1. Recompute average rating of the current user.
            List<Text> textsByUser = textGroupByUser.get(tempUser);
            Integer outItem = tempItem;
            textsByUser = textsByUser.stream().filter(e -> !e.getItemNum().equals(outItem)).collect(Collectors.toList());
            tempItemAverageRating = textsByUser.stream().mapToDouble(Text::getScore).sum() / textsByUser.size();

            // Step 2. Recompute neighbors, at the same time obtain the ratings
            // Of neighbors.
            int tempNeighbors = 0;
            double tempTotal = 0;
            List<Text> texts = textGroupByItem.get(tempItem);
            for (int j = 0; j < texts.size(); j++) {
                Text userText = texts.get(j);
                if (tempUser == j) {
                    continue;// Ignore itself.
                }

                if (Math.abs(tempItemAverageRating - userAverageRatings[userText.getUserNum()]) < radius) {
                    tempTotal += userText.getScore();
                    tempNeighbors++;
                }
            }
            // Step 3. Predict as the average value of neighbors.
            if (tempNeighbors > 0) {
                predictions[i] = tempTotal / tempNeighbors;
            } else {
                predictions[i] = DEFAULT_RATING;
                numNonNeighbors++;
            }
        }

    }

2.4 代码结果

在这里插入图片描述

  • day54-55代码
package machinelearing.knn;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

public class MBR {
    /**
     * Default rating for 1-5 points
     */
    public static final double DEFAULT_RATING = 3.0;

    /**
     * the total number of users (参与评分的用户数量)
     */
    private int numUsers;


    /**
     * the total number of items (评分的物品数量)
     */
    private  int numItems;

    /**
     * the total number of ratings (no-zero values) (非零评分值的数量)
     */
    private int numRatings;


    /**
     * the predictions
     */
    private double[] predictions;

    /**
     * Compressed rating matrix. user-item-rating triples (压缩的评分矩阵,存储用户-物品-评分的三元组)
     */
    private int[][] compressedRatingMatrix;

    /**
     * The degree of users (how many item he has rated). (用户已评分的物品数量)
     */
    private int[] userDegrees;

    /**
     * The average rating of the current user. (当前用户的平均评分。存储每个用户的平均评分值)
     */
    private double[] userAverageRatings;

    /**
     * The degree of users (how many item he has rated). (物品被评分的次数)
     */
    private int[] itemDegrees;

    /**
     * The average rating of the current item. (当前物品的平均评分。存储每个物品的平均评分值)
     */
    private double[] itemAverageRatings;

    /**
     * The first user start from 0. Let the first user has x ratings, the second user will start from x. (用户起始索引。第一个用户的起始索引为0,第二个用户的起始索引为前一个用户评分的数量。用于定位用户的评分在compressedRatingMatrix中的位置。)
     */
    private int[] userStartingIndices;

    /**
     * Number of non-neighbor objects. (非邻居对象的数量。用于表示在某个半径内不属于邻居的对象的数量。)
     */
    private int numNonNeighbors;

    /**
     * The radius (delta) for determining the neighborhood. (: 确定邻域的半径(delta)。用于确定邻域内的对象,即在该半径范围内的对象被视为邻居。)
     */
    private double radius;

    List<Text> textList = new ArrayList<>();

    private Map<Integer, List<Text>> textGroupByItem = new HashMap<>();

    private Map<Integer, List<Text>> textGroupByUser= new HashMap<>();

    class Text{
        private Integer userNum;
        private Integer itemNum;
        private Integer score;

        public Integer getUserNum() {
            return userNum;
        }

        public void setUserNum(Integer userNum) {
            this.userNum = userNum;
        }

        public Integer getItemNum() {
            return itemNum;
        }

        public void setItemNum(Integer itemNum) {
            this.itemNum = itemNum;
        }

        public Integer getScore() {
            return score;
        }

        public void setScore(Integer score) {
            this.score = score;
        }

        public Text(Integer userNum, Integer itemNum, Integer score) {
            this.userNum = userNum;
            this.itemNum = itemNum;
            this.score = score;
        }
    }


    public MBR(String paraFileName, int paraNumUsers, int paraNumItems, int paraNumRatings) throws Exception{
        //step1. initialize these arrays
        numItems = paraNumItems;
        numUsers = paraNumUsers;
        numRatings = paraNumRatings;

        userDegrees = new int[numUsers];
        userStartingIndices = new int[numUsers + 1];
        userAverageRatings = new double[numUsers];
        itemDegrees = new int[numItems];
        compressedRatingMatrix = new int[numRatings][3];
        itemAverageRatings = new double[numItems];
        predictions = new double[numRatings];

        System.out.println("Reading " + paraFileName);

        //step2. Read the data file
        File tempFile = new File(paraFileName);
        if (!tempFile.exists()) {
            System.out.println("File " + paraFileName + " does not exists.");
            System.exit(0);
        }
        BufferedReader tempBufReader = new BufferedReader(new FileReader(tempFile));
        String tempString;
        String[] tempStrArray;
        int tempIndex = 0;
        userStartingIndices[0] = 0;
        userStartingIndices[numUsers] = numRatings;
        while ((tempString = tempBufReader.readLine()) != null) {
            // Each line has three values
            tempStrArray = tempString.split(",");
            compressedRatingMatrix[tempIndex][0] = Integer.parseInt(tempStrArray[0]);
            compressedRatingMatrix[tempIndex][1] = Integer.parseInt(tempStrArray[1]);
            compressedRatingMatrix[tempIndex][2] = Integer.parseInt(tempStrArray[2]);

            userDegrees[compressedRatingMatrix[tempIndex][0]]++;
            itemDegrees[compressedRatingMatrix[tempIndex][1]]++;

            if (tempIndex > 0) {
                // Starting to read the data of a new user.
                if (compressedRatingMatrix[tempIndex][0] != compressedRatingMatrix[tempIndex - 1][0]) {
                    userStartingIndices[compressedRatingMatrix[tempIndex][0]] = tempIndex;
                }
            }
            tempIndex++;
        }
        tempBufReader.close();

        double[] tempUserTotalScore = new double[numUsers];
        double[] tempItemTotalScore = new double[numItems];
        for (int i = 0; i < numRatings; i++) {
            tempUserTotalScore[compressedRatingMatrix[i][0]] += compressedRatingMatrix[i][2];
            tempItemTotalScore[compressedRatingMatrix[i][1]] += compressedRatingMatrix[i][2];
        }

        for (int i = 0; i < numUsers; i++) {
            userAverageRatings[i] = tempUserTotalScore[i] / userDegrees[i];
        }

        for (int i = 0; i < numItems; i++) {
            itemAverageRatings[i] = tempItemTotalScore[i] / itemDegrees[i];
        }

    }

    public MBR(String paraFileName, int paraNumUsers, int paraNumItems, int paraNumRatings, boolean basedUser) throws Exception {
        if (basedUser){
            //基于用户的计算
            //step1. initialize these arrays
            numItems = paraNumItems;
            numUsers = paraNumUsers;
            numRatings = paraNumRatings;

            userDegrees = new int[numUsers];
            userAverageRatings = new double[numUsers];
            itemDegrees = new int[numItems];
            itemAverageRatings = new double[numItems];
            predictions = new double[numRatings];

            System.out.println("Reading " + paraFileName);

            //step2. Read the data file
            File tempFile = new File(paraFileName);
            if (!tempFile.exists()) {
                System.out.println("File " + paraFileName + " does not exists.");
                System.exit(0);
            }
            BufferedReader tempBufReader = new BufferedReader(new FileReader(tempFile));
            String tempString;
            String[] tempStrArray;
            while ((tempString = tempBufReader.readLine()) != null) {
                // Each line has three values
                tempStrArray = tempString.split(",");
                //把数据读入到textList列表中
                Text text = new Text(Integer.parseInt(tempStrArray[0]), Integer.parseInt(tempStrArray[1]), Integer.parseInt(tempStrArray[2]));
                textList.add(text);
            }
            tempBufReader.close();

            //按电影号分组
            textGroupByItem = textList.stream().collect(Collectors.groupingBy(Text::getItemNum));
            textGroupByUser = textList.stream().collect(Collectors.groupingBy(Text::getUserNum));
            double[] tempUserTotalScore = new double[numUsers];
            double[] tempItemTotalScore = new double[numItems];
            for (int i = 0; i < numUsers; i++) {
                // 用户的总分
                List<Text> textsByUser = textGroupByUser.get(i);
                tempUserTotalScore[i] = textsByUser.stream().mapToDouble(Text::getScore).sum();
                userDegrees[i] = textsByUser.size();
                userAverageRatings[i] = tempUserTotalScore[i] / userDegrees[i];
            }

            for (int i = 0; i < numItems; i++) {
                try {
                    // 电影的总分
                    List<Text> textsByItem = textGroupByItem.get(i);
                    tempItemTotalScore[i] = textsByItem.stream().mapToDouble(Text::getScore).sum();
                    itemDegrees[i] = textsByItem.size();
                    itemAverageRatings[i] = tempItemTotalScore[i] / itemDegrees[i];
                } catch (Exception e) {
                    System.out.println(e.getMessage());
                }

            }
        }
    }

    public void setRadius(double paraRadius) {
        if (paraRadius > 0) {
            radius = paraRadius;
        } else {
            radius = 0.1;
        }
    }

    public void leaveOneOutPrediction() {
        double tempItemAverageRating;
        // Make each line of the code shorter.
        int tempUser, tempItem, tempRating;
       // System.out.println("\r\nLeaveOneOutPrediction for radius " + radius);

        numNonNeighbors = 0;
        for (int i = 0; i < numRatings; i++) {
            tempUser = compressedRatingMatrix[i][0];
            tempItem = compressedRatingMatrix[i][1];
            tempRating = compressedRatingMatrix[i][2];

            // Step 1. Recompute average rating of the current item.
            tempItemAverageRating = (itemAverageRatings[tempItem] * itemDegrees[tempItem] - tempRating)
                    / (itemDegrees[tempItem] - 1);

            // Step 2. Recompute neighbors, at the same time obtain the ratings
            // Of neighbors.
            int tempNeighbors = 0;
            double tempTotal = 0;
            int tempComparedItem;
            for (int j = userStartingIndices[tempUser]; j < userStartingIndices[tempUser + 1]; j++) {
                tempComparedItem = compressedRatingMatrix[j][1];
                if (tempItem == tempComparedItem) {
                    continue;// Ignore itself.
                } // Of if

                if (Math.abs(tempItemAverageRating - itemAverageRatings[tempComparedItem]) < radius) {
                    tempTotal += compressedRatingMatrix[j][2];
                    tempNeighbors++;
                }
            }
            // Step 3. Predict as the average value of neighbors.
            if (tempNeighbors > 0) {
                predictions[i] = tempTotal / tempNeighbors;
            } else {
                predictions[i] = DEFAULT_RATING;
                numNonNeighbors++;
            }
        }
    }


    public void leaveOneOutPredictionByUser() {
        double tempItemAverageRating;
        // Make each line of the code shorter.
        int tempUser, tempItem, tempRating;
        // System.out.println("\r\nLeaveOneOutPredictionUser for radius " + radius);

        numNonNeighbors = 0;
        for (int i = 0; i < numRatings; i++) {
            Text text = textList.get(i);
            tempUser = text.getUserNum();
            tempItem = text.getItemNum();
            tempRating = text.getScore();

            // Step 1. Recompute average rating of the current user.
            List<Text> textsByUser = textGroupByUser.get(tempUser);
            Integer outItem = tempItem;
            textsByUser = textsByUser.stream().filter(e -> !e.getItemNum().equals(outItem)).collect(Collectors.toList());
            tempItemAverageRating = textsByUser.stream().mapToDouble(Text::getScore).sum() / textsByUser.size();

            // Step 2. Recompute neighbors, at the same time obtain the ratings
            // Of neighbors.
            int tempNeighbors = 0;
            double tempTotal = 0;
            List<Text> texts = textGroupByItem.get(tempItem);
            for (int j = 0; j < texts.size(); j++) {
                Text userText = texts.get(j);
                if (tempUser == j) {
                    continue;// Ignore itself.
                }

                if (Math.abs(tempItemAverageRating - userAverageRatings[userText.getUserNum()]) < radius) {
                    tempTotal += userText.getScore();
                    tempNeighbors++;
                }
            }
            // Step 3. Predict as the average value of neighbors.
            if (tempNeighbors > 0) {
                predictions[i] = tempTotal / tempNeighbors;
            } else {
                predictions[i] = DEFAULT_RATING;
                numNonNeighbors++;
            }
        }

    }


    public double computeMAE() throws Exception {
        double tempTotalError = 0;
        for (int i = 0; i < predictions.length; i++) {
            tempTotalError += Math.abs(predictions[i] - compressedRatingMatrix[i][2]);
        } // Of for i

        return tempTotalError / predictions.length;
    }

    public double computeMAE_User() throws Exception {
        double tempTotalError = 0;
        for (int i = 0; i < predictions.length; i++) {
            tempTotalError += Math.abs(predictions[i] - textList.get(i).getScore());
        } // Of for i

        return tempTotalError / predictions.length;
    }


    public double computeRSME() throws Exception {
        double tempTotalError = 0;
        for (int i = 0; i < predictions.length; i++) {
            tempTotalError += (predictions[i] - compressedRatingMatrix[i][2])
                    * (predictions[i] - compressedRatingMatrix[i][2]);
        } // Of for i

        double tempAverage = tempTotalError / predictions.length;

        return Math.sqrt(tempAverage);
    }

    public double computeRSME_User() throws Exception {
        double tempTotalError = 0;
        for (int i = 0; i < predictions.length; i++) {
            tempTotalError += (predictions[i] - textList.get(i).getScore())
                    * (predictions[i] - textList.get(i).getScore());
        } // Of for i

        double tempAverage = tempTotalError / predictions.length;

        return Math.sqrt(tempAverage);
    }

    public static void main(String[] args) {
        try {
            MBR tempRecommender = new MBR("C:/Users/Desktop/sampledata/movielens-943u1682m.txt", 943, 1682, 100000);
            MBR tempRecommender1 = new MBR("C:/Users/Desktop/sampledata/movielens-943u1682m.txt", 943, 1682, 100000, true);
            for (double tempRadius = 0.2; tempRadius < 0.6; tempRadius += 0.1) {
                tempRecommender.setRadius(tempRadius);
                tempRecommender1.setRadius(tempRadius);

                tempRecommender.leaveOneOutPrediction();
                double tempMAE = tempRecommender.computeMAE();
                double tempRSME = tempRecommender.computeRSME();

                tempRecommender1.leaveOneOutPredictionByUser();
                double tempMAE1 = tempRecommender1.computeMAE_User();
                double tempRSME1 = tempRecommender1.computeRSME_User();

                System.out.println("Radius_item = " + tempRadius + ", MAE_item = " + tempMAE + ", RSME_item = " + tempRSME
                        + ", numNonNeighbors_item = " + tempRecommender.numNonNeighbors);

                System.out.println("Radius_user = " + tempRadius + ", MAE_user = " + tempMAE1 + ", RSME_user = " + tempRSME1
                        + ", numNonNeighbors_user = " + tempRecommender1.numNonNeighbors);
            }
        } catch (Exception ee) {
            System.out.println(ee);
        }
    }


}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/576204.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ASEMI代理长电可控硅BT134的工作原理,BT134的应用领域

编辑-Z 本文将对可控硅BT134的工作原理及应用领域进行详细阐述。首先&#xff0c;我们将介绍可控硅BT134的基本概念和工作原理&#xff1b;其次&#xff0c;我们将探讨可控硅BT134在电力电子领域的应用&#xff1b;接着&#xff0c;我们将分析可控硅BT134在家用电器中的应用&a…

一种使用Java的快速将Web中表格转换成Excel的方法

背景 随着后疫情时代的到来&#xff0c;在过去的2022年&#xff0c;全国的经济情况&#xff0c;想必是很多学者和研究对象都非常关心的事。而这些数据在国家统计局网站上都有相应的记录。通过分析这些数据&#xff0c;可以从某一个角度来验证和观察当下的经济情况。 全国共计有…

Less基础速学 —— 嵌套、变量、注释、

关于CSS的预处理有读者了解到的就有主流的这三种&#xff1a;less 、sass/sacc 和 stylus&#xff0c;那么在使用这些CSS的预处理器&#xff0c;有浏览器的用法和服务器端的方法&#xff0c;例如使用Node来通过NPM包管理工具下载安装less在服务端来跑&#xff0c;那么在浏览器上…

go渲染静态html引擎模版

写个小例子介绍一下渲染html引擎模板的使用&#xff0c;大致目录结果如下&#xff1a; templates目录下三个html内容如下: index.html <title>模板1</title><link rel"stylesheet" href"/css/index.css" /><h1 class"text&quo…

数据链路层:差错检测

1.数据链路层&#xff1a;差错检测 笔记来源&#xff1a; 湖科大教书匠&#xff1a;差错检测 声明&#xff1a;该学习笔记来自湖科大教书匠&#xff0c;笔记仅做学习参考 比特差错&#xff1a;比特在传输过程中1可能变为0&#xff0c;0可能变为1 误码率BER&#xff1a;传输错误…

数据链路层:以太网/ARP协议

本文将分享数据链路层的相关知识点&#xff1a;以太网、以太网帧格式、MAC地址、MTU和ARP协议 以太网 "以太网" 不是一种具体的网络, 而是一种技术标准; 既包含了数据链路层的内容, 也包含了一些物理层的内容。例如: 规定了网络拓扑结构, 访问控制方式, 传输速率等。…

Linux学习笔记 --- Linux用户和权限

一. 认知root用户 目标&#xff1a;1. 了解什么是root用户(超级管理员) 2. 掌握用户切换的相关命令 3. 掌握sudo命令 root用户&#xff08;超级管理员&#xff09; 无论是Windows、MacOS、Linux均采用多用户的管理模式进行权限管理。 在Linux系统中&#xff0c;拥有…

STM32 UDS Bootloader开发-需求篇

文章目录 前言内存分配UDS诊断协议需求CAN ID及时间参数诊断服务Bootloader诊断服务 APP诊断服务DID刷写流程预编程主编程后编程 总结 前言 最近断断续续的在做基于STM32F103的UDS Bootloader,没有项目驱动&#xff0c;只是自己感兴趣。目前基本已经可以实现功能了&#xff0c…

关于 OpenCV 图像处理工具包 imutils 简单认知

写在前面 博文内容涉及 基本的图像处理工具包 imutils 的简单介绍以及使用Demo理解不足小伙伴帮忙指正 对每个人而言&#xff0c;真正的职责只有一个&#xff1a;找到自我。然后在心中坚守其一生&#xff0c;全心全意&#xff0c;永不停息。所有其它的路都是不完整的&#xff0…

2023年真无线蓝牙耳机品牌有哪些推荐?无线蓝牙耳机选购指南

今天就来给大家测评一下2023年最受用户欢迎的蓝牙耳机&#xff0c;在不断地测试耳机&#xff0c;并挖掘好的耳机出来&#xff0c;不得不说&#xff0c;蓝牙和麦克风以及音频技术的驱动的迭代更新&#xff0c;性能确实惊叹不已。 对于刚接触无线耳机的小白来说&#xff0c;选购…

【服务器】支付宝SDK接口调试

​ 文章目录 1.测试环境2.本地配置3. 内网穿透3.1 下载安装cpolar内网穿透3.2 创建隧道 4. 测试公网访问5. 配置固定二级子域名5.1 保留一个二级子域名5.2 配置二级子域名 6. 使用固定二级子域名进行访问 转发自cpolar内网穿透的文章&#xff1a;Java支付宝沙箱环境支付&#…

低代码的“钱景”——专业的事交给专业的人来做

你需要知道的低代码 低代码通常是指一种可视化的开发方法&#xff0c;用较少的代码、较快的速度来交付应用程序&#xff0c;相似的概念还有“无代码”&#xff0c;也是一种开发方法&#xff0c;通常是面向非技术性员工&#xff0c;让业务人员也可以成为“技术人员”&#xff0…

首发Yolov8涨点神器:华为诺亚2023极简的神经网络模型 VanillaNet---VanillaBlock助力检测,实现暴力涨点

强烈推荐:博主多个数据集亲测有效,实现暴力涨点,可快速迁移到论文创新性十足,刚新鲜出炉的论文,华为诺亚共同提出!!! 1.VanillaNet 论文:https://arxiv.org/pdf/2305.12972.pdf 来自华为诺亚、悉尼大学的研究者们提出了一种极简的神经网络模型 VanillaNet,以极简主义…

表情包APP小程序制作开发功能有哪些?

表情包小程序在实际的开发过程中所具备的功能都是至关重要的&#xff0c;功能完善好操作才能更受用户青睐。表情包小程序制作开发功能包括但不仅限于以下几点&#xff1a; 1、热门表情包推荐。表情包制作小程序为用户推荐最热门的表情包&#xff0c;让用户可以紧跟当下流…

串口屏-迪文10寸T5串口屏简单上手

先看效果 调试所需软件 DGUS上位机 下载链接 调试所需硬件 SD卡一张 读卡器一个 USB转TTL一个 DGUS上位机调试 点击新建工程(会让你选尺寸) 分别生成0号字库和DWIN ICL文件 0号字库文件在你软件的安装目录可以找到 ICL文件输出到你能找到的位置 等会要用 添加背景图片 …

异常检测论文1

本文仅作为个人阅读文献&#xff0c;做笔记记录。 <> \usepackage[dvipsnames]{xcolor} 一、摘要部分&#xff1a; 我们发现&#xff0c;现有的数据集偏向于局部结构异常&#xff0c;如划痕、凹痕或污染。特别是&#xff0c;它们缺乏违反逻辑约束形式的异常&#xff0…

【redis】Stream、String 超详细介绍

文章目录 一、Stream1.1 写入数据XADD条目 ID 的格式 1.2 获取数据XRANGE 和 XREVRANGEXREAD 监听新条目非阻塞形式阻塞形式 1.3 消费者组XGROUP 创建消费者组XREADGROUP 通过消费者组消费XACK 确认消息消费者组示例 1.4 XPENDING 和 XCLAIM 认领 其他消费者 的待处理消息XPEND…

(2022,MaskedGAN)掩蔽的生成对抗网络是数据高效生成学习者

Masked Generative Adversarial Networks are Data-Efficient Generation Learners 公众号&#xff1a;EDPJ 目录 0. 摘要 1. 简介 2. 相关工作 3. 方法 3.1 任务定义 3.2 掩蔽的生成对抗网络 3.3 理论见解 4. 实验 4.1 在 CIFAR-10、CIFAR-100 和 ImageNet 上使用…

算法基础学习笔记——④前缀和\差分\双指针\位运算

✨博主&#xff1a;命运之光 ✨专栏&#xff1a;算法基础学习 目录 ✨前缀和 ✨一维前缀和 &#x1f353;一维前缀和模板&#xff1a; ✨二维前缀和 &#x1f353;二位前缀和模板&#xff1a; 前言&#xff1a;算法学习笔记记录日常分享&#xff0c;需要的看哈O(∩_∩)O&a…

Python二级编程:分词去重

一、原题 参考编程模板&#xff0c;完善代码&#xff0c;实现以下功能。‪‪‪‪‪‫‫‪‪‪‪‪‪‪‪‪‪‪‪‪‪‪‪‪‫‪‪‪‪‪‪‪‪‪‪‪‪‪‫ 利用 jieba 库实现中文分词。对分词后的列表进行去重处理&#xff0c;然后将分词结果中字符数大于等于 3 的词语&…