Saltar para o conteúdo

C Sharp

Origem: Wikipédia, a enciclopédia livre.
 Nota: o título deste verbete seria C#. Porém, devido a limitações técnicas sobre o uso de cerquilhas em títulos, utilizou-se C Sharp.
 Nota: Este artigo é sobre a linguagem de programação. Para a nota/acorde musical, veja Dó sustenido.
C#
C Sharp
Paradigma
Surgido em julho de 2000 (24 anos)
Última versão 12.0 (14 de novembro de 2023; há 9 meses[1])
Criado por Microsoft
Estilo de tipagem
  • estática
  • dinâmica
  • forte
  • segura
  • insegura
  • nominativa
  • inferida
Principais implementações
Dialetos:
Influenciada por
Influenciou
Licença MIT
Extensão do arquivo .cs
Página oficial msdn.microsoft.com

C# é uma linguagem de programação, multiparadigma, de tipagem forte, desenvolvida pela Microsoft como parte da plataforma .NET. A sua sintaxe orientada a objetos foi baseada no C++ mas inclui muitas influências de outras linguagens de programação, como Object Pascal e, principalmente, Java. O código fonte é compilado para Common Intermediate Language (CIL) que é interpretado pela máquina virtual Common Language Runtime (CLR). C# é uma das linguagens projetadas para funcionar na Common Language Infrastructure da plataforma .NET Framework.[5]

Objetivos do projeto

[editar | editar código-fonte]

A norma ECMA lista os objetivos de design para C#:[5]

  • A linguagem C# destina-se a ser simples, moderna, de propósito geral e orientada a objetos.[5]
  • A linguagem e suas implementações devem fornecer suporte para princípios de engenharia de software, tais como verificação de tipo forte, verificação de limites de array, detecção de tentativas de usar variáveis não inicializadas e coleta automática de lixo. A robustez do software, a durabilidade e a produtividade do programador são importantes.[5]
  • A linguagem destina-se a ser utilizada no desenvolvimento de componentes de software adequados para implantação em ambientes distribuídos.[5]
  • A portabilidade é muito importante para o código fonte e programadores, especialmente aqueles já familiarizados com C e C++.[5]
  • O apoio à internacionalização é muito importante.[5]
  • C# deve ser adequada para escrever aplicações tanto para sistemas hospedados e incorporados, que vão desde o muito grande que usam sistemas operacionais sofisticados, até o muito pequeno com funções dedicadas.[5]
  • Embora aplicações C# se destinam a ser econômicas no que diz respeito à memória e requisitos de energia de processamento, a linguagem não foi concebida para competir diretamente no desempenho e tamanho com C ou linguagem Assembly.[5]
Anders Hejlsberg - criador do C#.

A linguagem C# faz parte do conjunto de ferramentas oferecidas na plataforma .NET e surge como uma linguagem simples, robusta, orientada a objetos, fortemente tipada e altamente escalável a fim de permitir que uma mesma aplicação possa ser executada em diversos dispositivos de hardware, independentemente destes serem PCs, handhelds ou qualquer outro dispositivo móvel.[6]

O avanço das ferramentas de programação e dos dispositivos eletrônicos inteligentes criou problemas e novas exigências. As novas versões de componentes compartilhados eram incompatíveis com o software antigo. Os desenvolvedores reconheceram a necessidade de software que fosse acessível para qualquer um e disponível por meio de praticamente qualquer tipo de dispositivo. Para tratar dessas necessidades, a Microsoft anunciou sua iniciativa .NET e a linguagem de programação C#.[6]

Durante o desenvolvimento da plataforma .NET, as bibliotecas foram escritas originalmente numa linguagem chamada Simple Managed C (SMC), que tinha um compilador próprio. Mas, em Janeiro de 1999, uma equipe de desenvolvimento foi formada por Anders Hejlsberg, que fora escolhido pela Microsoft para desenvolver a linguagem. Dá-se início à criação da linguagem chamada Cool. Um pouco mais tarde, em 2000, o projeto .NET era apresentado ao público na Professional Developers Conference (PDC), e a linguagem Cool fora renomeada e apresentada como C#.[6]

A criação da linguagem, embora tenha sido feita por vários programadores, é atribuída principalmente a Anders, hoje um Distinguished Engineer na Microsoft. Ele fora o arquiteto de alguns compiladores da Borland, e entre suas criações mais conhecidas estão o Turbo Pascal e o Delphi.[6]

A Microsoft submeteu o C# à ECMA para uma padronização formal. Em Dezembro de 2001, a associação liberou a especificação ECMA-334 Especificação da Linguagem C#. Em 2003, tornou-se um padrão ISO (ISO/IEC 23270). Há algumas implementações em desenvolvimento, destacando-se a Mono, implementação open source da Novell, o dotGNU e o Portable.NET, implementações da Free Software Foundation, e o BDS 2008, implementação da CodeGear.[6]

A nota sustenido em clave de sol.

Pensava-se que o nome "C#" viria duma sobreposição de quatro símbolos +, dando a impressão de ++++, uma alusão à continuação do C++.[não consta na fonte citada] Entretanto, a cerquilha de "C#" se refere ao sinal musical sustenido (♯, em inglês: sharp), que aumenta em meio tom uma nota musical.[7]

Porém, devido a limitações técnicas (fontes padrões, navegadores, etc) e o fato do símbolo não estar presente nos teclados, o cerquilha (#) foi escolhido para ser usado no nome escrito.[8] Essa convenção é refletida no ECMA-334 C# Language Specification, a especificação técnica da linguagem.[5] Entretanto, em determinados lugares, como em propagandas e capas de livros, é usado o símbolo de sustenido.[9]

C# 1.0[10]
  • Alia o poder de C++ com a simplicidade do Visual Basic[10]
  • Código gerenciado com a noção de assemblies carregando o bytecode com segurança embutida e controle de versões[10]
  • Orientação a objeto[10]
  • Tudo deriva de Object[10]
  • Os tipos são Estruturas, Enumerações, Classes, Interfaces, Delegados e Ponteiros (só em contexto unsafe)[10]
  • Tipagem estática[10]
  • Eventos[10]
  • Interoperabilidade com código não gerenciado através de DLL nativa e componentes COM[10]
  • Coletor de lixo de memória baseado em gerações[10]
  • Compilador Just-in-time[10]
  • Facilidades de integração com ferramentas e IDE[10]
  • Sintaxe para documentação[10]
  • Sistema de exceção uniforme[10]
  • Mecanismo de metadados[10]
  • Extensiva biblioteca padrão[10]
C# 2.0[11]
  • Tipos e métodos genéricos[11]
  • Tipos Parciais[11]
  • Métodos anônimos[11]
  • Iteradores[11]
  • Tipos anuláveis[11]
  • Acessibilidade separada para Getter/setter[11]
  • Conversões de grupo de métodos para delegados[11]
  • Co e Contravariância para delegados[11]
  • Classes estáticas[11]
  • Inferência de delegado[11]
  • Interface web[11]
C# 3.0[12]
  • Variáveis locais implicitamente tipadas[12]
  • Inicializadores de objetos e coleções[12]
  • Propriedades auto-implementadas[12]
  • Tipos anônimos[12]
  • Métodos de extensão[12]
  • Expressões de consulta[12]
  • Expressões Lambda[12]
  • Árvores de expressão[12]
  • Métodos parciais[12]
C# 4.0[13]
  • Vinculação dinâmica[13]
  • Argumentos nomeados e opcionais[13]
  • Co e contravariância genérica[13]
  • Tipos embutidos de interoperabilidade ("NoPIA")[40][13]
C# 5.0[14]
  • Métodos assíncronos[14]
  • Atributos de informação do chamador[14]
C# 6.0[15]
  • Compilador como serviço (Roslyn)[15]
  • Importação de membros de tipos estáticos no espaço de nomes[15]
  • Filtros de exceção[15]
  • Await em blocos catch/finally[15]
  • Inicializadores de propriedades automáticas[15]
  • Valores padrões para propriedades apenas getter[15]
  • Membros com corpos como expressão[15]
  • Propagador de Null (operador condicional de null, checagem de null sucinta)[43][15]
  • Interpolação de string[15]
  • Operador nameof[15]
  • Inicializador de dicionário[15]
C# 7.0[16]

Características em processo de desenvolvimento[16]

  • Literais binários[16]
  • Separadores de dígitos[16]
  • Funções locais[16]
  • Switch para tipos[16]
  • Retorno de referências[16]
  • Tuplas[16]
  • Declaração Out (Out var ou Out "tipo")[16]
  • Pattern Matching[16]
  • Returno de async arbitrário[16]
  • Registros (Record)[16]
C# 7.1[17]
  • async Método Main[17]
  • default Expressões literais[17]
  • Nomes de elementos de tupla inferidos[17]
C# 7.2[18]
  • Técnicas para escrever código eficiente seguro[18]
  • Argumentos nomeados que não estejam à direita[18]
  • Sublinhados à esquerda em literais numéricos[18]
  • Modificador de acesso private protected[18]
  • Expressões ref condicionais[18]
C# 7.3[19]
C# 8.0[1]
C# 9.0[20]
C# 10.0[21]
  • Diretivas Using Globais e Implícitas[21]
  • Declaração de Namespace com escopo de arquivo[21]
  • Melhorias dos tipos de estrutura[21]
  • Manipulador de strings interpolado[21]

Características

[editar | editar código-fonte]

O C# é uma linguagem de programação visual dirigida por eventos e totalmente orientada a objetos. Permite um novo grau de intercâmbio entre linguagens (componentes de software de diferentes linguagens podem interagir). Os desenvolvedores podem empacotar até software antigo, para trabalhar com novos programas C#. Além disso, os aplicativos C# podem interagir pela Internet usando padrões do setor, como SOAP (protocolo de acesso a objetos simples) e XML (linguagem de marcação extensível).[22]

O C# tem raízes em C, C++ e Java, adaptando os melhores recursos de cada linguagem e acrescentando novas capacidades próprias. Ele fornece os recursos que são mais importantes para os programadores, como programação orientada a objetos, strings, elementos gráficos, componentes de interface com o usuário gráfica (GUI), tratamento de exceções, múltiplas linhas de execução, multimídia (áudio, imagens, animação e vídeo), processamento de arquivos, estruturas de dados pré-empacotadas, processamento de banco de dados, redes cliente/servidor com base na Internet e na World Wide Web e computação distribuída.[22]

Principais características do C#

[editar | editar código-fonte]

Dentre as características essenciais do C# podemos citar:[22]

  • Simplicidade: os projetistas de C# costumam dizer que essa linguagem é tão poderosa quanto o C++ e tão simples quanto o Visual Basic;[22]
  • Completamente orientada a objetos: em C#, qualquer variável tem de fazer parte de uma classe;[22]
  • Fortemente tipada: isso ajudará a evitar erros por manipulação imprópria de tipos e atribuições incorretas;[22]
  • Gera código gerenciado: assim como o ambiente .NET é gerenciado, assim também é o C#;[22]
  • Tudo é um objeto: System.Object é a classe base de todo o sistema de tipos de C#;[22]
  • Controle de versões: cada assembly gerado, seja como EXE ou DLL, tem informação sobre a versão do código, permitindo a coexistência de dois assemblies homônimos, mas de versões diferentes no mesmo ambiente;[22]
  • Suporte a código legado: o C# pode interagir com código legado de objetos COM e DLLs escritas em uma linguagem não gerenciada;[22]
  • Flexibilidade: se o desenvolvedor precisar usar ponteiros, o C# permite, mas ao custo de desenvolver código não gerenciado, chamado “unsafe”;[22]
  • Linguagem gerenciada: os programas desenvolvidos em C# executam num ambiente gerenciado, o que significa que todo o gerenciamento de memória é feito pelo runtime via o GC (Garbage Collector).[23]

A sintaxe da linguagem C# é semelhante à de outras linguagens de estilo C, como C, C ++ e Java. Em particular:[22]

  • Os pontos e vírgulas são usados para denotar o fim de uma declaração.[22]
  • As chaves são usadas para agrupar declarações. As declarações são geralmente agrupadas em métodos (funções), métodos em classes, e classes em namespaces.[22]
  • As variáveis são atribuídas usando um sinal de igual, mas comparadas usando dois sinais iguais consecutivos.[22]
  • Colchetes são usados com arrays, tanto para declará-los e para obter um valor em um determinado índice em um deles.[22]
  • Possui uma forma limitada de diretivas de compilação.[22]
  • Essencialmente imperativa, como todas linguagens mainstream.[22]

Orientação a objeto

[editar | editar código-fonte]

Um dos principais recursos da programação orientada a objetos é a herança. Herança é uma forma de reutilização de software em que as classes são criadas pela absorção dos dados e comportamentos de uma classe já existente e aprimorando-as com novos recursos. Ao criar uma classe, em vez de escrever variáveis de instância e métodos complementares novos, o programador pode indicar que a nova classe deve herdar as variáveis de classe, propriedades e métodos de outra classe. Uma classe derivada é mais específica do que sua classe base e representa um grupo mais especializado de objetos. Normalmente, a classe derivada contém os comportamentos de sua classe base e comportamentos adicionais. No caso de herança simples, uma classe é derivada de uma única classe base. Ao contrário do C++, o C# não suporta herança múltipla (que ocorre quando uma classe é derivada de mais de uma classe base direta).[24]

Esta foi uma decisão de design do arquiteto principal da linguagem para evitar complicações e simplificar os requisitos arquitetônicos em toda a CLI. Ao implementar múltiplas interfaces que contêm um método com a mesma assinatura, C# permite implementar cada método dependendo da interface com a qual esse método está sendo chamado ou, permite implementar o método uma vez e ter que ser uma invocação em uma chamada através de qualquer uma das interfaces da classe.[24]

A linguagem permite que se acesse um objeto como se ele fosse de outro tipo de acordo com o contrato, de maneira polimórfica.[24]

As classes podem ser abstratas e exigirem que sejam derivadas através da palavra-chave abstract. Nesse tipo de classe métodos não precisam ser obrigatoriamente implementados deixando para suas filhas implementar.[24]

Existem algumas visibilidades possíveis para determinar o encapsulamento:[24]

public Os membros assim declarados podem ser acessado de qualquer local da aplicação[24]
private Apenas os membros da classe atual podem acessar um membro declarado como privado[24]
protected Somente pode ser acessado pela própria classe e as derivadas dela[24]
internal Qualquer classe dentro um mesmo assembly pode acessar esse membro[24]
private protected Pode ser acessado pela classe atual e derivadas, dentro do mesmo assembly[24]

C# suporta sobrecarga de alguns operadores.[24]

Programação funcional

[editar | editar código-fonte]

Embora primariamente uma linguagem imperativa, até o C# 2.0 oferecia suporte limitado para programação funcional através de funções de primeira classe e encerramentos sob a forma de delegados anônimos. O suporte foi expandido no C# 3.0 para programação funcional com a introdução de uma sintaxe leve para expressões lambda, métodos de extensão e uma sintaxe de compreensão de lista na forma de uma linguagem de "compreensão de consulta". Em C# 7 mais um passo é dado com novos conceitos, tais como: records, pattern matching, tuplas, etc.[25]

Tipos de dados

[editar | editar código-fonte]

C# tem um sistema de tipo de dados unificado. Este sistema de tipo unificado é chamado Common Type System (CTS).[26]

Um sistema de tipo unificado implica que todos os tipos, incluindo primitivos, como inteiros, são subclasses da classe System.Object. Por exemplo, cada tipo herda um método ToString().[27]

O CTS separa os tipos de dados em duas categorias:[27]

Tipos de valor

[editar | editar código-fonte]

As instâncias de tipos de valores não têm identidade referencial nem semântica de comparação referencial - comparações de igualdade e desigualdade para tipos de valor comparam os valores de dados reais dentro das instâncias, a menos que os operadores correspondentes estejam sobrecarregados. Tipos de valor são derivados de System.ValueType, sempre têm um valor padrão e sempre podem ser criados e copiados. Algumas outras limitações em tipos de valor são que eles não podem derivar uns dos outros (mas podem implementar interfaces) e não podem ter um construtor padrão explícito (sem parâmetros).[27]

Os tipos de valor normalmente representam dados simples, como valores int ou bool. Os tipos de valor próprios da linguagem são os tipos integrais, o tipo decimal e os tipos de ponto flutuante float (um número de ponto flutuante IEEE de 32 bits) e double, e ainda char (uma unidade de código Unicode de 16 bits) e System.DateTime que identifica um ponto específico no tempo com nanossegundo de precisão). Outros exemplos são enum (enumerações) e struct (estruturas definidas pelo usuário).[27]

Tipos de referência

[editar | editar código-fonte]

Em contraste, os tipos de referência têm a noção de identidade referencial - cada instância de um tipo de referência é inerentemente distinta de todas as outras instâncias, mesmo se os dados dentro de ambas as instâncias forem iguais. Isso é refletido em comparações de igualdade e desigualdade padrão para tipos de referência, que testam a igualdade referencial em vez da estrutural, a menos que os operadores correspondentes estejam sobrecarregados (como o caso para System.String). Em geral, nem sempre é possível criar uma instância de um tipo de referência, nem copiar uma instância existente ou executar uma comparação de valor em duas instâncias existentes, embora tipos de referência específicos possam fornecer tais serviços expondo um construtor público ou implementando um Interface correspondente (como ICloneable ou IComparable). Exemplos de tipos de referência são objeto (a última classe base para todas as outras classes C #), System.String (uma seqüência de caracteres Unicode) e System.Array (uma classe base para todos os arrays C #).[27]

Ambas as categorias de tipos são extensíveis com tipos definidos pelo usuário.[27]

Box e unboxing

[editar | editar código-fonte]

Boxing é a operação de converter um objeto de tipo de valor em um valor de um tipo de referência correspondente. Box em C# é implícito.[27]

Unboxing é a operação de converter um valor de um tipo de referência (previamente encaixotado) em um valor de um tipo de valor. Unboxing em C # requer um cast de tipo explícito. Um objeto encapsulado de tipo T só pode ser unboxed para um T (ou um T anulável).[28]

Exemplo:

int foo = 42;         // Value type.
object bar = foo;     // foo is boxed to bar.
int foo2 = (int)bar;  // Unboxed back to value type.

Existem dois tipos de conversões no C#: implícita e explícita. C# é mais segura do ponto de vista de tipos que C++. Apenas conversões seguras são feitas implicitamente. Não há conversão implícita de inteiro para booleano, por exemplo.[27]

Tempo de vida

[editar | editar código-fonte]

Segundo seu escopo e disponibilidade, as variáveis podem ser do tipo static (o valor fica disponível para todos os objetos da classe em que a variável foi declarada), de instância (criada na memória cada vez que instanciamos um novo objeto).[27]

Declaração implícita

[editar | editar código-fonte]

C# permite declarar variáveis locais estaticamente tipadas sem indicar seu tipo através da palavra-chave var. Arrays são declarados implicitamente com new[]. Também é possível declarar variáveis diversas com tipagem dinâmica (não há verificação em tempo de compilação) com dynamic.[27]

Classes e Estruturas

[editar | editar código-fonte]

Em C#, a unidade de programação é a classe (class). Os objetos são eventualmente instanciados (criados) dessas classes, e as funções são encapsuladas dentro dos “limites” das classes como métodos. Definições de classe C# não necessitam de arquivos de cabeçalho ou arquivos separados (IDL, Definition Language). Em C#, uma classe pode herdar de apenas um objeto pai (parent), mas uma classe pode implementar múltiplas interfaces. Elas são tipos por referência.[27]

C# proporciona também um suporte a estruturas (struct). Esses são os tipo por valor.[27]

Classes estáticas

[editar | editar código-fonte]

Além das classes normais é possível criar classes estáticas onde os métodos não trabalham com instâncias e se assemelham com funções normais.[27]

Classe Console
[editar | editar código-fonte]

A classe Console, do namespace System, fornece suporte básico para leitura e escrita de caracteres. Os principais métodos usados para essas funcionalidades são:[27]

  • Write: escreve uma informação específica em tela.[27]
  • WriteLine: escreve uma informação específica em tela e pula uma linha em seguida.[27]
  • ReadLine: captura uma linha de texto digitada pelo usuário e retorna seu valor na forma de string.[27]
  • ReadKey: captura um único caractere digitado pelo usuário. O retorno desse método é um objeto da classe ConsoleKeyInfo, que contém informações sobre a tecla digitada pelo usuário.[27]
  • Clear: esse método limpa toda a tela do console. Útil para programas que exigem formatação dos resultados.[27]
Classe Convert
[editar | editar código-fonte]

Muitas vezes precisamos converter o tipo para outro tipo. Neste caso, podemos usar a classe Convert, que fornece diversos métodos para conversão entre os tipos básicos de dados. Nem sempre é possível fazer a conversão e um erro é gerado, por isso existem outros métodos nos tipos que podem tratar a conversão de forma mais apropriada. Exemplo:[27]

int num;
string texto = 2;
num = Convert.ToInt32 (texto);

Os métodos da classe Math permitem que o programador efetue certos cálculos matemáticos comuns. Os métodos são chamados escrevendo-se o nome do método, seguido de um parêntese de abertura, do argumento (ou uma lista de argumentos separados por vírgulas) do método e um parêntese de fechamento. Exemplo: para calcular e imprimir a raiz quadrada de 900.0 podemos escrever:[27]

Console.WriteLine(Math.Sqrt(900.0));

ou de forma compacta:

WriteLine(Sqrt(900.0));

Interfaces pertencem aos tipos de referência, em que definem contratos que são implementados por uma classe ou estrutura (que a utiliza para simular herança já que não oferece suporte para herdar outra estrutura). Uma classe pode implementar múltiplas interfaces.[29][30]

Uma interface pode conter métodos, propriedades, eventos e indexadores, mas não pode implementar seus métodos.[31]

Principais Propriedades:[31]

  • Uma interface é quase como uma classe base abstrata. Qualquer classe ou estrutura que implementa a interface deve implementar todos os seus membros;[31]
  • Uma interface não pode ser instanciada diretamente. Seus membros são implementados por qualquer classe ou estrutura que implementa a interface;[31]
  • Interfaces podem conter eventos, indexadores, métodos, propriedades;[31]
  • As interfaces não contêm qualquer implementação dos métodos;[31]
  • Uma classe ou estrutura pode implementar interfaces múltiplas. Uma classe pode herdar uma classe base e também implementar uma ou mais interfaces.[31]
interface ISampleInterface
{
    void SampleMethod();
}
class ImplementationClass : ISampleInterface
{
    // implementação explícita do membro da interface:
    void ISampleInterface.SampleMethod()
    {
        // Implementação do Método.
    }
    static void Main()
    {
        // Instância da interface.
        ISampleInterface obj = new ImplementationClass();
        // Chamada do membro.
        obj.SampleMethod();
    }
}

É um elemento que permite que você faça referência a um método. É semelhante a um ponteiro de função, mas sua diferença se baseia em ser totalmente orientado a objetos encapsulando a instância de método ou objeto.[32]

O objeto de delegação pode ser passado para o código que pode chamar o método referenciado sem ter que saber em tempo de compilação qual método será chamado. Usando delegates você tem a flexibilidade para implementar qualquer funcionalidade em tempo de execução. A única restrição é que o método seja compatível com o tipo de declaração do delegate, usando assim perfeitamente invocação anônima. Delegates possuem a seguintes propriedades:[33]

  • São como ponteiros em C++, mas são do tipo seguro;[33]
  • Permitem métodos para serem passados por parâmetros;[33]
  • Podem ser usados para definir métodos callback.[33]

Exemplo:

// Declare delegate -- defines required signature:
delegate double MathAction (double num);
class DelegateTest
{
    // Regular method that matches signature:
    static double Double(double input)
    {
        return input * 2;
    }
    static void Main()
    {
        // Instantiate delegate with named method:
        MathAction ma = Double;
        // Invoke delegate ma:
        double multByTwo = ma(4.5);
        Console.WriteLine("multByTwo: {0}", multByTwo);
        // Instantiate delegate with anonymous method:
        MathAction ma2 = delegate(double input)
        {
            return input * input;
        };

        double square = ma2(5);
        Console.WriteLine("square: {0}", square);
        // Instantiate delegate with lambda expression

        MathAction ma3 = s => s * s * s;

        double cube = ma3(4.375);
        Console.WriteLine("cube: {0}", cube);
    }
}
// Output:
// multByTwo: 9
// square: 25
// cube: 83.740234375

Enum em C# é usado para declarar uma enumeração, um tipo distinto que consiste em um conjunto de constantes nomeados lista de enumerador. Usados geralmente dentro de namespace para ser visível para acesso das classes e usados dentro de classes e structs, sendo considerados constantes numéricas. Enumerações mantém seus membros em seu próprio escopo. Por característica começa com 0 mas pode ser inicializado com outro valor. Se for atribuído a uma variável, deve ser feita uma conversão explícita do tipo enum para o tipo da variável tratada. Exemplo:[34]

public class EnumTest2
{
    enum Range : long { Max = 2147483648L, Min = 255L };
    static void Main()
    {
        long x = (long)Range.Max;
        long y = (long)Range.Min;
        Console.WriteLine("Max = {0}", x);
        Console.WriteLine("Min = {0}", y);
    }
}
/* Saída:
Max = 2147483648
Min = 255
*/

Método é um membro de uma classe que permite implementar uma ação. Consiste de um cabeçalho (contém um atributo, um modificador, um tipo de retorno, um nome do membro e uma lista formal de parâmetros) e um corpo (bloco de declarações). O C# inclui 8 modificadores:[35]

new O modificador new em um método indica que o método derivado é planejado para esconder o método base
public É usado para garantir que um método é acessado por todas as classes sem qualquer restrição
static O modificador static indica que o método pertence somente ao tipo static e não pode referir-se a métodos non-static ou variáveis em uma classe
virtual É usado quando a implementação de um método pode ser mudada por classes derivadas
sealed Em um método, previne uma classe derivada sobrescrever um método
override É usado para modificar um método de uma classe base na classe derivada
abstract Em um método, indica que o método não contém implementação
extern Indica que o método é implementado externamente

O C# permite que vários métodos de mesmo nome sejam definidos na mesma classe, desde que esses métodos tenham diferentes conjuntos de parâmetros (número, tipos ou ordem dos parâmetros). Isso é chamado sobrecarga de métodos. Quando um método sobrecarregado é chamado, o compilador C# seleciona o método correto examinando o número, os tipos e a ordem dos argumentos da chamada.[35]

Métodos de extensão

[editar | editar código-fonte]

Métodos de extensão em C # permitem que os programadores usem métodos estáticos como se fossem métodos da tabela de métodos de uma classe, permitindo que adicionem métodos a um objeto que eles acham que deveria existir nesse tipo e suas derivadas.[35]

Passagem de parâmetros

[editar | editar código-fonte]

Os tipos de parâmetros são quatro:[36]

  • valor: corresponde a uma variável local que recebe seu valor inicial do argumento fornecido quando um método é chamado. Isto acontece porque um parâmetro de valor declarado em um método refere-se à localização da memória diferente daquela do argumento chamado em um método.[36]
  • referência: um parâmetro declarado com o modificador ref é chamado de parâmetro de referência. Diferente do parâmetro de valor, o parâmetro de referência não cria uma nova localização na memória. Refere-se à mesma localização de memória do argumento chamado em um método.[36]
  • saída: um parâmetro declarado com o modificador out é um parâmetro de saída. Parâmetros de saída são usados em métodos que retornam vários valores. Refere-se à mesma localização de memória do argumento chamado em um método. É necessário atribuir um parâmetro de saída antes do método retornar um valor. Em C# 7 é menos incentivado por causa do advento das tuplas. Mas para código legado é possível declarar a variável junto ao argumento.[36]
  • número variável de argumentos: um parâmetro params pode ser um array de única dimensão que recebe vários argumentos.[36]

Uma propriedade é o atributo de uma classe (o tamanho de uma string, por exemplo). Propriedades não denotam localizações de armazenamento mas contém declarações executáveis que são usadas para ler ou escrever o valor da propriedade. As propriedades podem ser dos tipos:[37]

  • estática: usamos o modificador static. Não podemos nos referir a uma propriedade static usando this. Usamos apenas seu nome.[37]
  • instância: acessam um campo da instância.[37]
  • padrão: é a propriedade que é usada sem um nome, funciona como um índice

Os acessores das propriedades são declarações executáveis que leem ou escrevem o valor de uma propriedade. São eles o get (é um método sem nenhum parâmetro que retorna o valor da propriedade) e set (método que atribui o valor em uma propriedade). A propriedade é apenas um açúcar sintático para esse par de métodos.[37]

A palavra-chave de namespace é usada para declarar um escopo que contém um conjunto de objetos relacionados. Você pode usar um namespace para organizar elementos de código e para criar globalmente tipos exclusivos.[38][39][40][41][42]

Namespaces possuem as principais propriedades:[43]

  • Usados para organizar códigos de grandes projetos;[43]
  • A diretiva using evita a necessidade de especificar o nome para cada classe;[43]
  • Há sempre um namespace global que faz referência ao namespace System do .NET framework.[43]

Iteradores são utilizados para percorrer uma coleção de dados como uma lista.[44]

O método iterator é um get personalizado para coleções, em C# usa-se a declaração de produzir retorno de um elemento de cada vez, como o operador yield return recordando o local atual do código e retornando a executar no mesmo local para a próxima iteração.[44]

Por meio do foreach um iterator pode retornar mais de um valor até o fim do bloco ou quando o comando yield break for encontrado. Seus tipos de retorno mais usados são: IEnumerable, IEnumerable<T>, IEnumerator ou IEnumerator<T>.[44]

Exemplo:

static void Main()
{
    foreach (int number in SomeNumbers())
    {
        Console.Write(number.ToString() + " ");
    }
    // Saída: 3 5 8
    Console.ReadKey();
}
public static System.Collections.IEnumerable SomeNumbers()
{
    yield return 3;
    yield return 5;
    yield return 8;
}

Principais comandos de controle

[editar | editar código-fonte]
  • if : executa um bloco de código com base no valor de uma expressão booleana;[45]
  • goto : transfere o controle para uma estrutura de comandos que é identificada por um rótulo;[45]
  • while : executa condicionalmente o código dentro do bloco while zero ou mais vezes;[45]
  • do...while : executa condicionalmente o código dentro do do uma ou mais vezes;[45]
  • for : seu uso é recomendado quando sabemos a quantidade de repetições que devemos mostrar no console;[45]
  • foreach : percorre cada um dos elementos de um array (ou qualquer classe que implementa IEnumerable);[45]
  • switch : executa o bloco de código que é associado com o valor da expressão de controle.[45]

Exceptions – Exceções

[editar | editar código-fonte]

Exceções em C# fornecem uma maneira estruturada, uniforme e segura de manipulação, tanto em nível de sistema e erros condicionais de aplicação. Exceptions em C# são similares a exceções em C++, mas com algumas importantes diferenças:[46]

  • Em C#, todas as exceptions representam um instância de classe derivada de System.Exception. Em C++, qualquer valor de qualquer tipo pode ser usado para representar uma exceção.[46]
  • Em C# o Bloco finally pode ser escrito ao final do código para executar em ambas as situações de exceção ou execução normal. Em C++ é difícil escrever sem duplicar códigos.[46]

Exceções possuem as principais propriedades:[46]

  • Um bloco try em torno de uma instrução pode lançar exceções;[46]
  • Após a execução do bloco try, o fluxo de controle vai para o primeiro manipulador de exceção associado que está presente em qualquer lugar na pilha de chamadas. O catch é usado para definir o manipulador de exceção;[46]
  • Se nenhum manipulador de exceção para uma determinada exceção estiver presente, o programa apresenta erro.[46]

Exceções podem ser explicitamente geradas por um programa usando a palavra-chave throw.[46]

Exemplo:

static void TestCatch2()
{
    System.IO.StreamWritersw = null;
    try
    {
        sw = newSystem.IO.StreamWriter(@"C:\test\test.txt");
        sw.WriteLine("Hello");
    }
    catch (System.IO.FileNotFoundException ex)
    {
        // Put the more specific exception first.
        System.Console.WriteLine(ex.ToString());
    }
    catch (System.IO.IOException ex)
    {
        // Put the less specific exception last.
        System.Console.WriteLine(ex.ToString());
    }
    finally
    {
        sw.Close();
    }
    System.Console.WriteLine("Done");
}

Coleta de lixo e gerenciamento de memória

[editar | editar código-fonte]

O operador new aloca memória para o objeto e, então, chama o construtor desse objeto. Ao contrário do C e do C++, em que os programadores devem gerenciar a memória de maneira explícita, o C# realiza o gerenciamento de memória internamente. A plataforma .NET realiza a coleta de lixo da memória para retornar a memória do sistema que não é mais necessária. Quando o coletor de lixo executa, ele localiza os objetos para os quais o aplicativo não tem referências. Tais objetos podem ser coletados nesse momento ou durante uma execução subsequente do coletor de lixo.[47]

A alocação e liberação de recursos, como conexões de rede, conexões de banco de dados e arquivos, devem ser manipuladas explicitamente pelo programador. Uma técnica empregada em conjunto com o coletor de lixo é definir um finalizador. Com a sintaxe da declaração (não confundir com a diretiva) using[47]

Em C#, os ponteiros de endereço de memória só podem ser usados dentro de blocos especificamente marcados como unsafe e programas com código inseguro precisam de permissões apropriadas para serem executados. A maior parte do acesso a objetos é feita através de referências de objeto seguras, que sempre apontam para um objeto "vivo" ou têm o valor nulo bem definido; É impossível obter uma referência a um objeto "morto" (um que foi coleta de lixo), ou a um bloco de memória aleatória. Um ponteiro inseguro pode apontar para uma instância de um tipo de valor, matriz, sequência de caracteres ou um bloco de memória alocada em uma pilha. Código que não está marcado como inseguro ainda pode armazenar e manipular ponteiros através do tipo System.IntPtr, mas não pode desreferenciá-los.[48]

public class Ponteiro
{
    unsafe static void Main()
    {
        int i = 7;
        int* j = &i;
        //Saída: 7
        System.Console.WriteLine(*j);
    }
}

Portabilidade

[editar | editar código-fonte]

Por design, C# é a linguagem de programação que reflete mais diretamente a infraestrutura de linguagem comum (CLI).[49] A maioria dos seus tipos intrínsecos corresponde a tipos de valor implementados pelo framework CLI. No entanto, a especificação de linguagem não indica os requisitos de geração de código do compilador: ou seja, não indica que um compilador C# deve segmentar um Common Language Runtime, ou gerar Common Intermediate Language (CIL) ou gerar qualquer outro formato específico. Teoricamente, um compilador C# poderia gerar código de máquina como compiladores tradicionais de C++, Pascal ou Fortran. De fato há iniciativas para que isso se torne realidade.[50]

Escrevendo um programa

[editar | editar código-fonte]

Um programa em C# contém 4 elementos principais:

  • uma declaração de namespace: contém as bibliotecas de classe que você pode usar em um programa. A diretiva using System especifica que o programa pode utilizar a biblioteca no namespace System.[43]
  • uma classe: contém as declarações de dados e métodos para uma aplicação;[27]
  • um método Main: semelhante ao C++ e Java, todos os programas C# começam executando o método Main;[51]
  • e uma declaração do programa.[51]

Exemplos de código

[editar | editar código-fonte]
Ver artigo principal: Programa Olá Mundo

Segue, abaixo, o programa teste Olá Mundo, que exibe a frase "Olá, Mundo!":

using System;

public class HelloWorld
{
    public static void Main()
    {
        Console.WriteLine("Olá, Mundo!");
    }
}

Compilando o código

[editar | editar código-fonte]

As aplicações desenvolvidas em C# são baseadas em arquivos (com extensão .cs) contendo o código-fonte dos programas. Quando você compila seu código, cria um assembly. Um assembly é uma coleção de arquivos que aparecem ao programador como uma única DLL, biblioteca de link dinâmico (Dynamic Link Library), ou executável (EXE). Em .NET, um assembly é a unidade básica de reutilização, versionamento, segurança e implantação. O CLR proporciona um número de classes para manipulação de assemblies.[52]

Captura de tela do Banshee, um media player desenvolvido em C#

Ao contrário das outras linguagens de programação, nenhuma implementação de C# atualmente inclui qualquer conjunto de bibliotecas de classes ou funções. Mesmo assim, esta linguagem está muito vinculada à plataforma .NET, da qual obtém as suas classes ou funções de execução. O código é organizado num conjunto de espaços de nomes que agrupam as classes com funções semelhantes. Por exemplo, System.Windows.Forms contém o sistema Windows Forms; System.Console é usado para entrada/saída de dados.[53]

Um nível de organização superior é fornecido pelo conceito de montador, que pode ser um simples arquivo ou múltiplos arquivos ligados juntos que podem conter muitos espaços de nomes ou objetos. Programas que precisam de classes para realizar uma função em particular podem se referenciar aos montadores como System.Drawing.dll e System.Windows.Forms.dll assim como a biblioteca core (conhecida como mscorlib.dll na implementação da Microsoft).[53]

Coleções são grupos de objetos de tipos semelhantes. O gerenciamento da quantidade de objetos que as compõem é feito internamente nas coleções; por sua vez, quando usamos um array o seu tamanho precisa ser gerenciado pelo programador. No namespace Collections.Generic (não use o System.Collections, é obsoleto), encontramos uma série de coleções predefinidas a partir das quais podemos criar nossas próprias coleções. Além disso, neste namespace podemos encontrar a implementação de estruturas de dados como pilhas, listas, listas ordenadas, filas, e todas são implementadas usando coleções. Algumas classes membros:[54]

  • List: usa um array cujo tamanho é incrementado dinamicamente conforme necessário;[54]
  • CollectionBase: usada como classe para implementar coleções fortemente tipadas;[54]
  • Queue: coleção de objetos que implementa uma fila do tipo “primeiro a entrar, primeiro a sair”;[54]
  • SortedList: coleção que implementa uma lista ordenada;[54]
  • Stack: coleção de objetos que implementa uma pilha (último a entrar, primeiro a sair);[54]
  • Dictionary: coleção com chave e valor sem ordem ou classificação.[54]

Padronização e licenciamento

[editar | editar código-fonte]

Em agosto de 2001, a Microsoft Corporation, a Hewlett-Packard e a Intel Corporation copatrocinaram o envio de especificações para o C#, bem como a Common Language Infrastructure (CLI) para a organização de padrões Ecma International. Em dezembro de 2001, a ECMA lançou ECMA-334 C# Language Specification. C# tornou-se um padrão ISO em 2003 (ISO/IEC 23270:2003 - Information technology — Programming languages — C#). A ECMA já havia adotado especificações equivalentes como a 2ª edição do C#, em dezembro de 2002.[55]

Em junho de 2005, a ECMA aprovou a edição 3 da especificação C# e atualizou o ECMA-334. As adições incluíram classes parciais, métodos anônimos, tipos anuláveis e genéricos (algo semelhante aos modelos C++).[56]

A definição de linguagem C# e o CLI são padronizados sob os padrões ISO e ECMA que fornecem proteção de licenciamento razoável e não discriminatória de reivindicações de patentes.[57]

Implementações

[editar | editar código-fonte]

O compilador de referência C# é Microsoft Visual C#, que é de código aberto.[58]

A Microsoft está liderando o desenvolvimento de um novo compilador open-source C# e um conjunto de ferramentas, anteriormente codinome "Roslyn". O compilador, que é inteiramente escrito em código gerenciado (C#), foi aberto e funcionalidade surgiu como APIs. Isso permite aos desenvolvedores criar ferramentas de refatoração e diagnóstico.[59]

Também há o desenvolvimento do .Net Core que é um projeto aberto e multiplataforma.[59]

Existem outros compiladores C#, incluindo uma implementação da Common Language Infrastructure e as bibliotecas de classe .NET até .NET 2.0:

  • O projeto Mono fornece um compilador C# de código aberto, uma implementação completa de código aberto da Common Language Infrastructure, incluindo as bibliotecas de estruturas necessárias, conforme elas aparecem na especificação ECMA, e uma implementação quase completa das bibliotecas de classe .NET proprietárias da Microsoft até .NET 3.5. A partir do Mono 2.6, não existem planos para implementar o WPF; A WF está planejada para uma versão posterior; E há apenas implementações parciais do LINQ to SQL e WCF.
  • O projeto DotGNU (agora descontinuado) também forneceu um compilador C# de código aberto, uma implementação quase completa da Common Language Infrastructure, incluindo as bibliotecas de estruturas necessárias conforme elas aparecem na especificação ECMA e um subconjunto de algumas classes proprietárias do .NET da Microsoft até .NET 2.0 (aquelas não documentadas ou incluídas na especificação ECMA, mas incluídas na distribuição padrão do Microsoft .NET Framework).
  • O projeto Rotor da Microsoft (atualmente chamado Shared Source Common Language Infrastructure) (licenciado somente para uso educacional e de pesquisa) fornece uma implementação de origem compartilhada do CLR runtime e um compilador C # e um subconjunto das bibliotecas de estrutura Common Language Infrastructure requeridas na especificação ECMA (Até C# 2.0, e suportado apenas no Windows XP).

Referências

  1. a b Dollard, Kathleen (14 de novembro de 2023). «Announcing C# 12». devblogs.microsoft.com (em inglês). Consultado em 1 de maio de 2024 
  2. «Why Microsoft's C# isn't» (em inglês). Consultado em 26 de agosto de 2010 
  3. «The A-Z of Programming Languages: C#» (em inglês). Consultado em 26 de agosto de 2010 
  4. «Java 5 catches up with C#» (em inglês). Consultado em 26 de agosto de 2010 
  5. a b c d e f g h i j «C# Language Specification» (PDF) (em inglês). Consultado em 26 de janeiro de 2018 
  6. a b c d e "C# - Como Programar", H. M. Deitel, São Paulo: Pearson Education, 2003
  7. «C#/.NET History Lesson» (em inglês). Consultado em 24 de agosto de 2011 
  8. «Microsoft C# FAQ» (em inglês). Consultado em 24 de agosto de 2011 
  9. «Visual C#.net Standard» (em inglês). Consultado em 24 de agosto de 2011 
  10. a b c d e f g h i j k l m n o p «Especificação da Linguagem CSharp» 
  11. a b c d e f g h i j k l «What's New in the C# 2.0 Language and Compiler». msdn.microsoft.com. Consultado em 26 de novembro de 2016 
  12. a b c d e f g h i j «Overview of C# 3.0». msdn.microsoft.com. Consultado em 26 de novembro de 2016 
  13. a b c d e «C# 4.0 - New C# Features in the .NET Framework 4». msdn.microsoft.com. Consultado em 26 de novembro de 2016 
  14. a b c «An Introduction to New Features in C# 5.0». The Microsoft MVP Award Program Blog 
  15. a b c d e f g h i j k l «dotnet/roslyn». GitHub. Consultado em 26 de novembro de 2016 
  16. a b c d e f g h i j k l «dotnet/roslyn». GitHub. Consultado em 26 de novembro de 2016 
  17. a b c d BillWagner. «Novidades no C# 7.1». docs.microsoft.com. Consultado em 11 de março de 2019 
  18. a b c d e f BillWagner. «Novidades no C# 7.2». docs.microsoft.com. Consultado em 11 de março de 2019 
  19. «What's new in C# 7.3». Consultado em 24 de junho de 2020 
  20. «C# 9.0 on the record». .NET Blog (em inglês). 10 de novembro de 2020. Consultado em 30 de dezembro de 2021 
  21. a b c d e «Welcome to C# 10». .NET Blog (em inglês). 8 de novembro de 2021. Consultado em 30 de dezembro de 2021 
  22. a b c d e f g h i j k l m n o p q r Visual Studio 2010 and .NET 4 Six-in-One. [S.l.]: Wrox Press. 2010. ISBN 978-0470499481 
  23. "C# e .NET - Guia do Desenvolvedor", Edwin Lima, Eugênio Reis, Rio de Janeiro: Campus, 2002
  24. a b c d e f g h i j k Object-oriented Programming in C# for C and Java programmers. Kurt Nørmark, 2011. http://people.cs.aau.dk/~normark/oop-csharp/html/notes/theme-index.html
  25. Wagner, Bill. «Anonymous functions - C# Programming Guide» (em inglês). Consultado em 15 de maio de 2021 
  26. «Book sources». Wikipedia (em inglês) 
  27. a b c d e f g h i j k l m n o p q r s t u v Archer, Tom. «Part 2, Chapter 4: The Type System». Inside C#. Redmond, Washington: Microsoft Press. ISBN 0-7356-1288-9 
  28. «Tipo de dado». Wikipédia, a enciclopédia livre. 16 de agosto de 2016 
  29. http://www.microsoft.com/downloads/details.aspx?displaylang=en&FamilyID=dfbf523c-f98c-4804-afbd-459e846b268e CSharp Language Specification C# 3.0 (em inglês) Página visitada em 29 de Março de 2013
  30. http://msdn.microsoft.com/pt-br/library/vstudio/ms228593 CSharp Language Specification C# 5.0 (em inglês) Página visitada em 30 de Março de 2013
  31. a b c d e f g «Technical committees - JTC 1/SC 22 - Programming languages, their environments and system software interfaces». ISO. Consultado em 4 de outubro de 2012 
  32. http://msdn.microsoft.com/pt-br/library/vstudio/618ayhy6 Referência de C# Página visitada em 30 de Março de 2013
  33. a b c d http://msdn.microsoft.com/pt-br/library/vstudio/900fyy8e.aspx 30/03/13
  34. «Tipos de enumeração». Microsoft. Consultado em 18 de março de 2022 
  35. a b c Curso online Bradesco - C#, http://www.ev.org.br/Paginas/Home.aspx, página visitada em 30/03/13
  36. a b c d e «Passando parâmetros». Microsoft. Consultado em 18 de março de 2022 
  37. a b c d «Propriedades». Microsoft. Consultado em 18 de março de 2022 
  38. http://msdn.microsoft.com/pt-br/library/vstudio/67ef8sbd Página visitada em 30 de Março de 2013
  39. http://download.microsoft.com/download/a/9/e/a9e229b9-fee5-4c3e-8476-917dee385062/CSharp%20Language%20Specification%20v1.0.doc CSharp Language Specification C# 1.0 (em inglês) Página visitada em 29 de Março de 2013
  40. http://download.microsoft.com/download/5/e/5/5e58be0a-b02b-41ac-a4a3-7a22286214ff/csharp%20language%20specification%20v1.2.doc CSharp Language Specification C# 1.2 (em inglês) Página visitada em 30 de Março de 2013
  41. http://download.microsoft.com/download/9/8/f/98fdf0c7-2bbd-40d3-9fd1-5a4159fa8044/csharp%202.0%20specification_sept_2005.doc CSharp Language Specification C# 2.0 (em inglês) Página visitada em 30 de Março de 2013
  42. http://www.microsoft.com/en-us/d ownload/details.aspx?id=7029 CSharp Language Specification C# 4.0 (em Inglês) Página visitada em 29 de Março de 2013
  43. a b c d e «Declarar namespaces para organizar tipos». Microsoft. Consultado em 18 de março de 2022 
  44. a b c «Iteradores (C#)». Microsoft. Consultado em 18 de março de 2022 
  45. a b c d e f g «Instruções (Guia de Programação em C#)». Microsoft. Consultado em 18 de março de 2022 
  46. a b c d e f g h «Exceções e manipulação de exceções». Microsoft. Consultado em 18 de março de 2022 
  47. a b «Noções básicas da coleta de lixo». Microsoft. Consultado em 18 de março de 2022 
  48. «Código não seguro, tipos de ponteiro e ponteiros de função». Microsoft. Consultado em 18 de março de 2022 
  49. «Book sources». Wikipedia (em inglês) 
  50. «Compiling Apps with .NET Native». msdn.microsoft.com. Consultado em 26 de novembro de 2016 
  51. a b «Main() e argumentos de linha de comando». Microsoft. Consultado em 18 de março de 2022 
  52. «Design Goals of C#». www.java-samples.com 
  53. a b «Framework libraries». Microsoft. Consultado em 18 de março de 2022 
  54. a b c d e f g «Coleções (C#)». Microsoft. Consultado em 18 de março de 2022 
  55. «Open Specification Promise». Microsoft. Consultado em 18 de março de 2022 
  56. «C# Language Specification» (PDF). ECMA International. Consultado em 18 de março de 2022 
  57. «Form For The ECMA Patent Declaration» (PDF). ECMA International. Consultado em 18 de março de 2022 
  58. «Repositório» 
  59. a b «The RyuJIT transition is complete!». Microsoft. Consultado em 18 de março de 2022 

Ligações externas

[editar | editar código-fonte]