質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
Visual Studio

Microsoft Visual StudioはMicrosoftによる統合開発環境(IDE)です。多種多様なプログラミング言語に対応しています。

VB.NET

Microsoft Visual Basic .NETのことで、Microsoft Visual Basic(VB6)の後継。 .NET環境向けのプログラムを開発することができます。 現在のVB.NETでは、.NET Frameworkを利用して開発を行うことが可能です。

Q&A

解決済

5回答

10362閲覧

[VB.NET]Excelのタスクの消し方

reeeo

総合スコア42

Visual Studio

Microsoft Visual StudioはMicrosoftによる統合開発環境(IDE)です。多種多様なプログラミング言語に対応しています。

VB.NET

Microsoft Visual Basic .NETのことで、Microsoft Visual Basic(VB6)の後継。 .NET環境向けのプログラムを開発することができます。 現在のVB.NETでは、.NET Frameworkを利用して開発を行うことが可能です。

0グッド

3クリップ

投稿2017/08/09 05:25

visual studio2012 でwindowsフォームアプリを作成します。
データグリッドビューのデータをExcelに出力するようにしています。
ですが、Excelを閉じてもタスクが残ってしまいます。
調べながら色々と試してみましたが、全然だめでした。
ソースは以下の通りです。

'出力ボタン Private Sub Button1_Click(sender As Object, e As EventArgs) Handles Button1.Click Dim app As Excel.Application = Nothing Dim book As Excel.Workbook = Nothing Dim sheet As Excel.Worksheet = Nothing Dim books As Excel.Workbooks = Nothing Try 'Excelを開く app = New Excel.Application() 'Bookを開く books = app.Workbooks book = books.Open("C:\form3\Book1.xlsx") 'シートの指定 sheet = book.Worksheets(1) 'Excelに出力する For i As Integer = 0 To DataGridView1.Rows.Count - 1 sheet.Range("A" & i + 2 & "").Value = DataGridView1.Rows(i).Cells(1).Value sheet.Range("B" & i + 2 & "").Value = DataGridView1.Rows(i).Cells(4).Value sheet.Range("C" & i + 2 & "").Value = DataGridView1.Rows(i).Cells(5).Value sheet.Range("D" & i + 2 & "").Value = DataGridView1.Rows(i).Cells(6).Value sheet.Range("E" & i + 2 & "").Value = DataGridView1.Rows(i).Cells(7).Value Next 'Excelの表示 app.Visible = True '5秒だけ表示する System.Threading.Thread.Sleep(5000) '保存する book.Save() '閉じる app.Quit() Catch ex As Exception Throw ex Finally Marshal.ReleaseComObject(sheet) Marshal.ReleaseComObject(books) Marshal.ReleaseComObject(book) Marshal.ReleaseComObject(app) End Try End Sub

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

sazi

2017/08/09 13:59

「Excelを閉じてもタスクが残ってしまいます。」 とありますが、ここでいう「タスク」って何ですか?
guest

回答5

0

ベストアンサー

私はC#の方を書く事が多いので読み替えてもらえると助かるのですが。。。
以下のように、エクセルを開く場合以下のようにTry構文の階層を作って確実にリリースすることをお勧めします。

C#

1Microsoft.Office.Interop.Excel.Application(); 2try 3{ 4 Workbooks workbooks = excelApplication.Workbooks; 5 try 6 { 7 Workbook workbook = workbooks.Open("d:\\Book1.xlsx"); 8 try 9 { 10 Sheets worksheets = workbook.Sheets; 11 try 12 { 13 Worksheet worksheet = worksheets[1]; 14 try 15 { 16 処理 17 } 18 finally { Marshal.ReleaseComObject(worksheet); } 19 } 20 finally { Marshal.ReleaseComObject(worksheets); } 21 } 22 finally 23 { 24 if (workbook != null) 25 { 26 workbook.Close(false); 27 } 28 Marshal.ReleaseComObject(workbook); 29 } 30 } 31 finally { Marshal.ReleaseComObject(workbooks); } 32} 33finally 34{ 35 if (excelApplication != null) 36 { 37 excelApplication.Quit(); 38 } 39 Marshal.ReleaseComObject(excelApplication); 40} 41

投稿2017/08/09 06:50

Yuki_S

総合スコア356

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

reeeo

2017/08/09 07:21

ソースまでありがとうございます。 読み替えて試してみたところ sheet = DirectCast(sheets(1), Excel.Worksheet)の箇所で 型 'System.__ComObject' の COM オブジェクトをインターフェイス型 'Microsoft.Office.Interop.Excel.Worksheets' にキャストできません。とエラーが出てしまいました。
Yuki_S

2017/08/09 07:35

一番最初の Dimの所のNothing消したらそのエラー消えないですかね?
reeeo

2017/08/09 07:39

Nothingを消しても同じでした。。
Yuki_S

2017/08/10 00:20

ちょっと考えます。他の原因思いついたらコメントさせていただきます!
reeeo

2017/08/10 01:00

申し訳ございません。上記のエラーはデータ型が間違っていた為起きたエラーでした。 現状、for文をコメントアウトして動かしたところタスクは残らず正常に終了できていました。あと解放するのはrangeだけでよろしいのでしょうか?
Yuki_S

2017/08/10 01:13

上記のFor内のrangeは値にアクセスするだけなので解放は恐らく不要かと思います。 うまくいかない様なら入れてあげてください。
reeeo

2017/08/12 14:33

無事できました。また何かありましたらお願いします。
guest

0

WorksheetsとRangeの解放が必要ですね。以下サンプルコードです。

VB.NET

1'参照設定はCOM(Microsoft Excel 16.0 Object Library)ではなく 2'.NETアセンブリ(Microsoft.Office.Interop.Excel)を使用。 3 4Imports Microsoft.Office.Interop 5 6Public Class Form1 7 8 '自分で解放版 9 Private Sub Button1_Click(sender As Object, e As EventArgs) Handles Button1.Click 10 Using coms As New ComStack 11 Dim app As New Excel.Application : coms.Push(app) 12 Dim books = app.Workbooks : coms.Push(books) 13 Dim book = books.Open("C:\Users\(user)\Documents\Book1.xlsx") : coms.Push(book) 14 Dim sheets = book.Worksheets : coms.Push(sheets) 15 Dim sheet = DirectCast(sheets(1), Excel.Worksheet) : coms.Push(sheet) 16 Dim range = sheet.Range("A1") : coms.Push(range) : range.Value = "abc" 17 range = sheet.Range("B2") : coms.Push(range) : range.Value = "def" 18 app.Visible = True 19 System.Threading.Thread.Sleep(5000) 20 book.Save() 21 app.Quit() 22 End Using 23 End Sub 24 25 'GCで解放版 26 Private Sub Button2_Click(sender As Object, e As EventArgs) Handles Button2.Click 27 DoExcelTask() 28 GC.Collect() 29 'メモ:Excel操作とGC.Collectは別プロシージャに書く。 30 End Sub 31 32 Private Sub DoExcelTask() 33 Dim app As New Excel.Application 34 Dim book = app.Workbooks.Open("C:\Users\(user)\Documents\Book1.xlsx") 35 Dim sheet = DirectCast(book.Worksheets(1), Excel.Worksheet) 36 sheet.Range("A1").Value = "abc" 37 sheet.Range("B2").Value = "def" 38 app.Visible = True 39 System.Threading.Thread.Sleep(5000) 40 book.Save() 41 app.Quit() 42 End Sub 43 44End Class 45 46'COM解放用ヘルパークラス 47Class ComStack 48 Inherits Stack(Of Object) 49 Implements IDisposable 50 51 Sub Dispose() Implements IDisposable.Dispose 52 Do Until Count = 0 53 System.Runtime.InteropServices.Marshal.ReleaseComObject(Pop()) 54 Loop 55 GC.SuppressFinalize(Me) 56 End Sub 57End Class

投稿2017/08/11 05:31

退会済みユーザー

退会済みユーザー

総合スコア0

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

'シートの指定

sheet = book.Worksheets(1)

Worksheets はメソッドではなくプロパティなので、変数で受けて開放する必要がありますね。

投稿2017/08/09 06:47

hihijiji

総合スコア4150

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

reeeo

2017/08/09 07:22

回答ありがとうございます。 変数で sheet = DirectCast(sheets(1), Excel.Worksheet)としたところ 型 'System.__ComObject' の COM オブジェクトをインターフェイス型 'Microsoft.Office.Interop.Excel.Worksheets' にキャストできません。とエラーが出てしまいました。
hihijiji

2017/08/09 07:31

こんな感じで Dim sheets As Sheets sheets = book.Worksheets sheet = sheets(1) 代入して sheets も開放してください。
reeeo

2017/08/09 07:41

Dim sheets As Excel.Worksheets = Nothing sheets = book.Worksheets sheet = sheets(1) . . Marshal.ReleaseComObject(sheets) とやってもだめでした。 Nothingは解放の箇所で波線が引かれてしまうので書きました。
hihijiji

2017/08/09 08:09

Range オブジェクトも同様に解放する必要があるのかも知れませんね。 また、セキュリティソフトなどでリアルタイム監視している場合は、 book.Save() と app.Quit() の間にも適当にSleepをいれると解放されたなどの話を聞いたことがあります。
reeeo

2017/08/09 08:12

とりあえずはRangeを使用している箇所はコメントアウトして実行しているのですが、使用していなくても解放しなくてはいけないのですか??
hihijiji

2017/08/09 08:15

使ってないものは解放できません。
guest

0

app→books→book→sheetの順でアクセスしているのでリリースするのはこの逆の順でリリースする必要があります。

パッと見た感じ

VB.net

1Marshal.ReleaseComObject(***)

のbookとbooksが逆になっているので順番を変えてみてください。
またttyp03さんのリンクにあるようにリリース前にチェックをしておいた方がいいです。

投稿2017/08/09 06:04

Yuki_S

総合スコア356

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

reeeo

2017/08/09 06:25

コメントありがとうございます。 順番を変えてもだめでした。 ちなみにsheetをコメントアウトして行ったところタスクはちゃんと閉じられていたのでsheetがおかしいんではないかと思いました。。。
Yuki_S

2017/08/09 06:43

状況があまりわかりませんが、一度アプリを閉じる前にbook.close()でブック自体を閉じる処理を入れてみたらどうなりますかね?
guest

0

環境がなくて試していませんが、処理の最後にオブジェクトを明示的に解放してみてはいかがでしょうか。

VBNET

1app = Nothing 2book = Nothing 3sheet = Nothing 4books = Nothing

下記リンク先に似たような話があります。
詳しくは読んでないですし、結構古い話なので今とは違うかもしれませんが、参考まで。
https://qa.atmarkit.co.jp/q/1227

投稿2017/08/09 05:43

ttyp03

総合スコア16998

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

reeeo

2017/08/09 06:25

コメントありがとうございます。 すいません。リンク先を読んでもあまりよく理解できませんでした。 ちなみにsheetをコメントアウトして行ったところタスクはちゃんと閉じられていたのでsheetがおかしいんではないかと思いました。。。
ttyp03

2017/08/09 06:28

リンク先は回答のネタ元にした話なので、参考程度に見てもらえれば良いです。 要は全部のオブジェクトを解放しないとダメなんじゃないの、ってことを言いたいだけです。 Yuki_Sさんの回答にあるように、解放の順番は考慮しないとダメかもしれないですね。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問