Equivalents were produced with the Free Edition of C# to Python Converter.
The closest thing in Python to an abstract method is an empty method:
C# | Python |
---|---|
public abstract class AbstractClass { public abstract void abstractMethod(); } |
class AbstractClass: def abstractMethod(self): pass |
You can convert C# arrays to Python's 'list'.
Unsized Array
C# | Python |
---|---|
int[] myArray; | myArray = [] |
Sized Array
C# | Python |
---|---|
int[] myArray = new int[2]; | myArray = [0 for _ in range(2)] |
Access Array Element
C# | Python |
---|---|
x = myArray[0]; | x = myArray[0] # no change |
Jagged Array
C# | Python |
---|---|
float[][] myArray = new float[][] { new float[] {x, y}}; | myArray = [[x, y]] |
Rectangular Array
C# | Python |
---|---|
int[][] myArray = new int[][] { new int[] {10, 20, 30 ,40}, new int[] {50, 60, 70, 80, 90, 100}, new int[] {110, 120} }; |
myArray = [[10, 20, 30, 40], [50, 60, 70, 80, 90, 100], [110, 120]] |
C# requires statements to end with semi-colons and multi-statement blocks to be enclosed in braces, while Python only requires indentation to correspond to the logical block levels and block headers to end in colons. The current instance is only available within instance methods via the first parameter of the method, usually named 'self' by convention (static methods are distinguished by the @staticmethod decorator).
C# | Python |
---|---|
class FooClass { void instanceMethod() { if (fooCondition) { fooLogicOne(); fooLogicTwo(); } } } |
class FooClass: def instanceMethod(self): if fooCondition: fooLogicOne() fooLogicTwo() |
C# | Python |
---|---|
void casts() { x = (int)y; x = (float)y; x = (string)y; } |
def casts(self): x = int(y) x = float(y) x = str(y) |
Lists
C#'s System.Collections.Generic.List collection and Python's 'list' are very close equivalents.
C# | Python |
---|---|
void Lists() { List<int> myList = new List<int>(); myList.Add(1); myList.Insert(1, 2); int i = 1; myList[0] = i; i = myList[0]; } |
def arrayLists(self): myList = [] myList.append(1) myList.insert(1, 2) i = 1 myList[0] = i i = myList[0] |
Dictionaries
C#'s System.Collections.Generic.Dictionary collection and Python's 'dict' are very close equivalents.
C# | Python |
---|---|
void Dictionaries() { Dictionary<string, int> map = new Dictionary<string, int>(); string s = "test"; map[s] = 1; int i = map[s]; i = map.Count; bool b = map.Count == 0; map.Remove(s); } |
def hashMaps(self): map = {} s = "test" map[s] = 1 i = map[s] i = len(map) b = not map map.pop(s) |
Python has a single comment format, starting with the '#' symbol. Multiline comments can also be created using triple quoted multiline strings.
C# | Python |
---|---|
// single-line comment foo(); // end-of-line comment /* comment over multiple lines */ |
# single-line comment foo() # end-of-line comment # comment # over multiple lines # # using multiline strings to simulate multiline comments: """ comment over multiple lines """ |
Local Variable
C# | Python |
---|---|
int myVar = 2; | myVar = 2 # Python infers the type |
Inferred Types
Inferred typing is available in C#, but Python always infers the type:
C# | Python |
---|---|
var myVar = 2; | myVar = 2 |
Static Field
In Python, static fields are initialized at the class level:
C# | Python |
---|---|
class FooClass { public static int staticField = 7; } |
class FooClass: staticField = 7 |
Instance Field
In Python, instance fields are not listed at the class level and are qualified with the instance object, given by the first instance method parameter (usually named 'self' by convention):
C# | Python |
---|---|
class FooClass { public int instanceField = 2; } |
class FooClass: def __init__(self): self.instanceField = 2 |
Local Constant
C# | Python |
---|---|
const int myConst = 2; | MY_CONST = 2 # at method level — up to programmers to respect the naming convention and avoid changing it |
Class Constant
C# | Python |
---|---|
public const int myConst = 2; | MY_CONST = 2 # at class level — up to programmers to respect the naming convention and avoid changing it |
Python constructors are named __init__, but a serious shortcoming of Python is the restriction of allowing only one constructor per class.
C# | Python |
---|---|
class Foo { public Foo() { } } |
class Foo: def __init__(self): pass |
Python and C# both allow default or optional parameters.
C# | Python |
---|---|
public void defaultParam(int param = 0) { ... } |
def defaultParam(self, param=0): ... |
Since methods are first class objects in Python delegates are converted easily:
C# | Python |
---|---|
public class Test { public delegate void FooDelegate(int arg); public static void static_method(int p) { Console.WriteLine("static_method called with arg " + p); } public void instance_method(int p) { Console.WriteLine("instance_method called with arg " + p); } public void method() { FooDelegate f1 = new FooDelegate(static_method); f1(1); FooDelegate f2 = instance_method; f2(2); FooDelegate f3 = (int i) => instance_method(i); f3(3); } } |
class Test: class FooDelegate: def __init__(self, method, instance): self._method = method self._instance = instance def invoke(self, arg): if self._instance is None: self._method(arg) else: self._method(self._instance, arg) @staticmethod def static_method(p): print("static_method called with arg " + str(p)) def instance_method(self, p): print("instance_method called with arg " + str(p)) def method(self): f1 = Test.FooDelegate(Test.static_method, None) f1.invoke(1) f2 = Test.FooDelegate(Test.instance_method, self) f2.invoke(2) f3 = Test.FooDelegate(lambda i : self.instance_method(i), None) f3.invoke(3) |
Enums in C# have equivalents in Python using the 'enum' module:
C# | Python |
---|---|
public enum Simple { Foo, Bar } |
from enum import Enum class Simple(Enum): FOO = 0 BAR = 1 |
......
C# | Python |
---|---|
// value equality with primitives: int i = 1; int j = 1; bool valuesIdentical = i == j; // identity equality with class types: Foo f = new Foo(); Foo g = f; bool objectsIdentical = f == g; // strings: string s1 = "abc"; string s2 = s1; valuesIdentical = s1 == s2; // caution: C# string internment often // makes this true whenever '==' is true: objectsIdentical = string.ReferenceEquals(s1, s2); |
# value equality: i = 1 j = 1 valuesIdentical = i == j # identity equality f = Foo() g = f objectsIdentical = f is g # strings: s1 = "abc" s2 = s1 valuesIdentical = s1 == s2 # caution: Python string internment often # makes this true whenever '==' is true: objectsIdentical = s1 is s2 |
The Python try/except/finally scheme is equivalent to C#'s try/catch/finally:
C# | Python |
---|---|
void exceptionHandling() { try { } catch (Foo f) { throw f; } catch (Bar) { throw new Bar(); } finally { } } |
def exceptionHandling(self): try: pass except Foo as f: raise f except Bar: raise Bar() finally: pass |
Python doesn't have extension methods, so a C# extension method is just converted to an ordinary Python static method (calls to the method have to be adjusted to static calls using the class name).
C# | Python |
---|---|
public static class ContainsExtension { public static void ExtensionMethod(this string myParam) { // ... } } class TestClass { void TestMethod() { string s = "abc"; s.ExtensionMethod(); } } |
class ContainsExtension: @staticmethod def ExtensionMethod(myParam): # ... pass class TestClass: def _TestMethod(self): s = "abc" ContainsExtension.ExtensionMethod(s) |
C# | Python |
---|---|
if (conditionA) { } else if (conditionB) { } else { } |
if conditionA: pass elif conditionB: pass else: pass |
C# allows incrementing and decrementing integers within expressions, while Python doesn't. The following shows how C# to Python Converter handles some cases.
C# | Python |
---|---|
public void incrementDecrement() { i = ++j; i = j++; i = j--; i = s + --t; i = s + t--; i = s + ++t; i = s + t++; } |
def incrementDecrement(self): j += 1 i = j i = j j += 1 i = j j -= 1 t -= 1 i = s + t i = s + t t -= 1 t += 1 i = s + t i = s + t t += 1 |
Python does not have indexers, so you must use get/set methods instead:
C# | Python |
---|---|
public int this[int index] { get { return field[index]; } set { field[index] = value; } } |
def get(self, index): return field[index] def set(self, index, value): field[index] = value |
C# | Python |
---|---|
public class one { protected int baseField = 1; public one(int i) { } public virtual void baseMethod() { } } public class two : one { public two() : base(0) { } public void method() { base.baseField = 2; base.baseMethod(); } } |
class one: def __init__(self, i): self.baseField = 1 def baseMethod(self): pass class two(one): def __init__(self): super().__init__(0) def method(self): self.baseField = 2 super().baseMethod() |
Defining Interfaces
Interfaces in Python are just classes with empty methods:
C# | Python |
---|---|
public interface IFooInterface { void method(); } |
class IFooInterface: def method(self): pass |
Implementing Interfaces
C# | Python |
---|---|
public class Foo : IFooInterface { public void method() { ... some code } } |
class Foo(IFooInterface): def method(self): ... some code |
Expression Lambda
C# | Python |
---|---|
myVar = (String text) => text.Length; | myVar = lambda text : len(text) |
Multi-statement Lambda
C# | Python |
---|---|
myVar = (Foo param1, Bar param2) => { // ...multiple statements } |
No direct Python equivalent |
C# | Python |
---|---|
x = y && z; x = y || z; x = !y; i = j | k; i = j & k; i = j ^ k; |
x = y and z x = y or z x = not y i = j | k i = j & k i = j ^ k |
C# | Python |
---|---|
// while loop: while (while_condition) { foo(); // break and continue: if (b) break; else continue; } // do-while loop: do { foo(); } while (do_condition); // traditional for loop: for (int i = 0; i < 10; i++) { foo(); } // 'for each' loop: foreach (Foo f in FooList) { } |
# while loop: while while_condition: foo() # break and continue: if b: break else: continue # do-while loop: # Python doesn't have a do-while loop, so simulate it: condition = True while condition: foo() condition = do_condition # traditional for loop: for i in range(0, 10): foo() # 'for each' loop: for f in FooList: pass |
C# | Python |
---|---|
void exponentiation() { x = Math.Pow(y, z); } void integerDivision() { // C# integer division always rounds towards 0: int i = -5; int j = 2; int result = i / j; // result is -2 } void modulusOperator() { // C# and Python '%' operators are only equivalent for positive numbers: int i = 2; int j = -3; int result = i % j; // result is 2 } void otherMathOperations() { x = Math.Abs(y); // see the System.Math class for many more functions: x = Math.Cos(y); x = Math.E; x = Math.PI; } |
import math def exponentiation(self): x = y ** z def integerDivision(self): i = -5 j = 2 result = math.trunc(i / float(j)) # result is -2 # Python integer division rounds away from 0 when negative: result = i / j # result is -3 def modulusOperator(self): i = 2 j = -3 result = math.fmod(i, j) # result is 2 # Python modulus operator produces a different result: result = i % j # result is -1 def otherMathOperations(self): x = abs(y) # 'abs' is a built-in Python function # see the Python 'math' module for many more functions: x = math.cos(y) x = math.e x = math.pi |
Python instance methods have a first parameter indicating the instance ('self' is the convention for this parameter and not a keyword), and static methods have the '@staticmethod' decorator.
C# | Python |
---|---|
public void instanceMethod() { } public static void staticMethod() { } |
def instanceMethod(self): pass @staticmethod def staticMethod(): pass |
C# | Python |
---|---|
public class SomeType { private int IntValue = 1; public static int operator +(SomeType X, SomeType Y) { return X.IntValue + Y.IntValue; } public void OperatorTest() { SomeType o = new SomeType(); SomeType p = new SomeType(); i = o + p; } } |
class SomeType: def __init__(self): # instance fields found by C# to Python Converter: self._IntValue = 1 @staticmethod def op_add(X, Y): return X._IntValue + Y._IntValue def OperatorTest(self): o = SomeType() p = SomeType() i = SomeType.op_add(o, p) |
C# | Python |
---|---|
void method(params string[] args) { foreach (string x in args) { ... logic for each item in args } } |
def method(self, *args): for x in args: ... logic for each item in args |
Python does not have properties, so you must use get/set methods instead:
C# | Python |
---|---|
public int IntProperty { get { return intField; } set { intField = value; } } |
def get_int_property(self): return intField def set_int_property(self, value): intField = value |
Python does not support C#-style 'ref' parameters. All Python parameters are passed by value (if it's a reference, then the reference is passed by value). However, you can wrap the parameter type in another type (we call it 'RefObject').
Here's a simple example:
C# | Python |
---|---|
public void refParamMethod(ref int i) { i = 1; } public void callRefParamMethod() { int i = 0; refParamMethod(ref i); } |
def refParamMethod(self, i): i.arg_value = 1 def callRefParamMethod(self): i = 0 temp_ref_i = RefObject(i) self.refParamMethod(temp_ref_i) i = temp_ref_i.arg_value # ---------------------------------------------------------------------------------------- # Copyright © 2022 - 2024 Tangible Software Solutions Inc. # This class can be used by anyone provided that the copyright notice remains intact. # # This class is used to replicate the ability to have 'ref' parameters in Python. # ---------------------------------------------------------------------------------------- class RefObject: def __init__(self, ref_arg): self.arg_value = ref_arg |
C# static constructors and Python code at the class level serve the same purpose.
C# | Python |
---|---|
class Foo { public static int field; static Foo() { ... logic to set 'field' value } } |
class Foo: field = 0 ... logic to set 'field' value |
C# | Python |
---|---|
String s = initValue; int i = s.IndexOf(y); i = s.LastIndexOf(y); i = s.Length; bool b = s.Contains(y); s = s.Substring(i); s = s.Substring(i, j); b = s.EndsWith(y); b = s.StartsWith(y); s = s.ToLower(); s = s.ToUpper(); s = s.TrimStart(); s = s.TrimEnd(); |
s = initValue i = s.find(y) i = s.rfind(y) i = len(s) b = y in s s = s[i:] s = s[i:i + j] b = s.endswith(y) b = s.startswith(y) s = s.casefold() s = s.upper() s = s.lstrip() s = s.rstrip() |
Python 3.10 has 'match' syntax, but if/elif/else is used prior to Python 3.10.
C# | Python |
---|---|
switch (pivot) { case 1: foo(); break; case 2: case 3: bar(); break; case 4: break; default: ack(); break; } |
Python 3.10: match pivot: case 1: foo() case 2 | 3: bar() case 4: pass case other: ack() Prior to Python 3.10: if pivot == 1: foo() elif pivot == 2 or pivot == 3: bar() elif pivot == 4: pass else: ack() |
C# | Python |
---|---|
result = condition ? truePart : falsePart; | result = truePart if condition else falsePart |
C# | Python |
---|---|
// checking if 'f' is an instance of type 'Foo': bool b = f is Foo; |
# checking if 'f' is an instance of type 'Foo': b = isinstance(f, Foo) |
C# | Python |
---|---|
using FooNamespace; // type import with alias matching type name: using FooClass = FooNamespace.FooClass; // namespace alias: using fooalias = FooNamespace; // type import: using static FooNamespace.BarClass; |
from FooNamespace import * from FooNamespace import FooClass * no equivalent to type or namespace aliases * * no equivalent to C# 'using static' * |
C# | Python |
---|---|
using (FooType foo = new FooType()) { ... } |
with FooType() as foo: ... |
C# | Python |
---|---|
// verbatim string: string s = @"multiline verbatim string"; // C#11 raw string literal: string s = """ multiline verbatim string """; |
s = """multiline verbatim string""" |
Copyright © 2004 – 2025 Tangible Software Solutions Inc.