OpenXML и c # ошибка для вставки данных в SQL Server - PullRequest
0 голосов
/ 30 мая 2018

У меня есть приложение C #, которое служит для восстановления данных из файла .XLSX и вставки данных в таблицу сервера SQL.

Например, у меня есть данные с идентификатором и 3 столбцами.Когда я выполняю команду SQL.У меня есть эта ошибка:

ВСТАВИТЬ в столбец идентификации не допускается для переменных таблицы.Данные для табличного параметра "@product" не соответствуют табличному типу параметра.Ошибка SQL Server: 200, состояние: 7

Хранимая процедура

ALTER PROCEDURE [dbo].[InserimentoDatiExcel](@product [udtInserimentoDatiExcel] readonly)

AS
BEGIN
    INSERT INTO Prova
    (ID, PIPPO, PLUTO, PAPERINO)
    SELECT *
    FROM @product
    --FROM Prova
    --WHERE ID IN (SELECT ID FROM @product)

END

Таблица

CREATE TYPE [dbo].[udtInserimentoDatiExcel] AS TABLE(
    [ID] [int] IDENTITY(1,1) NOT NULL,
    [PIPPO] [varchar](1) NOT NULL,
    [PLUTO] [varchar](1) NOT NULL,
    [PAPERINO] [varchar](1) NOT NULL,
    PRIMARY KEY CLUSTERED 
(
    [ID] ASC
)WITH (IGNORE_DUP_KEY = ON)
)
GO

Datatable : Изображение

Код

using System;
using System.Collections.Generic;
using System.Data;
using System.Data.SqlClient;
using System.Linq;
using System.Text;
using System.Text.RegularExpressions;
using System.Threading.Tasks;
using DocumentFormat.OpenXml;
using DocumentFormat.OpenXml.Packaging;
using DocumentFormat.OpenXml.Spreadsheet;

namespace BatchReadExcel
{
    class Program
    {
        static void Main()
        {
            try
            {
                DataTable dt = new DataTable();
                using (SpreadsheetDocument spreadSheetDocument = SpreadsheetDocument.Open(@"C:\Users\f.piano\Documents\prova.xlsx", false))
                {
                    WorkbookPart workbookPart = spreadSheetDocument.WorkbookPart;
                    IEnumerable<Sheet> sheets = spreadSheetDocument.WorkbookPart.Workbook.GetFirstChild<Sheets>().Elements<Sheet>();
                    string relationshipId = sheets.First().Id.Value;
                    WorksheetPart worksheetPart = (WorksheetPart)spreadSheetDocument.WorkbookPart.GetPartById(relationshipId);
                    Worksheet workSheet = worksheetPart.Worksheet;
                    SheetData sheetData = workSheet.GetFirstChild<SheetData>();
                    IEnumerable<Row> rows = sheetData.Descendants<Row>();
                    //TEST
                    //dt.Columns.Add("ID", typeof(Int32));
                    dt.Columns.Add("ID", typeof(Guid));
                    dt.Columns["ID"].AutoIncrement = true;
                    dt.Columns["ID"].AutoIncrementSeed = 0;
                    dt.Columns["ID"].AutoIncrementStep = 1;
                    //TEST
                    foreach (Cell cell in rows.ElementAt(0))
                    {
                        dt.Columns.Add(GetCellValue(spreadSheetDocument, cell));
                    }
                    foreach (Row row in rows) //this will also include your header row...
                    {
                        DataRow tempRow = dt.NewRow();
                        /*Si inserisce columnIndex = 1 quando devi utilizzare
                          l'autoincrement sulla prima colonna (in questo caso ID)*/
                        int columnIndex = 1;
                        foreach (Cell cell in row.Descendants<Cell>())
                        {
                            // Gets the column index of the cell with data
                            int cellColumnIndex = (int)GetColumnIndexFromName(GetColumnName(cell.CellReference));
                            //cellColumnIndex--; //zero based index
                            if (columnIndex < cellColumnIndex)
                            {
                                do
                                {
                                    tempRow[columnIndex] = 1;
                                    columnIndex++;
                                }
                                while (columnIndex < cellColumnIndex);
                            }

                            tempRow[columnIndex] = GetCellValue(spreadSheetDocument, cell);

                            columnIndex++;
                        }
                        dt.Rows.Add(tempRow);
                    }
                }
                dt.Rows.RemoveAt(0); //rimuove la prima riga in cui ci sono le colonne

                //leggo le righe di una colonna specifica
                //string pippoRows = "";
                //for (int i = 0; i < dt.Rows.Count; i++)
                //{
                //    //pippoRows = Convert.ToInt32(dt.Rows[i]["PIPPO"]);
                //    pippoRows = Convert.ToString(dt.Rows[i]["PIPPO"]);
                //    //Similarly for QuantityInIssueUnit_uom.
                //}

                //TODO: Insert data in SQL Server
                var stringaConnessione = Properties.Settings.Default.connectionString;
                SqlConnection sqlcon = new SqlConnection();
                sqlcon.ConnectionString = stringaConnessione;
                SqlCommand sqlcmd = new SqlCommand("InserimentoDatiExcel", sqlcon);
                sqlcmd.CommandType = CommandType.StoredProcedure;

                //sqlcmd.Parameters.AddWithValue("@pippo", dt.Columns["PIPPO"].Container);
                sqlcmd.Parameters.AddWithValue("@product", dt);
                //sqlcmd.Parameters.Add(new SqlParameter("@pippo", dt.Columns["PIPPO"].ColumnMapping));
                sqlcon.Open();
                sqlcmd.ExecuteNonQuery();
                sqlcon.Close();

            }
            catch (Exception ex)
            {
            }
        }
        /// <summary>
        /// Given a cell name, parses the specified cell to get the column name.
        /// </summary>
        /// <param name="cellReference">Address of the cell (ie. B2)</param>
        /// <returns>Column Name (ie. B)</returns>
        public static string GetColumnName(string cellReference)
        {
            // Create a regular expression to match the column name portion of the cell name.
            Regex regex = new Regex("[A-Za-z]+");
            Match match = regex.Match(cellReference);
            return match.Value;
        }
        /// <summary>
        /// Given just the column name (no row index), it will return the zero based column index.
        /// Note: This method will only handle columns with a length of up to two (ie. A to Z and AA to ZZ). 
        /// A length of three can be implemented when needed.
        /// </summary>
        /// <param name="columnName">Column Name (ie. A or AB)</param>
        /// <returns>Zero based index if the conversion was successful; otherwise null</returns>
        public static int? GetColumnIndexFromName(string columnName)
        {

            //return columnIndex;
            string name = columnName;
            int number = 0;
            int pow = 1;
            for (int i = name.Length - 1; i >= 0; i--)
            {
                number += (name[i] - 'A' + 1) * pow;
                pow *= 26;
            }
            return number;
        }
        public static string GetCellValue(SpreadsheetDocument document, Cell cell)
        {
            SharedStringTablePart stringTablePart = document.WorkbookPart.SharedStringTablePart;
            if (cell.CellValue == null)
            {
                return "";
            }
            string value = cell.CellValue.InnerXml;
            if (cell.DataType != null && cell.DataType.Value == CellValues.SharedString)
            {
                return stringTablePart.SharedStringTable.ChildElements[Int32.Parse(value)].InnerText;
            }
            else
            {
                return value;
            }
        }
    }
}

1 Ответ

0 голосов
/ 30 мая 2018

В своем первом фрагменте кода вы вставляете столбец идентификатора, который является столбцом идентификаторов (не может быть вставлен).

Вам необходимо обновить код, чтобы НЕ включать столбец идентификатора.

...