Compartilhar via


Como criar e gerenciar Q# projetos e bibliotecas personalizadas

Neste artigo, você aprenderá a criar, gerenciar e compartilhar Q# projetos. Q# Os projetos são estruturas de pastas com vários Q# arquivos que podem acessar as operações e funções uns dos outros. Os projetos são úteis para organizar logicamente seu código-fonte. Você também pode usar projetos como bibliotecas personalizadas que podem ser acessadas de fontes externas.

Pré-requisitos

  • Um espaço de trabalho do Azure Quantum em sua assinatura do Azure. Para criar um espaço de trabalho, consulte Criar um espaço de trabalho no Azure Quantum.
  • Visual Studio Code com a extensão do Azure Quantum Development Kit e do Python instalada.
  • Uma conta do GitHub, se você estiver planejando publicar seu projeto externo em um repositório público do GitHub.

Para executar programas Python, você também precisa de:

  • Um ambiente Python com Python e Pip instalados.
  • O Azure Quantum qsharp e azure-quantum os pacotes.

Como Q# funcionam os projetos

Um Q# projeto contém um Q# arquivo de manifesto, chamado qsharp.json, e um ou mais arquivos *.qs em uma estrutura de pastas especificada. Quando um usuário abre um arquivo *.qs no VS Code ou define o project_root em um arquivo Jupyter Notebook ou Python, o compilador pesquisa a hierarquia de pastas ao redor do arquivo de manifesto e determina o escopo do projeto. Se nenhum arquivo de manifesto for encontrado, o compilador operará em um modo de arquivo único. Você pode criar um Q# projeto manualmente ou diretamente no VS Code.

Um projeto externo Q# é um projeto padrão Q# que reside em outro diretório ou em um repositório público do GitHub e atua como uma biblioteca personalizada. Um projeto externo usa export instruções para definir quais funções e operações podem ser acessadas por programas externos. Os programas definem o projeto externo como uma dependência em seu arquivo de manifesto e usam import instruções para acessar os itens (operações, funções, structs e namespaces) no projeto externo. Para obter mais informações, consulte Usando projetos como dependências externas.

Definir um Q# projeto

Um Q# projeto é definido pela presença de um arquivo de manifesto, chamado qsharp.json, e uma pasta src (que contém os Q# arquivos de origem), ambos os quais devem estar na pasta raiz do projeto. Para Q# programas e projetos externos, o Q# compilador detecta a pasta do projeto automaticamente. Para programas Python e Jupyter Notebooks, você deve Q# do projeto com uma qsharp.init chamada. A estrutura de pastas de um Q# projeto, no entanto, permanece a mesma para todos os tipos de programas.

A estrutura e a hierarquia de pastas de um Q# projeto.

Definindo a pasta do projeto (Q# programas)

Quando um arquivo *.qs é aberto no VS Code, o compilador pesquisa um arquivo de manifesto Q# para cima na estrutura de pastas. Se encontrar um arquivo de manifesto, o compilador incluirá todos os Q# arquivos no diretório /src ou em qualquer um de seus subdiretórios. Os itens de cada arquivo são disponibilizados para todos os outros arquivos dentro do projeto.

Por exemplo, dada esta estrutura de pastas:

  • Teleportation_project
    • qsharp.json
    • src
      • Main.qs
      • TeleportOperações
        • TeleportLib.qs
        • PrepareState
          • PrepareStateLib.qs

quando você abre o arquivo /src/TeleportOperation/PrepareState/PrepareStateLib.qs, o Q# compilador:

  1. Verifica /src/TeleportOperation/PrepareState/ para qsharp.json.
  2. Verifica /src/TeleportOperation para qsharp.json.
  3. Verifica /src para qsharp.json.
  4. Verifica / se há qsharp.json.
  5. Estabelece / como o diretório raiz do projeto e inclui todos os arquivos *.qs na raiz do projeto, de acordo com as configurações do arquivo de manifesto.

Criar um arquivo de manifesto

Um arquivo de manifesto é um arquivo .json simples chamado qsharp.json que pode incluir opcionalmente os campos autor, licença e lints . O arquivo de manifesto mínimo viável é a cadeia de caracteres {}. Quando você cria um Q# projeto no VS Code, um arquivo de manifesto mínimo é criado para você.

{}

Exemplos de arquivo de manifesto

Veja a seguir alguns exemplos de como os arquivos de manifesto podem definir o escopo do seu Q# projeto.

  • Neste exemplo, author é o único campo especificado e, portanto, todos os arquivos *.qs neste diretório e todos os seus subdiretórios são incluídos no Q# projeto.

    {
        "author":"Microsoft",
        "license": "MIT"
    }
    
  • Em um Q# projeto, você também pode usar o arquivo de manifesto para ajustar as configurações do VS Code Q# Linter. Por padrão, as três regras do Linter são:

    • needlessParens: padrão = allow

    • divisionByZero: padrão = warn

    • redundantSemicolons: padrão = warn

      Usando o arquivo de manifesto, você pode definir cada regra como allow, warn, ou error, por exemplo

      {
          "author":"Microsoft",
          "lints": [
              {
                "lint": "needlessParens",
                "level": "allow"
              },
              {
                "lint": "redundantSemicolons",
                "level": "warn"
              },
              {
                "lint": "divisionByZero",
                "level": "error"
              }
            ]
      }
      
  • Você também pode usar o arquivo de manifesto para definir um projeto externo Q# como uma dependência e acessar remotamente operações e funções nesse projeto externo. Para obter mais informações, consulte Usando projetos como dependências externas.

Q# Requisitos e propriedades do projeto

Os requisitos e configurações a seguir se aplicam a todos os Q# projetos.

  • Todos os arquivos *.qs que você deseja incluir no projeto devem estar em uma pasta chamada src, que deve estar na pasta raiz do Q#. Quando você cria um Q# projeto no VS Code, a /src pasta é criada automaticamente.

  • O arquivo de manifesto deve estar no mesmo nível da pasta src . Quando você cria um Q# projeto no VS Code, um arquivo mínimo é criado automaticamente.

  • Use import instruções para fazer referência a operações e funções de outros arquivos no projeto.

    import MyMathLib.*;  //imports all the callables in the MyMathLib namespace
    ...
        Multiply(x,y);
    

    ou referenciá-los individualmente com o namespace

    MyMathLib.Multiply(x,y);  
    

Apenas para Q# projetos

  • Apenas um arquivo *.qs em um Q# projeto pode ter um ponto de entrada definido, definido por uma única Main() operação.
  • O arquivo *.qs com a definição do ponto de entrada pode ser localizado em qualquer nível abaixo do arquivo de manifesto.
  • Qualquer operação ou função armazenada em cache de um arquivo *.qs em qualquer lugar do Q# projeto é exibida em texto preditivo no VS Code.
  • Se o namespace de uma operação ou função selecionada ainda não tiver sido importado, o VS Code adicionará automaticamente a instrução necessária import .

Etapas para criar um Q# projeto

Essas etapas se aplicam a todos os Q# projetos.

  1. No explorador de arquivos do VS Code, clique com o botão direito do mouse na pasta que deseja usar para a pasta raiz do projeto e selecione Criar projeto ou abra a pasta e selecione Q#.Q#

  2. O VS Code cria um arquivo de manifesto mínimo na pasta e adiciona uma /src pasta com um arquivo de Main.qs modelo.

  3. Edite o arquivo de manifesto conforme necessário. Consulte Exemplos de arquivo de manifesto.

  4. Adicione e organize seus Q# arquivos de origem na /src pasta.

  5. Se você estiver acessando o Q# projeto de um programa Python ou Jupyter Notebook, defina o caminho da pasta raiz usando qsharp.init. Este exemplo pressupõe que seu programa esteja na pasta /src do Q# projeto:

    qsharp.init(project_root = '../Teleportation_project')
    
  6. Se você estiver usando apenas Q# arquivos no VS Code, ao abrir um Q# arquivo, o compilador procurará um arquivo de manifesto, determinará a pasta raiz do projeto e verificará a subpasta em busca de arquivos *.qs.

Observação

Você também pode criar manualmente o arquivo de manifesto e a /src pasta na etapa 2.

Projeto de exemplo

Esse programa de teletransporte quântico é um exemplo de um Q# projeto baseado na estrutura de pasta única mostrada anteriormente e é executado no simulador local no VS Code. Para executar o programa em simuladores de hardware ou de terceiros do Azure Quantum, consulte Introdução aos programas Q# e ao VSCode para obter etapas para compilar seu programa e se conectar ao workspace do Azure.

O exemplo usa esta estrutura de diretório:

  • Teleportation_project
    • qsharp.json
    • src
      • Main.qs
      • TeleportOperações
        • TeleportLib.qs
        • PrepareState
          • PrepareStateLib.qs

O arquivo de manifesto contém os campos author e license :

{
    "author":"Microsoft",
    "license":"MIT"
}

Q# Arquivos de origem

O arquivo principal, Main.qs, contém o ponto de entrada e faz referência ao TeleportOperations.TeleportLib namespace de TeleportLib.qs.


    import TeleportOperations.TeleportLib.Teleport;   // references the Teleport operation from TeleportLib.qs

    operation Main() : Unit {
        use msg = Qubit();
        use target = Qubit();

        H(msg);
        Teleport(msg, target);    // calls the Teleport() operation from TeleportLib.qs
        H(target);

        if M(target) == Zero {
            Message("Teleported successfully!");
        
        Reset(msg);
        Reset(target);
        }
    }

TeleportLib.qs define a Teleport() operação e chama a PrepareBellPair() operação de PrepareStateLib.qs.


    import TeleportOperations.PrepareState.PrepareStateLib.*;     // references the namespace in PrepareStateLib.qs
 
    operation Teleport(msg : Qubit, target : Qubit) : Unit {
        use here = Qubit();

        PrepareBellPair(here, target);      // calls the PrepareBellPair() operation from PrepareStateLib.qs
        Adjoint PrepareBellPair(msg, here);

        if M(msg) == One { Z(target); }
        if M(here) == One { X(target); }

        Reset(here);
    }

O arquivo PrepareStateLib.qs contém uma operação reutilizável padrão para criar um par de Bell.

    
    operation PrepareBellPair(left : Qubit, right : Qubit) : Unit is Adj + Ctl {
        H(left);
        CNOT(left, right);
    }

Executando os programas

Selecione a guia do ambiente no qual você está executando o programa.

Para executar esse programa, abra o arquivo Main.qs no VS Code e selecione Executar.

Configurando Q# projetos como dependências externas

Um Q# projeto também pode ser configurado como uma dependência externa para outros projetos, agindo como uma biblioteca, onde as funções e operações no projeto externo Q# são disponibilizadas para vários Q# projetos. Uma dependência externa pode residir em um compartilhamento de unidade ou publicada em um repositório público do GitHub.

Para usar um Q# projeto como uma dependência externa, você precisa:

  • Adicione o projeto externo como uma dependência no arquivo de manifesto do projeto de chamada.
  • Se o projeto externo for publicado no GitHub, adicione a propriedade "files" ao arquivo de manifesto do projeto externo.
  • Adicione export instruções ao projeto externo.
  • Adicione import instruções ao projeto de chamada.

Configurando os arquivos de manifesto

Os projetos externos Q# podem residir em um compartilhamento de unidade local ou de rede ou publicados em um repositório público do GitHub.

O arquivo de manifesto do projeto de chamada

Para adicionar uma dependência a um projeto externo em um compartilhamento de unidade, defina a dependência no arquivo de manifesto do projeto de chamada.

{
    "author": "Microsoft",
    "license": "MIT",
    "dependencies": {
        "MyDependency": {
            "path": "/path/to/project/folder/on/disk"
        }
    }
}

em que "MyDependency" é uma cadeia de caracteres definida pelo usuário que identifica o namespace ao chamar uma operação. Por exemplo, se você criar uma dependência chamada "MyMathFunctions", chamará uma função dessa dependência com MyMathFunctions.MyFunction().

Para adicionar uma dependência a um projeto publicado em um repositório público do GitHub

{
    "author": "Microsoft",
    "dependencies": {
        "MyDependency": {
            "github": {
                "owner": "GitHubUser",
                "repo": "GitHubRepoName",
                "ref": "CommitHash",
                "path": "/path/to/dependency"
            }
        }
}

Observação

Para dependências do GitHub, "ref" refere-se a um refspec do GitHub. A Microsoft recomenda sempre usar um hash de confirmação, para que você possa contar com uma versão específica de sua dependência.

O arquivo de manifesto do projeto externo

Se o projeto externo Q# for publicado em um repositório público do GitHub, você deverá adicionar a propriedade files ao arquivo de manifesto do projeto externo, incluindo todos os arquivos usados no projeto.

{
    "author": "Microsoft",
    "license": "MIT",
    "files": [ "src/MyMathFunctions.qs", "src/Strings/MyStringFunctions.qs" ]
}

A propriedade "files" é opcional para um projeto externo que está sendo importado via "path" (ou seja, uma importação baseada em caminho de arquivo local). Ele só é necessário para projetos publicados no GitHub.

Usando a instrução export

Para tornar as funções e operações em um projeto externo acessíveis a projetos de chamada, use a export instrução. Você pode exportar qualquer um ou todos os chamáveis no arquivo. Não há suporte para a sintaxe curinga, você deve especificar cada chamável para exportar.

operation Operation_A() : Unit {
...
}
operation Operation_B() : Unit  {
...
}

// makes just Operation_A available to calling programs
export Operation_A;           

// makes Operation_A and Operation_B available to calling programs         
export Operation_A, Operation_B, etc.; 

// makes Operation_A available as 'OpA'
export Operation_A as OpA;             

Usando a instrução import

No programa de chamada, você usa import instruções para disponibilizar itens de uma dependência externa. import As instruções usam o namespace definido para a dependência no arquivo de manifesto. Por exemplo, para essa dependência

{
    "author": "Microsoft",
    "license": "MIT",
    "dependencies": {
        "MyMathFunctions": {
            "path": "/path/to/project/folder/on/disk"
        }
    }
}

Você importa callables como

import MyMathFunctions.MyFunction;  // imports "MyFunction()" from the namespace
...

A import instrução também dá suporte à sintaxe e aliases curinga

// imports all items from the "MyMathFunctions" namespace
import MyMathFunctions.*;        

// imports the namespace as "Math", all items are accessible via "Math.<callable>"
import MyMathFunctions as Math;   

// imports a single item, available in the local scope as "Add"
import MyMathFunctions.MyFunction as Add;        

// imports can be combined on one line
import MyMathFunctions.MyFunction, MyMathFunctions.AnotherFunction as Multiply; 

Observação

A instrução usada open atualmente em Q#, que é usada para fazer referência a bibliotecas e namespaces, ainda é suportada, mas será descontinuada eventualmente. Enquanto isso, você pode atualizar opcionalmente seus arquivos atuais para usar a import instrução. Por exemplo, open Microsoft.Quantum.Diagnostics; pode ser substituído por import Microsoft.Quantum.Diagnostics.*;. Observe também que, ao usar a import instrução com as bibliotecas padrão Q# , você pode encurtar o namespace raiz para Std. Por exemplo, import Microsoft.Quantum.Diagnostics.*; pode ser escrito como import Std.Diagnostics.*;.

Exemplo de projeto externo

Para este exemplo, você usará o mesmo programa de teletransporte do exemplo anterior, mas separará o programa de chamada e os chamáveis em projetos diferentes.

  1. Crie duas pastas em sua unidade local, por exemplo, "Project_A" e "Project_B".

  2. Crie um Q# projeto em cada pasta seguindo as etapas em Etapas para criar um Q# projeto.

  3. Em Project_A, o programa de chamada, copie o código a seguir para o arquivo de manifesto, editando o caminho conforme necessário para Project_B

    {
      "author": "Microsoft",
      "license": "MIT",
      "dependencies": {
        "MyTeleportLib": {
          "path": "/Project_B" 
          }
        }
      }    
    
  4. Em Project_A, copie o código a seguir para Main.qs

    import MyTeleportLib.Teleport;   // imports the Teleport operation from the MyTeleportLib namespace defined in the manifest file
    
    operation Main() : Unit {
        use msg = Qubit();
        use target = Qubit();
    
        H(msg);
        Teleport(msg, target);    // calls the Teleport() operation from the MyTeleportLib namespace
        H(target);
    
        if M(target) == Zero {
            Message("Teleported successfully!");
    
        Reset(msg);
        Reset(target);
        }
    }   
    
  5. Em Project_B, copie o código a seguir em Main.qs

    
        operation Teleport(msg : Qubit, target : Qubit) : Unit {
            use here = Qubit();
    
            PrepareBellPair(here, target); 
            Adjoint PrepareBellPair(msg, here);
    
            if M(msg) == One { Z(target); }
            if M(here) == One { X(target); }
    
            Reset(here);
        }
    
        operation PrepareBellPair(left : Qubit, right : Qubit) : Unit is Adj + Ctl {
            H(left);
            CNOT(left, right);
        }
    
        export Teleport;       //  makes the Teleport operation available to external programs
    

    Observação

    Observe que a PrepareBellPair operação não precisa ser exportada porque não é chamada diretamente do seu programa no Project_A. Como ele está no escopo local do Project_B, ele já está acessível pela operação de Teleport

  6. Para executar o programa, abra /Project_A/Main.qs no VS Code e selecione Executar.

Projetos e namespaces implícitos

Em Q# projetos, se um namespace não for especificado em um programa *.qs, o compilador usará o nome do arquivo como namespace. A referência a um chamável de uma dependência externa usa a sintaxe <dependencyName>.<namespace>.<chamável>. No entanto, se o arquivo for nomeado "Main.qs", o compilador assumirá o namespace e a sintaxe de chamada será <dependencyName>.<chamável>, como no exemplo anterior, import MyTeleportLib.Teleport.

Como não é incomum ter vários arquivos de projeto, você precisa levar em conta a sintaxe correta ao fazer referência a callables. Por exemplo, em um projeto com a seguinte estrutura de arquivos

  • src/
    • Main.qs
    • MathFunctions.qs

chamadas para a dependência externa seriam

import MyTeleportLib.MyFunction;        // "Main" namespace is implied

import MyTeleportLib.MathFunctions.MyFunction;   // "Math" namespace must be explicit 

Para obter mais informações sobre o comportamento do namespace, consulte Namespaces do usuário.