mirror of
https://github.com/adambard/learnxinyminutes-docs.git
synced 2025-01-15 13:45:59 +00:00
633 lines
22 KiB
Markdown
633 lines
22 KiB
Markdown
---
|
|
language: c#
|
|
contributors:
|
|
- ["Irfan Charania", "https://github.com/irfancharania"]
|
|
- ["Max Yankov", "https://github.com/golergka"]
|
|
translators:
|
|
- ["Olfran Jiménez", "https://twitter.com/neslux"]
|
|
filename: LearnCSharp-es.cs
|
|
lang: es-es
|
|
---
|
|
|
|
C# es un lenguaje orientado a objetos elegante y de tipado seguro que
|
|
permite a los desarrolladores construir una variedad de aplicaciones
|
|
seguras y robustas que se ejecutan en el Framework .NET.
|
|
|
|
[Lee más aquí.](http://msdn.microsoft.com/es-es/library/vstudio/z1zx9t92.aspx)
|
|
|
|
```c#
|
|
// Los comentarios de una sola línea comienzan con //
|
|
/*
|
|
Los comentarios de múltiples líneas son de esta manera
|
|
*/
|
|
/// <summary>
|
|
/// Este es un comentario de documentación XML
|
|
/// </summary>
|
|
|
|
// Especifica el espacio de nombres que estará usando la aplicación
|
|
using System;
|
|
using System.Collections.Generic;
|
|
|
|
|
|
// Define un ambito para organizar el código en "paquetes"
|
|
namespace Learning
|
|
{
|
|
// Cada archivo .cs debe contener al menos una clase con el mismo nombre que el archivo
|
|
// Se permite colocar cualquier nombre, pero no deberías por cuestiones de consistencia.
|
|
public class LearnCSharp
|
|
{
|
|
// Una aplicación de consola debe tener un método main como punto de entrada
|
|
public static void Main(string[] args)
|
|
{
|
|
// Usa Console.WriteLine para imprimir líneas
|
|
Console.WriteLine("Hello World");
|
|
Console.WriteLine(
|
|
"Integer: " + 10 +
|
|
" Double: " + 3.14 +
|
|
" Boolean: " + true);
|
|
|
|
// Para imprimir sin una nueva línea, usa Console.Write
|
|
Console.Write("Hello ");
|
|
Console.Write("World");
|
|
|
|
|
|
///////////////////////////////////////////////////
|
|
// Variables y Tipos
|
|
//
|
|
// Declara una variable usando <tipo> <nombre>
|
|
///////////////////////////////////////////////////
|
|
|
|
// Sbyte - Entero de 8 bits con signo
|
|
// (-128 <= sbyte <= 127)
|
|
sbyte fooSbyte = 100;
|
|
|
|
// Byte - Entero de 8 bits sin signo
|
|
// (0 <= byte <= 255)
|
|
byte fooByte = 100;
|
|
|
|
// Short - Entero de 16 bits con signo
|
|
// (-32,768 <= short <= 32,767)
|
|
short fooShort = 10000;
|
|
|
|
// Ushort - Entero de 16 bits sin signo
|
|
// (0 <= ushort <= 65,535)
|
|
ushort fooUshort = 10000;
|
|
|
|
// Integer - Entero de 32 bits con signo
|
|
// (-2,147,483,648 <= int <= 2,147,483,647)
|
|
int fooInt = 1;
|
|
|
|
// Uinteger - Entero de 32 bits sin signo
|
|
// (0 <= uint <= 4,294,967,295)
|
|
uint fooUint = 1;
|
|
|
|
// Long - Entero de 64 bits con signo
|
|
// (-9,223,372,036,854,775,808 <= long <= 9,223,372,036,854,775,807)
|
|
long fooLong = 100000L;
|
|
// L es usado para indicar que esta variable es de tipo long o ulong
|
|
// un valor sin este sufijo es tratado como int o uint dependiendo del tamaño.
|
|
|
|
// Ulong - Entero de 64 bits sin signo
|
|
// (0 <= ulong <= 18,446,744,073,709,551,615)
|
|
ulong fooUlong = 100000L;
|
|
|
|
// Float - Precisión simple de 32 bits. IEEE 754 Coma flotante
|
|
// Precisión: 7 dígitos
|
|
float fooFloat = 234.5f;
|
|
// f es usado para indicar que el valor de esta variable es de tipo float
|
|
// de otra manera sería tratado como si fuera de tipo double.
|
|
|
|
// Double - Doble precisión de 32 bits. IEEE 754 Coma flotante
|
|
// Precisión: 15-16 dígitos
|
|
double fooDouble = 123.4;
|
|
|
|
// Bool - true & false (verdadero y falso)
|
|
bool fooBoolean = true;
|
|
bool barBoolean = false;
|
|
|
|
// Char - Un solo caracter Unicode de 16 bits
|
|
char fooChar = 'A';
|
|
|
|
// Strings
|
|
string fooString = "My string is here!";
|
|
Console.WriteLine(fooString);
|
|
|
|
// Formato de cadenas
|
|
string fooFs = string.Format("Check Check, {0} {1}, {0} {1:0.0}", 1, 2);
|
|
Console.WriteLine(fooFormattedString);
|
|
|
|
// Formato de fechas
|
|
DateTime fooDate = DateTime.Now;
|
|
Console.WriteLine(fooDate.ToString("hh:mm, dd MMM yyyy"));
|
|
|
|
// \n es un caracter de escape que comienza una nueva línea
|
|
string barString = "Printing on a new line?\nNo Problem!";
|
|
Console.WriteLine(barString);
|
|
|
|
// Puede ser escrito mejor usando el símbolo @
|
|
string bazString = @"Here's some stuff
|
|
on a new line!";
|
|
Console.WriteLine(bazString);
|
|
|
|
// Las comillas deben ser escapadas
|
|
// usa \" para escaparlas
|
|
string quotedString = "some \"quoted\" stuff";
|
|
Console.WriteLine(quotedString);
|
|
|
|
// usa "" cuando las cadenas comiencen con @
|
|
string quotedString2 = @"some MORE ""quoted"" stuff";
|
|
Console.WriteLine(quotedString2);
|
|
|
|
// Usa const o readonly para hacer las variables inmutables
|
|
// los valores const son calculados en tiempo de compilación
|
|
const int HOURS_I_WORK_PER_WEEK = 9001;
|
|
|
|
// Tipos que aceptan valores NULL (Nullable)
|
|
// cualquier tipo de dato puede ser un tipo nulo añadiendole el sufijo ?
|
|
// <tipo>? <variable> = <valor>
|
|
int? nullable = null;
|
|
Console.WriteLine("Nullable variable: " + nullable);
|
|
|
|
// Para usar valores nulos, tienes que usar la propiedad Value
|
|
// o usar conversión explícita
|
|
string? nullableString = "not null";
|
|
Console.WriteLine("Nullable value is: " + nullableString.Value + " or: " + (string) nullableString );
|
|
|
|
// ?? is una manera corta de especificar valores por defecto
|
|
// en caso de que la variable sea null
|
|
int notNullable = nullable ?? 0;
|
|
Console.WriteLine("Not nullable variable: " + notNullable);
|
|
|
|
// var - el compilador escogerá el tipo de dato más apropiado basado en el valor
|
|
var fooImplicit = true;
|
|
|
|
///////////////////////////////////////////////////
|
|
// Estructura de datos
|
|
///////////////////////////////////////////////////
|
|
Console.WriteLine("\n->Data Structures");
|
|
|
|
// Arreglos
|
|
// El tamaño del arreglo debe decidirse al momento de la declaración
|
|
// El formato para declarar un arreglo es el siguiente:
|
|
// <tipo_de_dato>[] <nombre_variable> = new <tipo_de_dato>[<tamaño>];
|
|
int[] intArray = new int[10];
|
|
string[] stringArray = new string[1];
|
|
bool[] boolArray = new bool[100];
|
|
|
|
// Otra forma de declarar e inicializar un arreglo
|
|
int[] y = { 9000, 1000, 1337 };
|
|
|
|
// Indexar arreglos - Acceder a un elemento
|
|
Console.WriteLine("intArray @ 0: " + intArray[0]);
|
|
|
|
// Los arreglos son de índice cero y son mutables.
|
|
intArray[1] = 1;
|
|
Console.WriteLine("intArray @ 1: " + intArray[1]); // => 1
|
|
|
|
// Listas
|
|
// Las listas son usadas más frecuentemente que los arreglos ya que son más flexibles
|
|
// El formato para declarar una lista es el siguiente:
|
|
// List<tipo_de_dato> <nombre_variable> = new List<tipo_de_dato>();
|
|
List<int> intList = new List<int>();
|
|
List<string> stringList = new List<string>();
|
|
|
|
// Otra forma de declarar e inicializar una lista
|
|
List<int> z = new List<int> { 9000, 1000, 1337 };
|
|
|
|
// Indexar una lista - Acceder a un elemento
|
|
// Las listas son de índice cero y son mutables.
|
|
Console.WriteLine("z @ 0: " + z[2]);
|
|
|
|
// Las listas no tienen valores por defecto;
|
|
// Un valor debe ser añadido antes de acceder al índice
|
|
intList.Add(1);
|
|
Console.WriteLine("intList @ 0: " + intList[0]);
|
|
|
|
|
|
// Otras estructuras de datos a chequear:
|
|
//
|
|
// Pilas/Colas
|
|
// Diccionarios
|
|
// Colecciones de sólo lectura
|
|
// Tuplas (.Net 4+)
|
|
|
|
|
|
///////////////////////////////////////
|
|
// Operadores
|
|
///////////////////////////////////////
|
|
Console.WriteLine("\n->Operators");
|
|
|
|
int i1 = 1, i2 = 2; // Modo corto para múltiples declaraciones
|
|
|
|
// La aritmética es sencilla
|
|
Console.WriteLine("1+2 = " + (i1 + i2)); // => 3
|
|
Console.WriteLine("2-1 = " + (i2 - i1)); // => 1
|
|
Console.WriteLine("2*1 = " + (i2 * i1)); // => 2
|
|
Console.WriteLine("1/2 = " + (i1 / i2)); // => 0 (0.5 truncated down)
|
|
|
|
// Módulo
|
|
Console.WriteLine("11%3 = " + (11 % 3)); // => 2
|
|
|
|
// Operadores de comparación
|
|
Console.WriteLine("3 == 2? " + (3 == 2)); // => false
|
|
Console.WriteLine("3 != 2? " + (3 != 2)); // => true
|
|
Console.WriteLine("3 > 2? " + (3 > 2)); // => true
|
|
Console.WriteLine("3 < 2? " + (3 < 2)); // => false
|
|
Console.WriteLine("2 <= 2? " + (2 <= 2)); // => true
|
|
Console.WriteLine("2 >= 2? " + (2 >= 2)); // => true
|
|
|
|
// Operadores a nivel de bits
|
|
/*
|
|
~ Complemento a nivel de bits
|
|
<< Desplazamiento a la izquierda con signo
|
|
>> Desplazamiento a la derecha con signo
|
|
>>> Desplazamiento a la derecha sin signo
|
|
& AND a nivel de bits
|
|
^ XOR a nivel de bits
|
|
| OR a nivel de bits
|
|
*/
|
|
|
|
// Incremento
|
|
int i = 0;
|
|
Console.WriteLine("\n->Inc/Dec-remento");
|
|
Console.WriteLine(i++); //i = 1. Posincrementación
|
|
Console.WriteLine(++i); //i = 2. Preincremento
|
|
Console.WriteLine(i--); //i = 1. Posdecremento
|
|
Console.WriteLine(--i); //i = 0. Predecremento
|
|
|
|
|
|
///////////////////////////////////////
|
|
// Estructuras de control
|
|
///////////////////////////////////////
|
|
Console.WriteLine("\n->Control Structures");
|
|
|
|
// Las condiciones if son como en lenguaje c
|
|
int j = 10;
|
|
if (j == 10)
|
|
{
|
|
Console.WriteLine("I get printed");
|
|
}
|
|
else if (j > 10)
|
|
{
|
|
Console.WriteLine("I don't");
|
|
}
|
|
else
|
|
{
|
|
Console.WriteLine("I also don't");
|
|
}
|
|
|
|
// Operador ternario
|
|
// Un simple if/else puede ser escrito de la siguiente manera;
|
|
// <condición> ? <true> : <false>
|
|
string isTrue = (true) ? "True" : "False";
|
|
Console.WriteLine("Ternary demo: " + isTrue);
|
|
|
|
|
|
// Bucle while
|
|
int fooWhile = 0;
|
|
while (fooWhile < 100)
|
|
{
|
|
//Console.WriteLine(fooWhile);
|
|
//Incrementar el contador
|
|
//Iterar 99 veces, fooWhile 0->99
|
|
fooWhile++;
|
|
}
|
|
Console.WriteLine("fooWhile Value: " + fooWhile);
|
|
|
|
// Bucle Do While
|
|
int fooDoWhile = 0;
|
|
do
|
|
{
|
|
//Console.WriteLine(fooDoWhile);
|
|
//Incrementar el contador
|
|
//Iterar 99 veces, fooDoWhile 0->99
|
|
fooDoWhile++;
|
|
} while (fooDoWhile < 100);
|
|
Console.WriteLine("fooDoWhile Value: " + fooDoWhile);
|
|
|
|
// Bucle For
|
|
int fooFor;
|
|
//Estructura del bucle for => for(<declaración_inicial>; <condición>; <incremento>)
|
|
for (fooFor = 0; fooFor < 10; fooFor++)
|
|
{
|
|
//Console.WriteLine(fooFor);
|
|
//Iterated 10 times, fooFor 0->9
|
|
}
|
|
Console.WriteLine("fooFor Value: " + fooFor);
|
|
|
|
// Switch Case
|
|
// El switch funciona con los tipos de datos byte, short, char e int
|
|
// También funciona con las enumeraciones (discutidos en in Tipos Enum),
|
|
// la clase string y algunas clases especiales que encapsulan
|
|
// tipos primitivos: Character, Byte, Short, Integer.
|
|
int month = 3;
|
|
string monthString;
|
|
switch (month)
|
|
{
|
|
case 1:
|
|
monthString = "January";
|
|
break;
|
|
case 2:
|
|
monthString = "February";
|
|
break;
|
|
case 3:
|
|
monthString = "March";
|
|
break;
|
|
default:
|
|
monthString = "Some other month";
|
|
break;
|
|
}
|
|
Console.WriteLine("Switch Case Result: " + monthString);
|
|
|
|
|
|
////////////////////////////////
|
|
// Conversión de tipos de datos
|
|
////////////////////////////////
|
|
|
|
// Convertir datos
|
|
|
|
// Convertir String a Integer
|
|
// esto generará una excepción al fallar la conversión
|
|
int.Parse("123");//retorna una versión entera de "123"
|
|
|
|
// TryParse establece la variable a un tipo por defecto
|
|
// en este caso: 0
|
|
int tryInt;
|
|
int.TryParse("123", out tryInt);
|
|
|
|
// Convertir Integer a String
|
|
// La clase Convert tiene algunos métodos para facilitar las conversiones
|
|
Convert.ToString(123);
|
|
|
|
///////////////////////////////////////
|
|
// Clases y Funciones
|
|
///////////////////////////////////////
|
|
|
|
Console.WriteLine("\n->Classes & Functions");
|
|
|
|
// (Definición de la clase Bicycle (Bicicleta))
|
|
|
|
// Usar new para instanciar una clase
|
|
Bicycle trek = new Bicycle();
|
|
|
|
// Llamar a los métodos del objeto
|
|
trek.speedUp(3); // Siempre deberías usar métodos setter y métodos getter
|
|
trek.setCadence(100);
|
|
|
|
// ToString es una convención para mostrar el valor del objeto.
|
|
Console.WriteLine("trek info: " + trek.ToString());
|
|
|
|
// Instanciar otra nueva bicicleta
|
|
Bicycle octo = new Bicycle(5, 10);
|
|
Console.WriteLine("octo info: " + octo.ToString());
|
|
|
|
// Instanciar un Penny Farthing (Biciclo)
|
|
PennyFarthing funbike = new PennyFarthing(1, 10);
|
|
Console.WriteLine("funbike info: " + funbike.ToString());
|
|
|
|
Console.Read();
|
|
} // Fin del método main
|
|
|
|
|
|
} // Fin de la clase LearnCSharp
|
|
|
|
// Puedes incluir otras clases en un archivo .cs
|
|
|
|
|
|
// Sintaxis para la declaración de clases:
|
|
// <public/private/protected> class <nombre_de_clase>{
|
|
// //campos, constructores, funciones todo adentro de la clase.
|
|
// //las funciones son llamadas métodos como en java.
|
|
// }
|
|
|
|
public class Bicycle
|
|
{
|
|
// Campos/Variables de la clase Bicycle
|
|
public int cadence; // Public: Accesible desde cualquier lado
|
|
private int _speed; // Private: Sólo es accesible desde dentro de la clase
|
|
protected int gear; // Protected: Accesible desde clases y subclases
|
|
internal int wheels; // Internal: Accesible en el ensamblado
|
|
string name; // Todo es privado por defecto: Sólo es accesible desde dentro de esta clase
|
|
|
|
// Enum es un tipo valor que consiste un una serie de constantes con nombres
|
|
public enum Brand
|
|
{
|
|
AIST,
|
|
BMC,
|
|
Electra,
|
|
Gitane
|
|
}
|
|
// Definimos este tipo dentro de la clase Bicycle, por lo tanto es un tipo anidado
|
|
// El código afuera de esta clase debería referenciar este tipo como Bicycle.Brand
|
|
|
|
public Brand brand; // Declaramos un tipo enum, podemos declarar un campo de este tipo
|
|
|
|
// Los miembros estáticos pertenecen al tipo mismo, no a un objeto en específico.
|
|
static public int bicyclesCreated = 0;
|
|
// Puedes acceder a ellos sin referenciar ningún objeto:
|
|
// Console.WriteLine("Bicycles created: " + Bicycle.bicyclesCreated);
|
|
|
|
// Los valores readonly (Sólo lectura) son establecidos en tiempo de ejecución
|
|
// sólo pueden ser asignados al momento de la declaración o dentro de un constructor
|
|
readonly bool hasCardsInSpokes = false; // privado de sólo lectura
|
|
|
|
// Los constructores son una forma de crear clases
|
|
// Este es un constructor por defecto
|
|
private Bicycle()
|
|
{
|
|
gear = 1;
|
|
cadence = 50;
|
|
_speed = 5;
|
|
name = "Bontrager";
|
|
brand = Brand.AIST;
|
|
bicyclesCreated++;
|
|
}
|
|
|
|
// Este es un constructor específico (contiene argumentos)
|
|
public Bicycle(int startCadence, int startSpeed, int startGear,
|
|
string name, bool hasCardsInSpokes, Brand brand)
|
|
{
|
|
this.gear = startGear; // La palabra reservada "this" señala el objeto actual
|
|
this.cadence = startCadence;
|
|
this._speed = startSpeed;
|
|
this.name = name; // Puede ser útil cuando hay un conflicto de nombres
|
|
this.hasCardsInSpokes = hasCardsInSpokes;
|
|
this.brand = brand;
|
|
}
|
|
|
|
// Los constructores pueden ser encadenados
|
|
public Bicycle(int startCadence, int startSpeed, Brand brand) :
|
|
this(startCadence, startSpeed, 0, "big wheels", true)
|
|
{
|
|
}
|
|
|
|
// Sintaxis para Funciones:
|
|
// <public/private/protected> <tipo_retorno> <nombre_funcion>(<args>)
|
|
|
|
// Las clases pueden implementar getters y setters para sus campos
|
|
// o pueden implementar propiedades
|
|
|
|
// Sintaxis para la declaración de métodos:
|
|
// <ámbito> <tipo_retorno> <nombre_método>(<argumentos>)
|
|
public int GetCadence()
|
|
{
|
|
return cadence;
|
|
}
|
|
|
|
// Los métodos void no requieren usar return
|
|
public void SetCadence(int newValue)
|
|
{
|
|
cadence = newValue;
|
|
}
|
|
|
|
// La palabra reservada virtual indica que este método puede ser sobrescrito
|
|
public virtual void SetGear(int newValue)
|
|
{
|
|
gear = newValue;
|
|
}
|
|
|
|
// Los parámetros de un método pueden tener valores por defecto.
|
|
// En este caso, los métodos pueden ser llamados omitiendo esos parámetros
|
|
public void SpeedUp(int increment = 1)
|
|
{
|
|
_speed += increment;
|
|
}
|
|
|
|
public void SlowDown(int decrement = 1)
|
|
{
|
|
_speed -= decrement;
|
|
}
|
|
|
|
// Propiedades y valores get/set
|
|
// Cuando los datos sólo necesitan ser accedidos, considera usar propiedades.
|
|
// Las propiedades pueden tener get, set o ambos
|
|
private bool _hasTassles; // variable privada
|
|
public bool HasTassles // acceso público
|
|
{
|
|
get { return _hasTassles; }
|
|
set { _hasTassles = value; }
|
|
}
|
|
|
|
// Las propiedades pueden ser auto implementadas
|
|
public int FrameSize
|
|
{
|
|
get;
|
|
// Puedes especificar modificadores de acceso tanto para get como para set
|
|
// esto significa que sólo dentro de la clase Bicycle se puede modificar Framesize
|
|
private set;
|
|
}
|
|
|
|
//Método para mostrar los valores de atributos de este objeto.
|
|
public override string ToString()
|
|
{
|
|
return "gear: " + gear +
|
|
" cadence: " + cadence +
|
|
" speed: " + _speed +
|
|
" name: " + name +
|
|
" cards in spokes: " + (hasCardsInSpokes ? "yes" : "no") +
|
|
"\n------------------------------\n"
|
|
;
|
|
}
|
|
|
|
// Los métodos también pueden ser estáticos. Puede ser útil para métodos de ayuda
|
|
public static bool DidWeCreateEnoughBycles()
|
|
{
|
|
// Dentro de un método esático,
|
|
// Sólo podemos hacer referencia a miembros estáticos de clases
|
|
return bicyclesCreated > 9000;
|
|
} // Si tu clase sólo necesita miembros estáticos,
|
|
// considera establecer la clase como static.
|
|
|
|
} // fin de la clase Bicycle
|
|
|
|
// PennyFarthing es una subclase de Bicycle
|
|
class PennyFarthing : Bicycle
|
|
{
|
|
// (Penny Farthings son las bicicletas con una rueda delantera enorme.
|
|
// No tienen engranajes.)
|
|
|
|
// llamar al constructor de la clase padre
|
|
public PennyFarthing(int startCadence, int startSpeed) :
|
|
base(startCadence, startSpeed, 0, "PennyFarthing", true)
|
|
{
|
|
}
|
|
|
|
public override void SetGear(int gear)
|
|
{
|
|
gear = 0;
|
|
}
|
|
|
|
public override string ToString()
|
|
{
|
|
string result = "PennyFarthing bicycle ";
|
|
result += base.ToString(); // Llamar a la versión base del método
|
|
return reuslt;
|
|
}
|
|
}
|
|
|
|
// Las interfaces sólo contienen las declaraciones
|
|
// de los miembros, sin la implementación.
|
|
interface IJumpable
|
|
{
|
|
void Jump(int meters); // todos los miembros de interfaces son implícitamente públicos
|
|
}
|
|
|
|
interface IBreakable
|
|
{
|
|
// Las interfaces pueden contener tanto propiedades como métodos, campos y eventos
|
|
bool Broken { get; }
|
|
}
|
|
|
|
// Las clases sólo heredan de alguna otra clase, pero pueden implementar
|
|
// cualquier cantidad de interfaces
|
|
class MountainBike : Bicycle, IJumpable, IBreakable
|
|
{
|
|
int damage = 0;
|
|
|
|
public void Jump(int meters)
|
|
{
|
|
damage += meters;
|
|
}
|
|
|
|
public void Broken
|
|
{
|
|
get
|
|
{
|
|
return damage > 100;
|
|
}
|
|
}
|
|
}
|
|
} // Fin del espacio de nombres
|
|
|
|
```
|
|
|
|
## Temas no cubiertos
|
|
|
|
* Flags
|
|
* Attributes
|
|
* Generics (T), Delegates, Func, Actions, lambda expressions
|
|
* Static properties
|
|
* Exceptions, Abstraction
|
|
* LINQ
|
|
* ASP.NET (Web Forms/MVC/WebMatrix)
|
|
* Winforms
|
|
* Windows Presentation Foundation (WPF)
|
|
|
|
|
|
|
|
## Lecturas recomendadas
|
|
|
|
* [DotNetPerls](http://www.dotnetperls.com)
|
|
* [C# in Depth](http://manning.com/skeet2)
|
|
* [Programming C#](http://shop.oreilly.com/product/0636920024064.do)
|
|
* [LINQ](http://shop.oreilly.com/product/9780596519254.do)
|
|
* [MSDN Library](http://msdn.microsoft.com/es-es/library/618ayhy6.aspx)
|
|
* [ASP.NET MVC Tutorials](http://www.asp.net/mvc/tutorials)
|
|
* [ASP.NET Web Matrix Tutorials](http://www.asp.net/web-pages/tutorials)
|
|
* [ASP.NET Web Forms Tutorials](http://www.asp.net/web-forms/tutorials)
|
|
* [Windows Forms Programming in C#](http://www.amazon.com/Windows-Forms-Programming-Chris-Sells/dp/0321116208)
|
|
|
|
|
|
|
|
[Convenciones de código de C#](http://msdn.microsoft.com/es-es/library/vstudio/ff926074.aspx)
|