SQLの窓

2017年09月28日


Microsoft Access に対してSQLを入力してその結果を DataGridView に表示する最も簡単なコード



最初に .accdb か .mdb を選択しておいて、自由な SQL をタイプしてその結果をそのまま Excel のごとく DataGridView に表示します。

DataGridView に表示するデータは全て文字列扱いとして、ArrayList に随時行を追加して、Rows.Add に、ToArray() したものをセットします。

一応表示のみを想定しているので、以下の設定を最初にします


表示件数は多すぎると時間がかかりすきるので、固定で 100 件にしています。
using System;
using System.Collections;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Data.OleDb;
using System.Diagnostics;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace WinApp1 {
	public partial class Form1 : Form {

		private OpenFileDialog ofd = new OpenFileDialog();

		public Form1() {
			InitializeComponent();
		}

		private void button1_Click(object sender, EventArgs e) {

			using (OleDbConnection myCon = new OleDbConnection())
			using (OleDbCommand myCommand = new OleDbCommand()) {

				// *********************
				// 接続文字列の作成
				// *********************
				myCon.ConnectionString = string.Format("Provider=Microsoft.ACE.OLEDB.12.0;Data Source={0};", ofd.FileName);
				Debug.WriteLine(myCon.ConnectionString);

				// *********************
				// 接続
				// *********************
				try {
					// 接続文字列を使用して接続
					myCon.Open();
					// コマンドオブジェクトに接続をセット
					myCommand.Connection = myCon;
					// コマンドを通常 SQL用に変更
					myCommand.CommandType = System.Data.CommandType.Text;
				}
				catch (Exception ex) {
					MessageBox.Show(this,ex.Message);
					return;
				}

				// *********************
				// 実行 SQL
				// *********************
				myCommand.CommandText = textBox1.Text;

				// *********************
				// レコードセット取得
				// *********************
				try {
					using (OleDbDataReader myReader = myCommand.ExecuteReader()) {

						// *********************
						// 列数
						// *********************
						int nCols = myReader.FieldCount;

						// DataGridView 初期化
						dataGridView1.Columns.Clear();
						dataGridView1.ColumnCount = nCols;		// カラム数

						Type fldType;
						ArrayList rowData = new ArrayList();

						// データ取得と表示
						int maxCount = 0;
						while (myReader.Read()) {

							// 行データをセットする ArrayList をクリア
							rowData.Clear();

							maxCount++;
							if ( maxCount > 100 ) {
								break;
							}

							// 初回のみ、タイトル文字列を設定
							if (maxCount == 1) {
								for (int idx = 0; idx < nCols; idx++) {
									dataGridView1.Columns[idx].Name = myReader.GetName(idx);
								}
							}

							// 行データの作成( ArrayList で作成して ToArray() したものを Rows.Add )
							for (int idx = 0; idx <nCols; idx++) {

								// NULL でない場合
								if (!myReader.IsDBNull(idx)) {
									// 列のデータ型を取得
									fldType = myReader.GetFieldType(idx);

									// 文字列
									if (fldType.Name == "String") {
										rowData.Add(myReader.GetValue(idx).ToString());
									}
									else if (fldType.Name == "Int32") {
										rowData.Add(myReader.GetInt32(idx).ToString());
									}
									else if (fldType.Name == "DateTime") {
										rowData.Add(myReader.GetDateTime(idx).ToString());
									}
									else {
										rowData.Add(myReader.GetValue(idx).ToString());
									}

								}
								else {
									rowData.Add("");
								}
							}

							// 行追加 ( BuildArray() は配列を作成する為のメソッド )
							dataGridView1.Rows.Add(rowData.ToArray());

						}

						// リーダを使い終わったので閉じる
						myReader.Close();
					}
				}
				catch (Exception ex) {
					myCon.Close();
					MessageBox.Show(this, ex.Message);
					return;
				}

				myCon.Close();

				// カラム幅の自動調整
				dataGridView1.AutoResizeColumns();

			}		// 最外の using の終わり

		}

		private void button2_Click(object sender, EventArgs e) {

			// *********************
			// ファイル選択
			// *********************
			ofd.Filter = "Excel(*.mdb;*.accdb)|*.mdb;*.accdb|すべてのファイル(*.*)|*.*";
			ofd.FilterIndex = 1;
			ofd.Title = "Access データベースを選択してください";
			ofd.RestoreDirectory = true;

			if (ofd.ShowDialog() != DialogResult.OK) {
				return;
			}


		}		// click イベントの終わり

	}		// class の終わり
}


※ テーブル・ビュー一覧は以下のようにして簡単に表示できます
			using (OleDbConnection myCon = new OleDbConnection())
			using (OleDbCommand myCommand = new OleDbCommand()) {

				// *********************
				// 接続文字列の作成
				// *********************
				myCon.ConnectionString = string.Format("Provider=Microsoft.ACE.OLEDB.12.0;Data Source={0};", ofd.FileName);
				Debug.WriteLine(myCon.ConnectionString);

				// *********************
				// 接続
				// *********************
				try {
					// 接続文字列を使用して接続
					myCon.Open();
					// コマンドオブジェクトに接続をセット
					myCommand.Connection = myCon;
					// コマンドを通常 SQL用に変更
					myCommand.CommandType = System.Data.CommandType.Text;
				}
				catch (Exception ex) {
					MessageBox.Show(this,ex.Message);
					return;
				}

				DataTable dataTable = myCon.GetOleDbSchemaTable(
					OleDbSchemaGuid.Tables,
					new object[] { null, null, null, null }
				);
				// TABLE のみの場合は以下( 上記は 全て )
				//	new object[] { null, null, null, "TABLE" }

				dataGridView1.DataSource = dataTable;

				myCon.Close();

				// カラム幅の自動調整
				dataGridView1.AutoResizeColumns();

			}		// 最外の using の終わり

さらに SQL の実行結果は、データの途中加工が全くできなくなりますが、DataTable を使って以下のようにさらに短く記述する事ができます
				// *********************
				// 実行 SQL
				// *********************
				myCommand.CommandText = textBox1.Text;

				try
				{
					using (OleDbDataReader myReader = myCommand.ExecuteReader())
					{
						DataTable dataTable = new DataTable();

						dataTable.Load(myReader);

						dataGridView1.DataSource = dataTable;
		
						myReader.Close();
					}

				}
				catch (Exception ex)
				{
					myCon.Close();
					MessageBox.Show(this, ex.Message);
					return;
				}

DataSource プロパティを使用した場合、新たに直接データをセットしていきたい場合は、いったん DataSource プロパティに null を設定しておく必要があります。



【VS(C#)の最新記事】
posted by lightbox at 2017-09-28 19:53 | Comment(0) | VS(C#) | このブログの読者になる | 更新情報をチェックする

2017年09月26日


MDB 接続から、MySQL と SQLServer を外部参照として結合して、Group By を使用して複雑な SQL を作成して Excel にエクスポートする SQL 文



※ SQLの窓では、Microsoft.Jet.OLEDB.4.0 使用しているので、最新の Access( .accdb ) への外部参照はできません。( C#、VB、VBScript、PHP で Microsoft.ACE.OLEDB.12.0 を使用すれば .accdb と下位の MDB 両方にアクセスできます  )

関連する記事

MDB 接続で、MySQL と SQLServer の表を結合表示する
System.Data.OleDb で select into 構文を使って簡単に Microsoft Access のデータを他の データベースにエクスポートする

SQL文全体

この SQL 文の最も大きな目的は、得意先毎の金額の集計です。その為のベースとなる取引先データは、売上伝票のデータで、伝票番号毎に売り上げた商品と金額が複数記録されています。

取引データには、得意先名や担当者名が無いので他のテーブルから参照する為に結合していますが、group by で指定した取引先コード以外は結果に表示出来ないので( MySQL は可能なようですが、通常は不可 )、max 関数を使用して得意先名と担当者の情報と住所を表示しています。

全て、取引先コードに対して必ず一つの値しか無いのでこのような記述で取り出す事ができます。
select 
	取引先コード,
	max(得意先名) as 得意先名,
	sum(金額) as 売上金額, 
	max(社員コード) as 社員コード, 
	max(氏名) as 担当者, 
	left(max(住所1) & max(住所2),30) as 住所

from 
	([ODBC;Driver={MySQL ODBC 5.3 Unicode Driver};server=localhost;database=lightbox;uid=root;pwd=password].取引データ left outer join [ODBC;Driver={MySQL ODBC 5.3 Unicode Driver};server=localhost;database=lightbox;uid=root;pwd=password].得意先マスタ on 取引データ.取引先コード = 得意先マスタ.得意先コード)
	left outer join [ODBC;Driver={SQL Server Native Client 11.0};server=localhost;database=lightbox;uid=sa;pwd=password].社員マスタ on 得意先マスタ.担当者 = 社員マスタ.社員コード

where
	Format(取引日付, 'yyyy/MM/dd') between '2005/06/01' and '2005/06/30'

group by 取引先コード

having
	max(得意先名) like '%商店%'




Microsoft Access ベースの記述

ここでは、SQL で使える関数は、Microsoft Access のルールに従う必要があるので、left 関数や、文字列の連結と日付との比較の為に Format 関数を使用しています。

このような SQL は、Microsoft の仕様に基づいて実行されて動作するので、from の 前に into [外部参照].表名 と書く事で、ある特定の RDBMS に対するエクスポートを容易に実行する事ができます。

ここでは、解りやすく Excel に出力する書式として以下のようになります
select 
	取引先コード,
	max(得意先名) as 得意先名,
	sum(金額) as 売上金額, 
	max(社員コード) as 社員コード, 
	max(氏名) as 担当者, 
	left(max(住所1) & max(住所2),30) as 住所

into [Excel 8.0;DATABASE=c:\tmp\集計関数結果出力.xls].取引先別売上

from 
	([ODBC;Driver={MySQL ODBC 5.3 Unicode Driver};server=localhost;database=lightbox;uid=root;pwd=password].取引データ left outer join [ODBC;Driver={MySQL ODBC 5.3 Unicode Driver};server=localhost;database=lightbox;uid=root;pwd=password].得意先マスタ on 取引データ.取引先コード = 得意先マスタ.得意先コード)
	left outer join [ODBC;Driver={SQL Server Native Client 11.0};server=localhost;database=lightbox;uid=sa;pwd=password].社員マスタ on 得意先マスタ.担当者 = 社員マスタ.社員コード

where
	Format(取引日付, 'yyyy/MM/dd') between '2005/06/01' and '2005/06/30'

group by 取引先コード

having
	max(得意先名) like '%商店%'




出力先のテーブルが既に存在している場合は、前もって削除しておいて下さい。Excel の場合は、テーブル名はシート名となり、ブックが無い場合は新規に作成されます。

SQL を MDB のビューとして登録

into によるエクスポートが無い SQL 文の場合は ビューとして登録が可能です。トンカチ + V のアイコンで、ビューの名前を入力するダイアログが表示されるので、その時に接続している MDB の中にビューが登録されて、後から容易に実行する事ができます。

SQL を 各 RDBMS のビューとして登録

例えば、MySQL で今回の SQL を記述すると以下のようになります。
select 
	取引先コード,
	得意先名,
	sum(金額) as 売上金額, 
	社員コード, 
	氏名 as 担当者, 
	left(concat(max(住所1),max(住所2)),30) as 住所

from 
	取引データ left outer join 得意先マスタ on 取引先コード = 得意先コード
	left outer join 社員マスタ on 担当者 = 社員コード

where
	取引日付 between '2005/06/01' and '2005/06/30'

group by 取引先コード

having
	得意先名 like '%商店%'

これを MySQL に ビューとして登録してしまえば、SQLの窓では『処理』メニューからエクスポートが各 RDBMS に対して容易に実行できるようになっています。


posted by lightbox at 2017-09-26 17:48 | Comment(0) | SQLの窓 Build C++ | このブログの読者になる | 更新情報をチェックする

Microsoft Access SQL では、DISTINCT 集計関数の参照がサポートされていません

Microsoft ドキュメント

Microsoft Access SQL と ANSI SQL の比較

代替として副問合せである、いわゆる『インラインビュー』を使用 2つ目と3つ目が『インラインビュー』ですが、これは Oracle では正式に使用される呼び名です。
select count(distinct 商品分類) from 商品マスタ
;
select count(商品分類) from (select distinct 商品分類 from 商品マスタ) A
;
select count(*) from (select distinct 商品分類 from 商品マスタ) A
;
select '----------','----------','----------','----------'
;
select * from コード名称マスタ

一つ目の実行が、Microsoft Access SQL で失敗します( MySQL や SQLServer等 では成功します )
Syntax error (missing operator) in query expression 'count(distinct 商品分類)'
ですから、distinct を使ったSQL 全体を括弧で囲んで、from の後ろに置いて別名を付けて実行します。 NULL を含むデータ ここでは、商品分類に NULL を含む場合があるので、集計関数を商品分類に使用した場合はカウントから除外されます。但し、行全体の数を取る count(*) では含まれます。 SQLの窓のコマンドエリアでのセミコロン act ボタンで実行した場合、セミコロンは SQL の実行単位の区切りとみなされ、全ての実行を行って下段に全て表示します MySQL SQL Express SQLの窓 Build C++ のダウンロード
タグ:MDB access DISTINCT
posted by lightbox at 2017-09-26 13:38 | Comment(0) | SQLの窓 Build C++ | このブログの読者になる | 更新情報をチェックする

MDB 接続で、MySQL と SQLServer の表を結合表示する



select

	*

from

	[ODBC;DSN=SQS32;Database=lightbox;UID=sa;PWD=password].得意先マスタ

left outer join 

	[ODBC;DSN=MYSQL32;Database=lightbox;UID=root;PWD=password].社員マスタ

on 

	得意先マスタ.担当者 = 社員マスタ.社員コード


where Format(生年月日, 'yyyy/MM/dd') = '2000/01/01'



ADO や OleDb 接続では、外部のデーターベースを容易に参照する事が可能です
関連する記事

System.Data.OleDb で select into 構文を使って簡単に Microsoft Access のデータを他の データベースにエクスポートする
SQLの窓では、同梱の販売管理C.MDB をテーブル一覧部分にドラッグ&ドロップすると容易に接続可能です。
関連する記事

SQLの窓 Build C++ : 準備不要の 2000 〜 2003 の .MDB へのアクセス
MDB への接続状態で、サンプルのような SQL をコマンドエリアに入力して、『コマンドエリアの左下の実行ボタン』をクリックして下さい。そうすると、下のエリアに結果が表示されます。 (actボタンは、セミコロンを複数のSQLの分割に使用するので、実行ボタンを使用します) 但し、この SQL は、Microsoft Access の SQL の構文で書く必要があるので、サンプルとして付加した条件部分は、Format 関数が使用されています。 また、さらに、into を付加して、他のデータベースにエクスポートする事ができます(キーはエクスポートされません) この SQL の処理は、C# ゃ VB、VBScript(ADO)、PHP(ADO) でアプリケーションとして実行可能です(SQLの窓内では ADO を使用しています) Microsoft Access の外部データ Microsoft Access では、外部データとして ODBC 経由である程度のデータベースは直接リンクする事ができます。書き方のドキュメントはありませんが、mdb 時代のシステムテーブルには、接続情報としてこのような記述が使われており、現在でもある程度ものは取得できます。(MSysObjects より) ※ SQLの窓は古いアプリケーションなので、MDB 接続ですが、.accdb 接続でも同じです SQLの窓 Build C++ のダウンロード
posted by lightbox at 2017-09-26 01:55 | Comment(0) | SQLの窓 Build C++ | このブログの読者になる | 更新情報をチェックする

2017年09月23日


ViewSwitcher を使用した2画面アプリ (5) : SQLiteデータを更新する

この内容は、ViewSwitcher を使用した2画面アプリ (4) : SQLiteデータをインポートしてリストビューに表示するより続いています。
更新部分に特化したコードにする為に、MainControl をさらに継承して、マニフェストの初期画面を新しいクラス( MainUpdate ) に変更します。 特別な部分は特に無く、更新ボタンをクリックすると更新するだけです。但し、新規と修正の二通りがあるので、textViewCode の表示状態を取得して、どちらを実行するか決定しています。
import android.database.Cursor;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.AdapterView;

public class MainUpdate extends MainControl {

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);

		listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
			@Override
			public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
				Log.i("lightbox","タップされました");

				String scode = arrayList.get(position);
				helper.getTextView(R.id.textViewCode).setText(scode);

				// キーを渡して SQLite にアクセスする
				loadData(scode);

				// 画面を切り替える
				helper.showEditView();

			}
		});

		// 更新ボタンの処理
		MainUpdate.this.findViewById(R.id.buttonUpdate).setOnClickListener(new View.OnClickListener() {
			@Override
			public void onClick(View v) {
				Log.i("lightbox","更新ボタンがクリックされました");

				//alertDialog.show();
				helper.showDialog("更新してよろしいですか?", new Helper.OnDialogOkListener() {

					@Override
					public void onDialogOkListener() {
						Log.i("lightbox","ダイアログで、OK ボタンがクリックされました");

						if (  helper.getTextView(R.id.textViewCode).getVisibility() == View.VISIBLE  ) {
							updateData();
						}
						else {
							insertData();
						}

					}

				});

			}
		});
	}

	// *******************************
	// データの更新
	// *******************************
	private void updateData() {

		Log.i("lightbox","updateData");

		String scode = helper.getTextViewString(R.id.textViewCode);
		String sname = helper.getEditTextString(R.id.editText1);
		String furi = helper.getEditTextString(R.id.editText2);

		values.clear();
		values.put("sname", sname);
		values.put("furi", furi);
		db.update("syain", values, String.format("scode='%s'", scode), null);

		loadView();
		helper.showFirstView();
	}

	// *******************************
	// データの登録
	// *******************************
	private void insertData() {

		Log.i("lightbox","insertData");

		String scode = helper.getTextViewString(R.id.editTextCode);
		String sname = helper.getTextViewString(R.id.editText1);
		String furi = helper.getTextViewString(R.id.editText2);

		values.clear();
		values.put("scode", scode);
		values.put("sname", sname);
		values.put("furi", furi);
		db.insert("syain", null, values);

		loadView();
		helper.showFirstView();

	}

	// *******************************
	// 編集用データの表示
	// *******************************
	private void loadData(String scode) {

		String sql = String.format("select * from %s where scode='%s' ", "syain", scode);
		Cursor cursor = mySQLite.getReadableDatabase().rawQuery(sql, null);
		Boolean isData = null;
		isData = cursor.moveToFirst();

		String sname = cursor.getString(cursor.getColumnIndex("sname"));
		helper.setEditTextString(R.id.editText1,sname);
		String furi = cursor.getString(cursor.getColumnIndex("furi"));
		helper.setEditTextString(R.id.editText2,furi);

		cursor.close();

	}

}




posted by lightbox at 2017-09-23 17:38 | Comment(0) | 2017 Android Studio | このブログの読者になる | 更新情報をチェックする
container 終わり

フリーフォントで簡単ロゴ作成
フリーフォントでボタン素材作成
フリーフォントで吹き出し画像作成
フリーフォントではんこ画像作成
ほぼ自由に利用できるフリーフォント
フリーフォントの書体見本とサンプル
画像を大きく見る為のウインドウを開くボタンの作成

Android SDK ポケットリファレンス
改訂版 Webデザイナーのための jQuery入門
今すぐ使えるかんたん ホームページ HTML&CSS入門
CSS ドロップシャドウの参考デモ
Google Hosted Libraries
cdnjs
BUTTONS (CSS でボタン)
イラストAC
ぱくたそ
写真素材 足成
フリーフォント一覧
utf8 文字ツール
右サイド 終わり
base 終わり