Python Comments and Docstrings
last modified April 2, 2025
Comments and docstrings are essential for documenting Python code. Comments explain implementation details, while docstrings describe usage and functionality. This guide covers all aspects from basic syntax to documentation generation. Proper documentation improves code maintainability and enables auto-generated documentation tools. Learn to write clear, effective comments and docstrings following Python best practices.
Single-Line Comments
Single-line comments start with the #
symbol and continue to the
end of the line. They are used for brief explanations and in-line notes. This
example demonstrates proper single-line comment usage. Effective comments
explain why, not what - the code itself shows what is being done.
# Calculate circle area (single-line comment before code) def circle_area(radius): return 3.14159 * radius ** 2 # Using approximate value of π # Bad example: Redundant comment x = x + 1 # Increment x by 1 # Good example: Explains non-obvious logic # Adjust for edge case when divisor is near zero if abs(divisor) < 1e-10: result = float('inf') # Additional example: Commented-out code (use sparingly) # old_value = calculate_old_way() # Deprecated 2024-01-15 new_value = calculate_new_way() # Additional example: End-of-line comments names = ["Alice", "Bob", "Charlie"] # List of user names
The first comment explains the purpose of the function, while the in-line comment notes the approximation of π. Avoid redundant comments that simply repeat what the code obviously does. Instead, document the reasoning behind non-obvious implementations.
Commented-out code should generally be removed (use version control instead), but when temporarily kept, include a deprecation note. End-of-line comments should be short and only used when truly helpful to understand that specific line.
Multi-Line Comments
Python doesn't have true multi-line comment syntax, but consecutive single-line comments or triple-quoted strings are used for longer explanations. This example shows both approaches. Multi-line explanations are useful for complex algorithms or module-level documentation.
""" This is technically a string, not a comment, but often used for module-level documentation. It can span multiple lines and is ignored when not assigned to a variable. """ # This is the preferred way for actual comments # that span multiple lines in Python. Each line # starts with a # and the comment is clearly # distinguished from docstrings. def complex_algorithm(data): # Phase 1: Data preprocessing # - Normalize input values # - Handle missing data # - Convert categorical variables # Phase 2: Core calculation # Uses the Smith-Waterman algorithm # with custom modifications for our use case pass # Additional example: Block comment before code ########################################### # Database Connection Handler # # Manages connection pooling and retries # ########################################### class DatabaseHandler: pass
The triple-quoted string at the top is actually a string literal, but when not assigned to a variable, it works like a comment. For actual multi-line comments, consecutive single-line comments are preferred as they're unambiguously comments.
The complex_algorithm
function shows how to structure detailed
explanations of multi-phase processes. Section headers (like the database
handler example) can help organize large code files when used judiciously.
Function Docstrings
Docstrings are string literals that appear as the first statement in modules, functions, classes, and methods. They follow PEP 257 conventions and describe the object's purpose and usage. This example demonstrates proper function docstring formatting. Good docstrings enable auto-generated documentation and help tools like help() and IDEs.
def calculate_interest(principal, rate, years): """Calculate compound interest. Args: principal (float): Initial investment amount rate (float): Annual interest rate (e.g., 0.05 for 5%) years (int): Investment period in years Returns: float: Final amount after compound interest Examples: >>> calculate_interest(1000, 0.05, 10) 1628.894626777442 """ return principal * (1 + rate) ** years # Additional example: One-line docstring def greet(name): """Return a greeting string for the given name.""" return f"Hello, {name}!" # Additional example: Multi-class function def process_data(data, verbose=False): """Process input data with optional verbosity. Performs data cleaning, normalization, and feature extraction in a single pipeline. Parameters: data (DataFrame): Input dataset verbose (bool): If True, print progress messages Raises: ValueError: If data contains invalid values Note: This function modifies the input DataFrame in place. """ # Implementation here pass
The calculate_interest
docstring follows the popular Google style,
with clear sections for arguments, returns, and examples. One-line docstrings
are appropriate for simple functions. Multi-class functions deserve more
detailed docstrings covering all parameters, return values, exceptions, and
special notes.
Docstrings should focus on what the function does (its interface), not how it
works (implementation). They become part of the function's __doc__
attribute and are accessible via help()
.
Class Docstrings
Class docstrings describe the class's purpose, attributes, and usage patterns. They appear immediately after the class definition and before any methods. This example shows comprehensive class documentation. Well-documented classes are easier to use and maintain.
class BankAccount: """A class representing a basic bank account. Attributes: account_number (str): Unique account identifier balance (float): Current account balance owner (str): Account holder's name """ def __init__(self, account_number, owner, balance=0.0): """Initialize a new bank account. Args: account_number (str): Account identifier owner (str): Account holder name balance (float, optional): Initial balance. Defaults to 0.0. """ self.account_number = account_number self.owner = owner self.balance = balance # Additional example: Detailed class with methods class Vector: """A mathematical vector in 2D space. Supports basic vector operations: addition, scalar multiplication, dot product, and magnitude calculation. Examples: >>> v1 = Vector(1, 2) >>> v2 = Vector(3, 4) >>> v1 + v2 Vector(4, 6) """ def __init__(self, x, y): self.x = x self.y = y def magnitude(self): """Calculate the vector's magnitude. Returns: float: The vector's length """ return (self.x**2 + self.y**2)**0.5
The BankAccount
docstring describes the class's purpose and lists
its attributes. The Vector
class shows more detailed
documentation including usage examples. Class docstrings should provide enough
information for someone to use the class without reading its implementation.
Method docstrings within classes follow the same principles as function
docstrings. The magnitude
method demonstrates a simple but clear
docstring. Together, these form comprehensive class documentation.
Module Docstrings
Module docstrings appear at the top of Python files and describe the module's purpose, contents, and usage. They should be triple-quoted strings before any imports or code. This example shows a properly documented module. Module docstrings help users understand the file's role in the larger system.
"""Financial calculations module. Provides functions for common financial calculations including interest, annuities, and investment returns. Example: >>> from financial import calculate_interest >>> calculate_interest(1000, 0.05, 10) 1628.894626777442 Copyright (c) 2025 Financial Tools Inc. License: MIT """ import math from datetime import date # Additional example: Detailed module docstring """Image processing utilities. This module contains functions for common image manipulation tasks: - Color space conversions - Filter applications - Feature detection - Image I/O operations The implementation uses NumPy for efficient array operations and supports both RGB and grayscale images. Note: For advanced computer vision tasks, consider using OpenCV instead of these basic functions. """
The financial module docstring describes its purpose, contents, and includes a quick example. Copyright and license information often appears in module docstrings. The image processing example shows more detailed documentation including implementation notes and recommendations.
Module docstrings should be comprehensive enough to let users understand what
the module offers without reading all its code. They become the module's
__doc__
attribute and are used by documentation generators.
Docstring Formats
Several docstring formats exist in Python, each with its own conventions. The most common are Google style, NumPy/SciPy style, and reStructuredText. This example compares these formats. Consistent docstring style improves codebase readability.
# Google Style def google_style(a, b): """Compute the sum of two numbers. Args: a (int or float): First operand b (int or float): Second operand Returns: int or float: Sum of a and b Examples: >>> google_style(2, 3) 5 """ return a + b # NumPy/SciPy Style def numpy_style(a, b): """Compute the sum of two numbers. Parameters ---------- a : int or float First operand b : int or float Second operand Returns ------- int or float Sum of a and b Examples -------- >>> numpy_style(2, 3) 5 """ return a + b # reStructuredText Style def rst_style(a, b): """Compute the sum of two numbers. :param a: First operand :type a: int or float :param b: Second operand :type b: int or float :return: Sum of a and b :rtype: int or float :example: >>> rst_style(2, 3) 5 """ return a + b
Google style is compact and readable, popular for general Python code. NumPy style uses section headers with underlines, common in scientific Python. reStructuredText is more verbose but works well with Sphinx documentation tools. All three formats are valid - the key is consistency within a project.
Each format includes the same basic information: parameters, return values, and examples. Choose based on your project's conventions and documentation tooling. Many IDEs can generate docstring stubs in your preferred format.
Documentation Generation
Python docstrings can be automatically converted to documentation using tools like Sphinx, pdoc, and mkdocs. This example shows docstrings formatted for Sphinx. Properly formatted docstrings enable professional documentation with minimal extra effort.
def sphinx_ready(a, b): """Compute the product of two numbers. This function multiplies two numbers with proper type checking. It serves as an example of Sphinx-compatible docstrings. :param a: First factor :type a: int or float :param b: Second factor :type b: int or float :return: Product of a and b :rtype: int or float :raises TypeError: If either argument isn't numeric .. note:: This function doesn't handle complex numbers. .. warning:: Floating-point multiplication may have precision issues. Example: .. code-block:: python result = sphinx_ready(3, 4) print(result) # 12 """ if not isinstance(a, (int, float)) or not isinstance(b, (int, float)): raise TypeError("Numeric arguments required") return a * b
Sphinx docstrings use reStructuredText syntax with special directives like
.. note::
and .. warning::
. The :param:
,
:type:
, :return:
, and :raises:
tags
create structured documentation. Code examples use .. code-block::
directives.
When processed by Sphinx, these docstrings generate nicely formatted HTML/PDF documentation with proper cross-references. Many open-source Python projects use this approach for their official documentation.
Best Practices
Write docstrings for all public modules, functions, classes, and methods. Keep comments and docstrings up-to-date when code changes. Use docstrings to document interface (what), comments to explain implementation (how). Follow PEP 8 and PEP 257 style guidelines. Choose a docstring format and stick with it consistently throughout your project.
Source References
Learn more from these resources: PEP 257 Docstring Conventions, Google Python Style Guide, and Sphinx Documentation.
Author
My name is Jan Bodnar, and I am a passionate programmer with extensive programming experience. I have been writing programming articles since 2007. To date, I have authored over 1,400 articles and 8 e-books. I possess more than ten years of experience in teaching programming.
List all Python tutorials.