利用API发送和接收消息,本代码测试全通过

本文介绍了一个程序员如何通过编程实现淘宝店的自动化操作,包括发送消息、获取消息、控件赋值和点击控件等关键步骤。

~_~一个程序员的淘宝店:点击打开链接

using System;
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;

namespace Set_Get_Messages
{
class APISendMsg
{
#region 常数
const int WM_COPYDATA = 0x004A;//发送消息到窗口
const int WM_SETTEXT = 0xC;//发送消息到控件
const int WM_CLOSE = 0x0010;//关闭窗口
const int BM_CLICK = 0xF5;//点击窗体
public const int WM_GETTEXT = 0x000D;//得到消息
#endregion
#region API函数
//查找窗体
[DllImport("user32.dll", EntryPoint = "FindWindow", CharSet = CharSet.Auto)]
public static extern IntPtr FindWindow(String classname, String text);
//查找子窗体
[DllImport("User32.dll", EntryPoint = "FindWindowEx", CharSet = CharSet.Auto)]
public static extern IntPtr FindWindowEx(IntPtr hwndParent, IntPtr hwndChildAfter, string lpClassName, string lpWindowName);

/// <summary>
/// 将指定的消息发送到一个或多个窗口
/// </summary>
/// <param name="hwnd">目标窗口句柄</param>
/// <param name="msg">消息类型</param>
/// <param name="hwndFrom">返回的消息</param>
/// <param name="cd">发送的消息</param>
/// <returns></returns>
[DllImport("user32.dll", EntryPoint = "SendMessage")]
public static extern long SendMessage(IntPtr hwnd, Int32 msg, Int32 hwndFrom, ref COPYDATASTRUCT cd);
[DllImport("user32.dll", EntryPoint = "SendMessage")]
public static extern int SendMessage(IntPtr hwnd, int wMsg, int wParam, int lParam);
/// <summary>
/// 将指定的消息发送到一个或多个窗体
/// </summary>
/// <param name="hWnd">窗体句柄</param>
/// <param name="Msg">消息类别</param>
/// <param name="wParam">返回消息</param>
/// <param name="lParam">发送的消息</param>
/// <returns></returns>
[DllImport("user32.dll", EntryPoint = "SendMessage")]
public static extern long SendMessage(IntPtr hWnd, Int32 Msg, Int32 wParam, [MarshalAs(UnmanagedType.LPStr)] string lParam);

//获取消息
[DllImport("user32.dll", EntryPoint = "SendMessage")]
public static extern bool SendMessageSetText(IntPtr hwnd, uint wMsg, uint wParam, string lParam);
[DllImport("user32.dll", EntryPoint = "SendMessage")]
public static extern int SendMessageGetText(IntPtr hwnd, uint wMsg, uint wParam, StringBuilder lParam);

#endregion
#region 发送消息
//发送消息结构体
[StructLayout(LayoutKind.Sequential)]
public struct COPYDATASTRUCT
{
public IntPtr dwData;//目标
public int cbData;//数据长度
[MarshalAs(UnmanagedType.LPStr)]
public string lpData;//消息
}

/// <summary>
/// 发送消息
/// </summary>
/// <param name="strMsg">消息</param>
/// <returns>成功非0,失败0</returns>
public static long API_sendMsgWindows(IntPtr windowHwnd, string strMsg)
{
long longResult = 0;
if (windowHwnd != IntPtr.Zero && !string.IsNullOrEmpty(strMsg.Replace(" ", "")))
{
//定义c,是为了取得字符串中一些非英文字符的长度,比如一个中文,占两个字长。
int longMsg = StrLength(strMsg);
Int32 id = 1;
COPYDATASTRUCT cd = new COPYDATASTRUCT();
cd.dwData = (IntPtr)id;
cd.lpData = strMsg + "/$";//一定要加上结束标志,不加的话会得到不确定的字符。
cd.cbData = strMsg.Length + longMsg + 2;//另外加上双字节的东西和结束符的长度
//发送消息
longResult = SendMessage(windowHwnd, WM_COPYDATA, 0, ref cd);
}
return longResult;
}

/// <summary>
/// 字符串长度(按字节算)
/// </summary>
/// <param name="str"></param>
/// <returns>字符串长度</returns>
private static int StrLength(string str)
{
int len = 0;
byte[] b;
if (!string.IsNullOrEmpty (str.Replace(" ","")))
{
for (int i = 0; i < str.Length; i++)
{
b = Encoding.Default.GetBytes(str.Substring(i, 1));//每个字
if (b.Length > 1)
len += 2;
else
len++;
}
}

return len;
}
#endregion
#region 得到窗体消息
/// <summary>
/// 取得指定句柄的窗口的Caption字符串
/// </summary>
/// <param name="hWnd"></param>
/// <param name="strWndText"></param>
/// <returns></returns>
public static int GetWindowControlTextX(IntPtr hWnd, out string strWndText)
{
/*
StringBuffer sLimitedString;
int nLen = GetWindowText(hWnd, out sLimitedString, 256);
if (nLen > 0)
strWndText = sLimitedString.szText;
else
strWndText = @"";

return nLen;
*/
//由于针对不同的进程的窗体控件,不能使用GetWindowText函数,只能使用WM_GETTEXT
StringBuilder strBuffer = new StringBuilder(1024);
int nLen = SendMessageGetText(hWnd,WM_GETTEXT, 1024, strBuffer);
strWndText = strBuffer.ToString();
return nLen;
}
#endregion
#region 共用方法
/// <summary>
/// 为控件赋值
/// </summary>
/// <param name="childControID">控件句柄</param>
/// <param name="strMsg">消息</param>
/// <returns>成功非0,失败0</returns>
public static long API_SetTextValue(IntPtr childControID, string strMsg)
{
long longResult = 0;
if (childControID != IntPtr.Zero && !string.IsNullOrEmpty(strMsg.Replace(" ", "")))
{
longResult = SendMessage(childControID, WM_SETTEXT, 0, strMsg);
}
return longResult;
}
/// <summary>
/// 点击控件
/// </summary>
/// <param name="childControID">按钮控件句柄</param>
/// <returns>成功非0,失败0</returns>
public static long API_ClickButton(IntPtr childControID)
{
long longResult = 0;
if (childControID != IntPtr.Zero)
{
longResult = SendMessage(childControID, BM_CLICK, 0, 0);
}
return longResult;
}

#endregion
}
}

//补

//另一种比较稳定的方法接收窗体中的代码

//自动接收消息
[StructLayout(LayoutKind.Sequential)]
public struct COPYDATASTRUCT
{
public IntPtr dwData;
public int cbData;
public IntPtr lpData;
}

protected override void DefWndProc(ref Message m)
{
try
{
switch (m.Msg)
{
//接收CopyData消息,读取发送过来的数据
case WM_COPYDATA:
COPYDATASTRUCT cds = new COPYDATASTRUCT();
Type mytype = cds.GetType();
cds = (COPYDATASTRUCT)m.GetLParam(mytype);
uint flag = (uint)(cds.dwData);
byte[] bt = new byte[cds.cbData];
Marshal.Copy(cds.lpData, bt, 0, bt.Length);
if (cds.cbData > 0)
{
string tmp = System.Text.Encoding.Default.GetString(bt);
if (!string.IsNullOrEmpty(tmp))
{
//交易方法
getRunPlan(tmp);
}
}
// MessageBox.Show(tmp);
break;
default:
base.DefWndProc(ref m);
break;
}
}
catch (Exception ex)
{
XtraMessageBox.Show(ex.ToString(), "Error", MessageBoxButtons.OK, MessageBoxIcon.Error);
// throw ex;
}
}

~_~一个程序员的淘宝店:点击打开链接

JS API 接口接收与返回 是一个轻型的、安的、跨网际的、跨语言的、跨平台的、跨环境的、跨域的、支持复杂对象传输的、支持引用参数传递的、支持内容输出重定向的、支持分级错误处理的、支持会话的、面向服务的高性能远程过程调用协议。 该版本直接解压后就可以使用,其中 属于公共文件。不论是客户端还是服务器端都需要这些文件。 是客户端文件,如果你只需要使用客户端,那么只要有上面那些公共文件这个文件就可以使用了,使用时,直接在你的程序中包含 phprpc_client.php 就可以,公共文件不需要单独包含。 这三个文件是服务器端需要的文件。 其中 dhparams 目录中包含的是加密传输时用来生成密钥的参数 dhparams.php 是用来读取 dhparams 目录中文件的类。 phprpc_server.php 是服务器端,如果你要使用 PHP 来发布 PHPRPC 服务,只需要包含这个文件就可以了。公共文件 dhparams.php 都不需要单独包含。 PHP 4.3+、PHP 5、PHP 6 客户端要求开启 socket 扩展。 服务器端需要有 IIS、Apache、lighttpd 等可以运行 PHP 程序的 Web 服务器。 如果服务器端需要加密传输的能力,必须要保证 session 配置正确。 <?php include('php/phprpc_server.php'); //加载文件 function hello($name) { return'Hello ' . $name; } $server = new PHPRPC_Server(); //创建服务端 $server->add(array('hello', 'md5', 'sha1')); //数组形式一次注册多个函数 $server->add('trim'); //单一注册 $server->start(); //开启服务 ?> <?php include ("php/phprpc_client.php"); //加载文件 $client = new PHPRPC_Client('http://127.0.0.1/server.php'); //创建客户端 并连接服务端文件 echo$client->Hello("word"); //调用方法 返回 hello word ?> -------------------------------------------------- --------------------------------------------------- ------------------------------ 服务端其他说明: <?php include('php/phprpc_server.php'); //加载文件 function hello($name) { return'Hello ' . $name; } class Example1 { staticfunction foo() { return'foo'; } function bar() { return'bar'; } } $server = new PHPRPC_Server(); //创建服务端 $server->add('foo', 'Example1'); //静态方法直接调用 $server->add('bar', new Example1()); //非静态方法 需要实例化 //注册别名调用 $server->add('hello', NULL, 'hi'); //第三参数是函数的别名 客户端通过别名来调用函数 $server->add('foo', 'Example1', 'ex1_foo'); $server->add('bar', new Example1(), 'ex1_bar'); $server->setCharset('UTF-8'); //设置编码 $server->setDebugMode(true); //打印错误 $server->setEnableGZIP(true); //启动压缩输出虽然可以让传输的数据量减少,但是它会占用更多的内存 CPU,因此它默认是关闭的。 $server->start(); //开启服务 ?> -------------------------------------------------- --------------------------------------------------- --------------------------- 客户端其他说明: <?php include ("php/phprpc_client.php"); $client = new PHPRPC_Client(); $client->useService('http://127.0.0.1/server.php'); //远程调用地址 $client->setKeyLength(1000); //密钥长度 $client->setEncryptMode(3); //加密等级0-3 $client->setCharset('UTF-8'); //设置编码 $client->setTimeout(10); //设置超时时间 echo$client->hi('PHPRPC'), "\r\n"; //调用函数 echo$client->getKeyLength(), "\r\n"; //下面是返回值 echo$client->getEncryptMode(), "\r\n"; echo$client->getCharset(), "\r\n"; echo$client->getTimeout(), "\r\n"; ?> -------------------------------------------------- --------------------------------------------------- ---------------------- 关于session <?php include('php/phprpc_server.php'); class ExampleCounter { function ExampleCounter() { if (!isset($_SESSION['count'])) { $_SESSION['count'] = 0; } } function inc() { $_SESSION['count'] += 1; } functioncount() { return$_SESSION['count']; } } $server = new PHPRPC_Server(); $server->add(array('inc', 'count'), new ExampleCounter()); $server->start(); ?> <?php include("php/phprpc_client.php"); $client = newPHPRPC_Client(); $client->useService('http://127.0.0.1/1.php'); $client->setTimeout(10); echo $client->inc(); echo $client->count(); echo $client->inc(); echo $client->count(); ?> 每次刷新都是新建的client 服务端并不能识别.
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值