๐ Diamond Valuation Prediction
๐บ Watch on YouTube
This repository contains a Python and Jupyter Notebook project developed for the AI Project Showcase Competition 2024, organized by Ready Tensor AI.
The project focuses on analyzing a dataset of diamond characteristics to predict their prices using machine learning techniques, including linear regression and K-Nearest Neighbors (KNN).
For more information and access to the project, please visit the GitHub repository.
This project explores the fascinating world of diamonds and aims to predict their price based on a variety of factors. Our goal is to uncover hidden relationships between diamond characteristics and their value, contributing to a deeper understanding of the diamond market.
The purpose of this predictive analysis is to create a website that determines the price of a diamond based on its characteristics: carat, cut, color, clarity, price, depth, table, x (length), y (width), and z (depth). In extreme cases where a quick estimate is required, it is not feasible to define all of these characteristics. Therefore, a study is necessary to determine the minimum characteristics needed to estimate the price accurately.
For the database study, we will use various statistical strategies, including linear regression, and apply chemistry knowledge to formulate mathematical equations to define diamond prices based on their characteristics. Additionally, to clean the database, which contains missing values, and to predict the value of diamonds based on their characteristics, we will employ the KNN (K-Nearest Neighbors) clustering algorithm. This is a supervised learning algorithm that will be used for both cleaning and predictions. To estimate missing values in the database, the KNN algorithm will individually calculate the distance between diamonds with missing values and those with known values, based on the known characteristics of the diamonds. Then, the KNN will identify the diamonds closest to the one being analyzed and use this information to predict the missing value. The same process will be applied to predict the price of diamonds.
Column Name | Description |
---|---|
carat | Weight of the diamond in carats |
cut | Quality of the diamond's cut (Ideal, Premium, Very Good, Good, Fair) |
color | Color of the diamond (D, E, F, G, H, I, J) |
clarity | Clarity of the diamond (IF, VVS1, VVS2, VS1, VS2, SI1, SI2, I1) |
depth | Percentage of the diamond's depth |
table | Percentage of the diamond's table width |
price | Price of the diamond in US dollars |
x | Length of the diamond in millimeters |
y | Width of the diamond in millimeters |
z | Depth of the diamond in millimeters |
import pandas as pd import seaborn as sns import matplotlib.pyplot as plt import math import numpy as np from sklearn.impute import KNNImputer from sklearn.preprocessing import OrdinalEncoder import random # Mudar o caminho da base de dados path = r"DataBases\Diamonds_values_faltantes.csv" diamonds = pd.read_csv(fr"{path}") diamonds
๐ Below is the number of missing values per column
counter = {} for x in range(diamonds.shape[1]): column_name = diamonds.columns[x] counter[column_name] = diamonds.shape[0] - len(diamonds[column_name].dropna()) counter_df = pd.DataFrame(list(counter.items()), columns=['Coluna', 'Quantidade de NaN']) counter_df plt.figure(figsize = (8, 6)) sns.heatmap((diamonds[["carat", "depth", "table", "price", "x", "y", "z"]]).corr(), vmin = -1, vmax = 1, annot = True, cmap = 'magma') plt.title("Coeficiente de Correlaรงรฃo Linear") plt.show()
Using Four Elements to Estimate the Carat of the Diamond:
For the second method of estimating the diamond's carat, four elements are required: Length (mm), Width (mm), Depth (mm), and Density
However, we don't have the diamond's volume. To obtain it, we'll break down the volume calculation of an object as follows:
Now, we need to find the diamond's carat. To do this, we'll use Formula 1 to estimate the diamond's carat:
The general formula becomes:
OR
We can conclude that the price does not have a good correlation with the total percentage of the diamond (depth) and also does not have a high correlation with the table, with an inversely proportional correlation of -0.0086 with depth, and a proportional relationship of 0.13 with the table.
We can also conclude that the price has a good linear correlation with the carat of 0.92, x (length) of 0.89, y (width) of 0.89, and z (depth) of 0.88.
Based on this heat map analysis, we can conclude that the larger the carat, x (length), y (width), and z (depth), the higher the diamond's price can be.
However, there may be some cases where a diamond has a very high carat but a low price, just as there may be diamonds with a low carat but a high price. This can also happen with x (length), y (width), and z (depth). Because of this, we question the following: how well can the carat, x (length), y (width), and z (depth) determine the value of the diamond? To answer this, we need to derive the Coefficient of Determination.
Copy code plt.figure(figsize=(8, 6)) sns.heatmap((diamonds[["carat", "depth", "table", "price", "x", "y", "z"]]).corr()**2, vmin=-1, vmax=1, annot=True, cmap='magma') plt.title("Coefficient of Determination") plt.show()
When analyzing the heat map above, we can see that we can define the price of the diamond more reliably using the numerical variable carat, with 85% reliability. This means that although we can say that the higher the carat of the diamond, the higher its price, unfortunately, this rule is only valid for 85% of the data.
For x (length), y (width), and z (depth), this reliability is only 79% for length and width and 78% for depth, which is not a strong determination. Therefore, they may be disregarded if the categorical variables can accurately define the price of the diamond.
Below we are performing the process of separating the diamonds database so that the machine learning process is more effective.
- Cut has 5 classification types: Ideal, Premium, Good, Very Good, and Fair
- Color has 7 classification types: E, I, J, H, F, G, and D
- Clarity has 8 classification types: SI2, SI1, VS1, VS2, VVS2, VVS1, I1, and IF
Setting length, width, and/or depth measurements of a diamond equal to 0 as NaN
Copy code for x in range(diamonds.shape[0]): for y in range(7, diamonds.shape[1]): if diamonds.iloc[x, y] == 0: diamonds.iloc[x, y] = np.nan elif diamonds.iloc[x, y] >= 30: diamonds.iloc[x, y] = np.nan diamonds
ps: Some books advise using the formula (K = log n) where n is the number of rows in the database.
To thus define the amount of K.
Copy code classification = KNNImputer(n_neighbors=round(math.log(diamonds.shape[0]))) diamonds[["carat", "depth", "table", "price", "x", "y", "z"]] = classification.fit_transform(diamonds[["carat", "depth", "table", "price", "x", "y", "z"]]) diamonds
Copy code '''KNN for categorical values''' encoder = OrdinalEncoder() diamonds_encoder = encoder.fit_transform(diamonds) knn_imputer = KNNImputer(n_neighbors = round(math.log(diamonds.shape[0]))) diamonds_imputer = knn_imputer.fit_transform(diamonds_encoder) diamonds_imputer = pd.DataFrame(diamonds_imputer, columns = diamonds.columns) diamonds_imputer = encoder.inverse_transform(diamonds_imputer)
Copy code for x in range(diamonds.shape[0]): for y in range(1, 4): if pd.isna(diamonds.iloc[x, y]): diamonds.iloc[x, y] = diamonds_imputer[x][y] diamonds
standardization of numerical columns diamonds[["carat", "x", "y", "z"]] = round(diamonds[["carat", "x", "y", "z"]], 2) diamonds[["table", "price"]] = round(diamonds[["table", "price"]]) diamonds["depth"] = round(diamonds["depth"], 1) diamonds
path = r"DataBases\Diamonds_clean.csv" try: pd.read_csv(f"{path}") print(f"This dataframe already exists in the directory: {path}") except FileNotFoundError: diamonds.to_csv(fr"{path}", index=False) print(f'''Cleaned database added to directory: {path} successfully!!''')
1- Carat is equivalent to 200mg
2- Points are equivalent to 0.01 carats
plt.figure(figsize=(17, 10)) plt.subplot(2, 1, 1) sns.scatterplot(data=diamonds, x="x", y="price") plt.xlabel("Length (mm)") plt.ylabel("Price") plt.gca().spines["right"].set_visible(False) plt.gca().spines["top"].set_visible(False) plt.gca().spines["left"].set_visible(False) plt.grid(axis="y", alpha=0.5) plt.subplot(2, 1, 2) sns.scatterplot(data=diamonds, x="x", y="carat") plt.xlabel("Length (mm)") plt.ylabel("Carat") plt.gca().spines["right"].set_visible(False) plt.gca().spines["top"].set_visible(False) plt.gca().spines["left"].set_visible(False) plt.grid(axis="y", alpha=0.5) plt.show()
plt.figure(figsize=(17, 10)) plt.subplot(2, 1, 1) sns.scatterplot(diamonds, x = "y", y = "price") plt.xlabel("Width (mm)") plt.ylabel("Price") plt.gca().spines["right"].set_visible(False) plt.gca().spines["top"].set_visible(False) plt.gca().spines["left"].set_visible(False) plt.grid(axis = "y", alpha = 0.5) plt.subplot(2, 1, 2) sns.scatterplot(diamonds, x = "y", y = "carat") plt.xlabel("Width (mm)") plt.ylabel("Carat") plt.gca().spines["right"].set_visible(False) plt.gca().spines["top"].set_visible(False) plt.gca().spines["left"].set_visible(False) plt.grid(axis = "y", alpha = 0.5) plt.show()
plt.figure(figsize=(17, 10)) plt.subplot(2, 1, 1) sns.scatterplot(diamonds, x = "z", y = "price") plt.xlabel("Depth (mm)") plt.ylabel("Price") plt.gca().spines["right"].set_visible(False) plt.gca().spines["top"].set_visible(False) plt.gca().spines["left"].set_visible(False) plt.grid(axis = "y", alpha = 0.5) plt.subplot(2, 1, 2) sns.scatterplot(diamonds, x = "z", y = "carat") plt.xlabel("Depth (mm)") plt.ylabel("Carat") plt.gca().spines["right"].set_visible(False) plt.gca().spines["top"].set_visible(False) plt.gca().spines["left"].set_visible(False) plt.grid(axis = "y", alpha = 0.5) plt.show()
plt.figure(figsize=(17, 5)) sns.scatterplot(diamonds, x = "carat", y = "price") plt.xlabel("Carat") plt.ylabel("Price") plt.title("Price and Carat Relationship") plt.gca().spines["right"].set_visible(False) plt.gca().spines["top"].set_visible(False) plt.gca().spines["left"].set_visible(False) plt.grid(axis = "y", alpha = 0.5) plt.show()
Carat: The strongest predictor of diamond price, with a determination coefficient of 85%.
Length, Width, Depth: Although correlated with the price, these features have a weaker relationship compared to the carat. These features are more useful for predicting the weight of the diamond (carat) than the price.
Cut, Color, Clarity: These features are not directly correlated with the price. However, analyzing their distribution in different price ranges reveals insights about how these factors influence the price range. For example, a higher percentage of diamonds with an "Ideal" cut can be found in higher price ranges.
The most reliable predictor of a diamond's price is its weight in carats.
Although length, width, and depth are correlated with the price, their relationship is weaker than the carat, suggesting that these dimensions are more useful for determining weight.
Categorical features such as cut, color, and clarity are not directly correlated with the price, but can provide a general indication of the price range.
A combination of numerical and categorical features can be used to build a more accurate price prediction model.
Explore more complex machine learning models (for example, random forests, support vector machines) to potentially improve prediction accuracy.
Analyze the distribution of categorical features in different price ranges to better understand their influence.
Consider incorporating other relevant features, such as diamond certification, origin, and specific gravity, to increase the predictive power of the model.
This data analysis project has successfully identified the key features that impact the price of a diamond and demonstrated the importance of feature engineering in building accurate prediction models. By understanding the relationships between the characteristics of diamonds and the price, this analysis can inform pricing strategies for diamond retailers and provide valuable insights for consumers.
Note: This report is based on the provided code snippet. More details about the prediction model and its performance are not available and would require additional information.
File Structure ๐
โโโ ๐บ๐ธ diamondValuationEnglish.ipynb
โโโ ๐บ๐ธ diamondValuationEnglish.py
โโโ ๐ง๐ท avaliacaoDiamante.inpyb
โโโ ๐ง๐ท avaliacaoDiamante
git clone https://github.com/Mindful-AI-Assistants/DiamondValuationPrediction.git
Install Required Packages:
pip install -r requirements.txt
Any contributions are highly appreciated. You can contribute in two ways:
Create an issue and tell us your idea ๐ก. Make sure that you use theย new ideaย label in this case;
Fork the project and submit a full requesto with your new idea. Before doing that, please make sure that you read and follow theย Contributions Guide.
Create a new branch:
git checkout -b feature/my-feature
Commit changes:
git commit -m "feat: Implemented new feature"
Push changes to remote:
git push origin feature/my-feature
Create a pull request
[Repo Link]
(https://github.com/Mindful-AI-Assistants/DiamondValuationPrediction)
Merge changes:
git merge feature/my-feature
Delete branch:
git branch -d feature/my-feature
๐ Tap here and teleport to the Streamlit Site
๐ QR Code of the Site on Streamlit
QR Code of the GitHub Repository
We are committed to fostering a welcoming and inclusive community for all team members. We expect everyone to adhere to the following principles:
Be respectful: Treat others with courtesy and respect, regardless of their background, identity, or opinions.
Be constructive: Focus on providing helpful feedback and constructive criticism.
Be open-minded: Be open to different perspectives and ideas.
Be open-minded: Be open to different perspectives and ideas.
Be accountable: Take responsibility for your actions and words.
Be inclusive: Promote a welcoming and inclusive environment for everyone.
If you witness any violation of this code of conduct, please contact [your contact information] so we can address the situation appropriately.
For more information, contact Mindful-AI-Assistants
Copyright 2025 Mindful AI Assistants. Code released under the MIT license.